Jak sprawdzić, czy ciąg zawiera inny ciąg w Objective-C?


1212

Jak mogę sprawdzić, czy ciąg ( NSString) zawiera inny mniejszy ciąg?

Miałem nadzieję na coś takiego:

NSString *string = @"hello bla bla";
NSLog(@"%d",[string containsSubstring:@"hello"]);

Ale najbliższe, jakie mogłem znaleźć, to:

if ([string rangeOfString:@"hello"] == 0) {
    NSLog(@"sub string doesnt exist");
} 
else {
    NSLog(@"exists");
}

W każdym razie, czy to najlepszy sposób na sprawdzenie, czy ciąg zawiera inny ciąg?


1
Chciałbym również, aby został dodany, ale tymczasem stosunkowo łatwo jest dodać go jako kategorię w NSString.
isaac

2
Przy użyciu if ([string rangeOfString:@"hello"] == 0) {...}występuje błąd niedopasowania typu dla NSRange i int. aby to naprawić, należy zmienić wiersz na następujący:if ([string rangeOfString:@"hello"].length == 0) {...}
Neeku,

1
iOS 8 dodaje zawiera ciąg: a tutaj jest minimalnie inwazyjny sposób na dodanie obsługi iOS 7 petersteinberger.com/blog/2014/…
Steve Moser

2
Jestem programistą iOS od samego początku i ciągle odwiedzam ten post, aby szybko wkleić kopię. Nie mogę zapamiętać tego. Najczęściej odwiedzany post stackoverflow w mojej historii.
VaporwareWolf

wystarczy użyć Swift;)
Fattie

Odpowiedzi:


2436
NSString *string = @"hello bla bla";
if ([string rangeOfString:@"bla"].location == NSNotFound) {
  NSLog(@"string does not contain bla");
} else {
  NSLog(@"string contains bla!");
}

Kluczem jest zauważenie, że rangeOfString:zwraca NSRangestrukturę, a dokumentacja mówi , że zwraca strukturę, {NSNotFound, 0}jeśli „stóg siana” nie zawiera „igły”.


A jeśli korzystasz z systemu iOS 8 lub OS X Yosemite, możesz teraz: (* UWAGA: spowoduje to awarię aplikacji, jeśli ten kod zostanie wywołany na urządzeniu z systemem iOS7).

NSString *string = @"hello bla blah";
if ([string containsString:@"bla"]) {
  NSLog(@"string contains bla!");
} else {
  NSLog(@"string does not contain bla");
}

(Tak też by to działało w Swift)

👍


286
Aby rozróżnić wielkość liter, użyj wyszukiwania „if ([ciąg rangeOfString: @„ bla ”opcje: NSCaseInsensitiveSearch] .location! = NSNotFound)”
Vanja

1
@Dave DeLong Chciałem tylko wspomnieć o kategorii, którą stworzyłem w tym celu, zanim przeczytałem Twoją odpowiedź na odpowiedź! Ponieważ jestem przede wszystkim programistą ac #, cieszę się, że dodali metodę NSSTring.
dherrin79

Dlaczego kompilator nic nie mówi, jeśli moim celem wdrażania jest system iOS7 i korzystam z metody includeString?
Ricardo

3
I jeszcze dalej: @Vanja: jeśli zamierzasz użyć kodu skrótu [string zawieraString], który został wprowadzony w iOS 8 / Yosemite, możesz użyć następującego kodu dla łańcucha bez rozróżniania wielkości liter: "[stringToSearch localizedCaseInsensitiveContainsString: string ] ”, a ten, jeśli chcesz przeprowadzić wyszukiwanie niewrażliwe na wielkość liter i znaki diakrytyczne:„ [stringToSearch localizedStandardContainsString: string] ”.
Scott Kohlert

1
Zauważ, że wyrażenie [string rangeOfString:@"bla"].location != NSNotFoundbędzie prawdziwe, gdy ciąg znaków jest nil!
funroll

158

W systemie iOS 8.0+ i macOS 10.10+ można używać natywnego narzędzia NSString containsString:.

W przypadku starszych wersji iOS i macOS możesz utworzyć własną (przestarzałą) kategorię dla NSString:

@interface NSString ( SubstringSearch )
    - (BOOL)containsString:(NSString *)substring;
@end

// - - - - 

@implementation NSString ( SubstringSearch )

- (BOOL)containsString:(NSString *)substring
{    
    NSRange range = [self rangeOfString : substring];
    BOOL found = ( range.location != NSNotFound );
    return found;
}

@end

Uwaga: zwróć uwagę na komentarz Daniela Galasko dotyczący nazewnictwa


13
+1 za wyraźniejszy wynikowy kod i możliwość ponownego użycia. Zamieniłem go w jedną linijkę return [self rangeOfString:substring].location != NSNotFound;i umieściłem w mojej bibliotece refaktoryzacji, es_ios_utils. github.com/peterdeweese/es_ios_utils
Peter DeWeese

4
Wygląda na to, że Apple polubił twój pomysł i dodał tę funkcję w iOS 8 i OSx 10.10 (Yosemite), jak wspomniano w jego odpowiedzi na @DaveDeLong. +1
Islam Pytanie

7
Podstawową zasadą dla kategorii obj-c jest przedrostek nazwy metody trzyliterowym prefiksem modułu. To doskonały przykład, ponieważ jest on teraz w konflikcie z wersją iOS 7 i 10.10
Daniel Galasko

54

Ponieważ wydaje się, że jest to wysoki wynik w Google, chcę dodać:

iOS 8 i OS X 10.10 dodają tę containsString:metodę do NSString. Zaktualizowana wersja przykładu Dave'a DeLonga dla tych systemów:

NSString *string = @"hello bla bla";
if ([string containsString:@"bla"]) {
    NSLog(@"string contains bla!");
} else {
    NSLog(@"string does not contain bla");
}

39
NSString *myString = @"hello bla bla";
NSRange rangeValue = [myString rangeOfString:@"hello" options:NSCaseInsensitiveSearch];

if (rangeValue.length > 0)
{
    NSLog(@"string contains hello");
} 
else 
{
    NSLog(@"string does not contain hello!");
}

// Możesz również użyć następujących opcji:

if (rangeValue.location == NSNotFound) 
{
    NSLog(@"string does not contain hello");
} 
else 
{
    NSLog(@"string contains hello!");
}

22

W iOS 8 i Swift możemy użyć localizedCaseInsensitiveContainsString metody

 let string: NSString = "Café"
 let substring: NSString = "É"

 string.localizedCaseInsensitiveContainsString(substring) // true

1
To jest dobre. Nie mam pojęcia, dlaczego nie mieli tej metody dla iOS 7
Lucas

@Lucas, ponieważ Swift został uruchomiony z iOS 8.0. ale dzięki swift nadal możesz obsługiwać urządzenia z iOS 7.
Hemang

13

Więc osobiście naprawdę nienawidzę, NSNotFoundale rozumiem jego konieczność.

Ale niektórzy ludzie mogą nie rozumieć złożoności porównywania z NSNotFound

Na przykład ten kod:

- (BOOL)doesString:(NSString*)string containString:(NSString*)otherString {
    if([string rangeOfString:otherString].location != NSNotFound)
        return YES;
    else
        return NO;
}

ma swoje problemy:

1) Oczywiście, jeśli otherString = nilten kod ulegnie awarii. prosty test to:

NSLog(@"does string contain string - %@", [self doesString:@"hey" containString:nil] ? @"YES": @"NO");

prowadzi do !! CRASH !!

2) Dla kogoś nowego w celu-c nie jest to tak oczywiste, że ten sam kod NIE ulega awarii, gdy string = nil. Na przykład ten kod:

NSLog(@"does string contain string - %@", [self doesString:nil containString:@"hey"] ? @"YES": @"NO");

i ten kod:

NSLog(@"does string contain string - %@", [self doesString:nil containString:nil] ? @"YES": @"NO");

spowoduje oba

does string contains string - YES

Które wyraźnie NIE jest tym, czego chcesz.

Lepszym rozwiązaniem, które moim zdaniem działa, jest wykorzystanie faktu, że rangeOfString zwraca długość 0, więc lepszym, bardziej niezawodnym kodem jest:

- (BOOL)doesString:(NSString*)string containString:(NSString*)otherString {
    if(otherString && [string rangeOfString:otherString].length)
        return YES;
    else
        return NO;
}

LUB PO PROSTU:

- (BOOL)doesString:(NSString*)string containString:(NSString*)otherString {
    return (otherString && [string rangeOfString:otherString].length);
}

które w przypadku 1 i 2 powrócą

does string contains string - NO

To moje 2 centy ;-)

Proszę sprawdzić moją Gist, aby uzyskać bardziej pomocny kod.


12

Ulepszona wersja P i jest rozwiązaniem , kategoria na NSString, który nie tylko pokaże, jeśli ciąg znajduje się wewnątrz innego łańcucha, ale również wykonuje szereg odnośnik, to:

@interface NSString (Contains)
-(BOOL)containsString: (NSString*)substring
              atRange:(NSRange*)range;

-(BOOL)containsString:(NSString *)substring;
@end

@implementation NSString (Contains)

-(BOOL)containsString:(NSString *)substring
              atRange:(NSRange *)range{

    NSRange r = [self rangeOfString : substring];
    BOOL found = ( r.location != NSNotFound );
    if (range != NULL) *range = r;
    return found;
}

-(BOOL)containsString:(NSString *)substring
{
    return [self containsString:substring
                        atRange:NULL];
}

@end

Użyj tego jak:

NSString *string = @"Hello, World!";

//If you only want to ensure a string contains a certain substring
if ([string containsString:@"ello" atRange:NULL]) {
    NSLog(@"YES");
}

// Or simply
if ([string containsString:@"ello"]) {
    NSLog(@"YES");
}

//If you also want to know substring's range
NSRange range;
if ([string containsString:@"ello" atRange:&range]) {
    NSLog(@"%@", NSStringFromRange(range));
}

8

Oto fragment funkcji kopiuj i wklej:

-(BOOL)Contains:(NSString *)StrSearchTerm on:(NSString *)StrText
{
    return [StrText rangeOfString:StrSearchTerm 
        options:NSCaseInsensitiveSearch].location != NSNotFound;
}

Czy mogę wiedzieć, dlaczego jestem przegłosowany? To działający fragment kodu
Durai Amuthan.H

6

Oneliner (mniejsza ilość kodu. OSUSZANIE, ponieważ masz tylko jeden NSLog):

NSString *string = @"hello bla bla";
NSLog(@"String %@", ([string rangeOfString:@"bla"].location == NSNotFound) ? @"not found" : @"cotains bla"); 

6
NSString *categoryString = @"Holiday Event";
if([categoryString rangeOfString:@"Holiday"].location == NSNotFound)
{
    //categoryString does not contains Holiday
}
else
{
    //categoryString contains Holiday
}

5

Spróbuj tego,

NSString *string = @"test Data";
if ([[string lowercaseString] rangeOfString:@"data"].location == NSNotFound) 
{
    NSLog(@"string does not contain Data");
}   
else 
{
    NSLog(@"string contains data!");
}

5

Najlepsze rozwiązanie Tak proste jak to! Jeśli chcesz znaleźć słowo lub część ciągu. Możesz użyć tego kodu. W tym przykładzie sprawdzimy, czy wartość słowa zawiera „aktor”.

NSString *word =@"find a word or character here";
if ([word containsString:@"acter"]){
    NSLog(@"It contains acter");
} else {
     NSLog (@"It does not contain acter");
}


4

Jeśli potrzebujesz tego raz, napisz:

NSString *stringToSearchThrough = @"-rangeOfString method finds and returns the range of the first occurrence of a given string within the receiver.";
BOOL contains = [stringToSearchThrough rangeOfString:@"occurence of a given string"].location != NSNotFound;

4

W przypadku szybkiego można tego użyć

let string = "Package #23"
if string.containsString("Package #") {
    //String contains substring
}
else {
    //String does not contain substring
}

2

Jeśli nie zawracaj sobie głowy rozróżnianiem wielkości liter. Spróbuj tego raz.

NSString *string  = @"Hello World!";

if([string rangeOfString:@"hello" options:NSCaseInsensitiveSearch].location !=NSNotFound)
{
    NSLog(@"found");
}
else
{
    NSLog(@"not found");
}

1

Proszę użyć tego kodu

NSString *string = @"hello bla bla";
if ([string rangeOfString:@"bla"].location == NSNotFound) 
{
    NSLog(@"string does not contain bla");
} 
else 
{  
    NSLog(@"string contains bla!");
}

1

Użyj opcji NSCaseInsensitiveSearch z rangeOfString: opcje:

NSString *me = @"toBe" ;
NSString *target = @"abcdetobe" ;
NSRange range = [target  rangeOfString: me options: NSCaseInsensitiveSearch];
NSLog(@"found: %@", (range.location != NSNotFound) ? @"Yes" : @"No");
if (range.location != NSNotFound) {
// your code
}

Znaleziono wynik wyjściowy: Tak

Opcje mogą być „lub” połączone i obejmują:

NSCaseInsensitiveSearch NSLiteralSearch NSBackwardsSearch i więcej


0

Pierwszy ciąg zawiera lub nie drugi ciąg,

NSString *first = @"Banana";
NSString *second = @"BananaMilk";
NSRange range = [first rangeOfString:second options:NSCaseInsensitiveSearch];

if (range.length > 0) {
    NSLog(@"Detected");
}
else {
    NSLog(@"Not detected");
}

0

Spróbuj tego:

Swift 4.1, 4.2:

let stringData = "Black board"

//swift quick way and case sensitive
if stringData.contains("bla") {
    print("data contains string");
}

//case sensitive
if stringData.range(of: "bla",options: .caseInsensitive) != nil {
    print("data contains string");
}else {
    print("data does not contains string");
}

Dla celu C:

NSString *stringData = @"Black board";

//Quick way and case sensitive
if ([stringData containsString:@"bla"]) {
    NSLog(@"data contains string");
}

//Case Insensitive
if ([stringData rangeOfString:@"bla" options:NSCaseInsensitiveSearch].location != NSNotFound) {
   NSLog(@"data contains string");
}else {
   NSLog(@"data does not contain string");
}

0

Swift 4 i wyżej

let str = "Hello iam midhun"

if str.contains("iam") {
  //contains substring
}
else {
  //doesn't contain substring
}

Cel C

NSString *stringData = @"Hello iam midhun";

if ([stringData containsString:@"iam"]) {
    //contains substring
}
else {
    //doesn't contain substring
}

-1
-(Bool)checkIf:(String)parentString containsSubstring:(String)checkString {
    NSRange textRange =[parentString rangeOfString:checkString];
    return textRange.location != NSNotFound // returns true if parent string contains substring else returns false
  }

1
W Objective-C mamy typ „NSString”. Nie mamy Stringtypu. Po drugie, mamy wszystkie przedmioty. Jako argument powinieneś przekazać NSString. Głosuję za skreśleniem z powodu złej jakości odpowiedzi w ramach bardzo aktywnego pytania.
Aleksey Potapov,

-2

Jeśli potrzebna jest określona pozycja ciągu, ten kod pojawia się w Swift 3.0 :

let string = "This is my string"
let substring = "my"

let position = string.range(of: substring)?.lowerBound
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.