Швидкий запуск методом?


151

Чи існує такий метод, як метод startWith () чи щось подібне у Swift?

Я в основному намагаюся перевірити, чи починається певна рядок з іншої рядки. Я також хочу, щоб це було нечутливим до регістру.

Як ви могли б сказати, я просто намагаюся зробити просту функцію пошуку, але, здається, я в цьому жалію не вдається.

Це те, що я хотів би:

введення "sa" повинно дати мені результати для "San Antonio", "Santa Fe" і т.д.

Я використовував

self.rangeOfString(find, options: NSStringCompareOptions.CaseInsensitiveSearch) != nil 

до iOS9, і він працював чудово. Однак після оновлення до iOS9 він припинив свою роботу, і тепер пошуки залежать від регістру.

    var city = "San Antonio"
    var searchString = "san "
    if(city.rangeOfString(searchString, options: NSStringCompareOptions.CaseInsensitiveSearch) != nil){
        print("San Antonio starts with san ");
    }

    var myString = "Just a string with san within it"

    if(myString.rangeOfString(searchString, options: NSStringCompareOptions.CaseInsensitiveSearch) != nil){
        print("I don't want this string to print bc myString does not start with san ");
    }

Чи можете ви навести конкретний приклад, коли rangeOfString з CaseInsensitiveSearch не працює так, як очікувалося? Я протестував його в iOS 9 Simulator, і він працював на мене.
Мартін Р

Відповіді:


362

використовувати hasPrefixзамість startsWith.

Приклад:

"hello dolly".hasPrefix("hello")  // This will return true
"hello dolly".hasPrefix("abc")    // This will return false

2
ОП просить бути нечутливими до справ, і ваша відповідь є чутливою до регістру
Cœur

13
Дуже просто зробити рядки в малі перед порівнянням, використовуючи"string".lowercased()
TotoroTotoro

12

ось впровадження розширень Swift запускаєWith:

extension String {

  func startsWith(string: String) -> Bool {

    guard let range = rangeOfString(string, options:[.AnchoredSearch, .CaseInsensitiveSearch]) else {
      return false
    }

    return range.startIndex == startIndex
  }

}

Приклад використання:

var str = "Hello, playground"

let matches    = str.startsWith("hello") //true
let no_matches = str.startsWith("playground") //false

10

Щоб відповісти на конкретні регістри, нечутливі до префіксу:

у чистому Swift (рекомендується більшість часу)

extension String {
    func caseInsensitiveHasPrefix(_ prefix: String) -> Bool {
        return lowercased().hasPrefix(prefix.lowercased())
    }
}

або:

extension String {
    func caseInsensitiveHasPrefix(_ prefix: String) -> Bool {
        return lowercased().starts(with: prefix.lowercased())
    }
}

Примітка: для порожнього префіксу ""повернуться обидві реалізаціїtrue

використовуючи Foundation range(of:options:)

extension String {
    func caseInsensitiveHasPrefix(_ prefix: String) -> Bool {
        return range(of: prefix, options: [.anchored, .caseInsensitive]) != nil
    }
}

Примітка: для порожнього префікса ""він повернетьсяfalse

і бути некрасивим з регулярним виразом (я це бачив ...)

extension String {
    func caseInsensitiveHasPrefix(_ prefix: String) -> Bool {
        guard let expression = try? NSRegularExpression(pattern: "\(prefix)", options: [.caseInsensitive, .ignoreMetacharacters]) else {
            return false
        }
        return expression.firstMatch(in: self, options: .anchored, range: NSRange(location: 0, length: characters.count)) != nil
    }
}

Примітка: для порожнього префікса ""він повернетьсяfalse


6

Швидкий 4 func starts<PossiblePrefix>(with possiblePrefix: PossiblePrefix) -> Bool where PossiblePrefix : Sequence, String.Element == PossiblePrefix.Elementбуде представлений.

Приклад використання:

let a = 1...3
let b = 1...10

print(b.starts(with: a))
// Prints "true"

6

Редагування: оновлено для Swift 3.

Клас Swift String має метод hasPrefix(), що відрізняється від регістру , але якщо вам потрібен нечутливий до регістру пошук, ви можете використовувати метод NSString range(of:options:).

Примітка: За замовчуванням, методи NSString є НЕ доступні, але якщо ви import Foundationїх.

Так:

import Foundation
var city = "San Antonio"
var searchString = "san "
let range = city.range(of: searchString, options:.caseInsensitive)
if let range = range {
    print("San Antonio starts with san at \(range.startIndex)");
}

Параметри можуть бути задані як або .caseInsensitiveабо [.caseInsensitive]. Ви б використали другий, якщо хочете скористатися додатковими параметрами, такими як:

let range = city.range(of: searchString, options:[.caseInsensitive, .backwards])

Цей підхід також має перевагу в тому, що можна використовувати інші параметри пошуку, наприклад .diacriticInsensitiveпошук. Такого ж результату неможливо досягти просто використанням. lowercased() рядків.


1

Версія Swift 3:

func startsWith(string: String) -> Bool {
    guard let range = range(of: string, options:[.caseInsensitive]) else {
        return false
    }
    return range.lowerBound == startIndex
}

можна швидше .anchored. Дивіться мою відповідь або відповідь Олівера Аткінсона.
Cœur

1

У Swift 4 з розширеннями

Моє розгинання приклад містить 3 функції: перевірку зробити рядки почати з підрядком, зробити строковий кінець до підрядку і зробити Рядок містить підрядок.

Встановіть параметр isCaseSensitive на значення false, якщо ви хочете ігнорувати символи "A" або "a", інакше встановіть значення true.

Дивіться коментарі в коді для отримання додаткової інформації про те, як це працює.

Код:

    import Foundation

    extension String {
        // Returns true if the String starts with a substring matching to the prefix-parameter.
        // If isCaseSensitive-parameter is true, the function returns false,
        // if you search "sA" from "San Antonio", but if the isCaseSensitive-parameter is false,
        // the function returns true, if you search "sA" from "San Antonio"

        func hasPrefixCheck(prefix: String, isCaseSensitive: Bool) -> Bool {

            if isCaseSensitive == true {
                return self.hasPrefix(prefix)
            } else {
                var thePrefix: String = prefix, theString: String = self

                while thePrefix.count != 0 {
                    if theString.count == 0 { return false }
                    if theString.lowercased().first != thePrefix.lowercased().first { return false }
                    theString = String(theString.dropFirst())
                    thePrefix = String(thePrefix.dropFirst())
                }; return true
            }
        }
        // Returns true if the String ends with a substring matching to the prefix-parameter.
        // If isCaseSensitive-parameter is true, the function returns false,
        // if you search "Nio" from "San Antonio", but if the isCaseSensitive-parameter is false,
        // the function returns true, if you search "Nio" from "San Antonio"
        func hasSuffixCheck(suffix: String, isCaseSensitive: Bool) -> Bool {

            if isCaseSensitive == true {
                return self.hasSuffix(suffix)
            } else {
                var theSuffix: String = suffix, theString: String = self

                while theSuffix.count != 0 {
                    if theString.count == 0 { return false }
                    if theString.lowercased().last != theSuffix.lowercased().last { return false }
                    theString = String(theString.dropLast())
                    theSuffix = String(theSuffix.dropLast())
                }; return true
            }
        }
        // Returns true if the String contains a substring matching to the prefix-parameter.
        // If isCaseSensitive-parameter is true, the function returns false,
        // if you search "aN" from "San Antonio", but if the isCaseSensitive-parameter is false,
        // the function returns true, if you search "aN" from "San Antonio"
        func containsSubString(theSubString: String, isCaseSensitive: Bool) -> Bool {

            if isCaseSensitive == true {
                return self.range(of: theSubString) != nil
            } else {
                return self.range(of: theSubString, options: .caseInsensitive) != nil
            }
        }
    }

Приклади використання:

Щоб перевірити, чи починається рядок з "TEST":

    "testString123".hasPrefixCheck(prefix: "TEST", isCaseSensitive: true) // Returns false
    "testString123".hasPrefixCheck(prefix: "TEST", isCaseSensitive: false) // Returns true

Щоб перевірити, чи починається рядок з "тесту":

    "testString123".hasPrefixCheck(prefix: "test", isCaseSensitive: true) // Returns true
    "testString123".hasPrefixCheck(prefix: "test", isCaseSensitive: false) // Returns true

Для перевірки завершується рядок "G123":

    "testString123".hasSuffixCheck(suffix: "G123", isCaseSensitive: true) // Returns false
    "testString123".hasSuffixCheck(suffix: "G123", isCaseSensitive: false) // Returns true

Для перевірки закінчується рядок "g123":

    "testString123".hasSuffixCheck(suffix: "g123", isCaseSensitive: true) // Returns true
    "testString123".hasSuffixCheck(suffix: "g123", isCaseSensitive: false) // Returns true

Щоб перевірити, чи містить рядок "RING12":

    "testString123".containsSubString(theSubString: "RING12", isCaseSensitive: true) // Returns false
    "testString123".containsSubString(theSubString: "RING12", isCaseSensitive: false) // Returns true

Щоб перевірити, чи містить рядок "ring12":

    "testString123".containsSubString(theSubString: "ring12", isCaseSensitive: true) // Returns true
    "testString123".containsSubString(theSubString: "ring12", isCaseSensitive: false) // Returns true
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.