Swift:文字列の最初から最後までの部分文字列を取得する方法


110

文字列を別の文字列に変換する最善/最も簡単な方法を学びたいのですが、サブセットのみを使用して、最初から文字の最後のインデックスに移動します。

たとえば、「www.stackoverflow.com」を「www.stackoverflow」に変換します。どのコードスニペットがそれを実行し、最も迅速なようになりますか?(これが議論を引き起こさないことを願っていますが、Swiftで部分文字列を処理する方法についての良いレッスンを見つけることができません。

回答:


202

後方にアクセスするだけ

最善の方法はsubstringToIndexendIndexプロパティとadvanceグローバル関数を組み合わせて使用することです。

var string1 = "www.stackoverflow.com"

var index1 = advance(string1.endIndex, -4)

var substring1 = string1.substringToIndex(index1)

後ろからひもを探す

利用rangeOfStringとセットoptions.BackwardsSearch

var string2 = "www.stackoverflow.com"

var index2 = string2.rangeOfString(".", options: .BackwardsSearch)?.startIndex

var substring2 = string2.substringToIndex(index2!)

拡張機能なし、純粋な慣用的なSwift

Swift 2.0

advanceはの一部にIndexなり、と呼ばれていadvancedByます。あなたはそれを好きです:

var string1 = "www.stackoverflow.com"

var index1 = string1.endIndex.advancedBy(-4)

var substring1 = string1.substringToIndex(index1)

Swift 3.0

可変サイズの要素がadvancedByあるStringため、aを呼び出すことはできません。あなたが使用する必要がありますindex(_, offsetBy:)

var string1 = "www.stackoverflow.com"

var index1 = string1.index(string1.endIndex, offsetBy: -4)

var substring1 = string1.substring(to: index1)

多くの名前が変更されました。ケースはcamelCaseで記述され、にstartIndexなりましたlowerBound

var string2 = "www.stackoverflow.com"

var index2 = string2.range(of: ".", options: .backwards)?.lowerBound

var substring2 = string2.substring(to: index2!)

また、強制アンラップはお勧めしませんindex2。オプションのバインディングまたはを使用できますmap。個人的に、私は以下を使用することを好みmapます:

var substring3 = index2.map(string2.substring(to:))

スウィフト4

Swift 3バージョンは引き続き有効ですが、インデックス範囲で添え字を使用できるようになりました。

let string1 = "www.stackoverflow.com"

let index1 = string1.index(string1.endIndex, offsetBy: -4)

let substring1 = string1[..<index1]

2番目のアプローチは変更されていません。

let string2 = "www.stackoverflow.com"

let index2 = string2.range(of: ".", options: .backwards)?.lowerBound

let substring3 = index2.map(string2.substring(to:))

1
@ fpg1503に感謝します。これは私が探していた一種の答えです。他の答えは私に言語の特徴を教えたり思い出させたりしましたが、これが私の問題解決に使用する方法です。
ジェイソンホッカー

1
Swift 2.0を使用すると、string1.endIndex.advancedBy(-4)代わりに機能しますadvance
Alex Koshy

92
これを単純に行う方法は真剣にありませんか?
devios1 2016

12
@deviosはすべてtrueです。私はObjective-Cと比較してSwiftに感謝していますが、このめちゃくちゃ厄介なコードにつながる私見は、コードの99%とそこにいるプログラマーには関係ありません。さらに悪いことに、私たちは自分で悪い便利な拡張機能を書くことになります。通常の開発者がこれらのより高いレベルの懸念を理解できない場合、それらはアドホックAPIを記述して、私たちがソフトウェアを作成する際の歩行者の懸念に対処できる最低限のものです。変換する傾向NSStringも明らかに悪いです。結局のところ、私たち全員がFoundation(レガシー)クラスから離れたいと思っているからです。だから...もっと怒鳴る!
Dan Rosenstark、2016年

2
Swift 4では、次のことが可能ですstring1[..<index1]。必要はありませんstring1.startIndex
bauerMusic

26

Swift 3、XCode 8

func lastIndexOfCharacter(_ c: Character) -> Int? {
    return range(of: String(c), options: .backwards)?.lowerBound.encodedOffset
}

以来advancedBy(Int)スウィフト3使用するので消えたStringの方法index(String.Index, Int)。このString拡張機能を部分文字列と友達と一緒にチェックしてください:

public extension String {

    //right is the first encountered string after left
    func between(_ left: String, _ right: String) -> String? {
        guard let leftRange = range(of: left), let rightRange = range(of: right, options: .backwards)
        , leftRange.upperBound <= rightRange.lowerBound
            else { return nil }

        let sub = self.substring(from: leftRange.upperBound)
        let closestToLeftRange = sub.range(of: right)!
        return sub.substring(to: closestToLeftRange.lowerBound)
    }

    var length: Int {
        get {
            return self.characters.count
        }
    }

    func substring(to : Int) -> String {
        let toIndex = self.index(self.startIndex, offsetBy: to)
        return self.substring(to: toIndex)
    }

    func substring(from : Int) -> String {
        let fromIndex = self.index(self.startIndex, offsetBy: from)
        return self.substring(from: fromIndex)
    }

    func substring(_ r: Range<Int>) -> String {
        let fromIndex = self.index(self.startIndex, offsetBy: r.lowerBound)
        let toIndex = self.index(self.startIndex, offsetBy: r.upperBound)
        return self.substring(with: Range<String.Index>(uncheckedBounds: (lower: fromIndex, upper: toIndex)))
    }

    func character(_ at: Int) -> Character {
        return self[self.index(self.startIndex, offsetBy: at)]
    }

    func lastIndexOfCharacter(_ c: Character) -> Int? {
        guard let index = range(of: String(c), options: .backwards)?.lowerBound else
        { return nil }
        return distance(from: startIndex, to: index)
    }
}

Swift 4の更新された拡張機能

public extension String {

    //right is the first encountered string after left
    func between(_ left: String, _ right: String) -> String? {
        guard
            let leftRange = range(of: left), let rightRange = range(of: right, options: .backwards)
            , leftRange.upperBound <= rightRange.lowerBound
            else { return nil }

        let sub = self[leftRange.upperBound...]
        let closestToLeftRange = sub.range(of: right)!            
        return String(sub[..<closestToLeftRange.lowerBound])
    }

    var length: Int {
        get {
            return self.count
        }
    }

    func substring(to : Int) -> String {
        let toIndex = self.index(self.startIndex, offsetBy: to)
        return String(self[...toIndex])
    }

    func substring(from : Int) -> String {
        let fromIndex = self.index(self.startIndex, offsetBy: from)
        return String(self[fromIndex...])
    }

    func substring(_ r: Range<Int>) -> String {
        let fromIndex = self.index(self.startIndex, offsetBy: r.lowerBound)
        let toIndex = self.index(self.startIndex, offsetBy: r.upperBound)
        let indexRange = Range<String.Index>(uncheckedBounds: (lower: fromIndex, upper: toIndex))
        return String(self[indexRange])
    }

    func character(_ at: Int) -> Character {
        return self[self.index(self.startIndex, offsetBy: at)]
    }

    func lastIndexOfCharacter(_ c: Character) -> Int? {
        guard let index = range(of: String(c), options: .backwards)?.lowerBound else
        { return nil }
        return distance(from: startIndex, to: index)
    }
}

使用法:

let text = "www.stackoverflow.com"
let at = text.character(3) // .
let range = text.substring(0..<3) // www
let from = text.substring(from: 4) // stackoverflow.com
let to = text.substring(to: 16) // www.stackoverflow
let between = text.between(".", ".") // stackoverflow
let substringToLastIndexOfChar = text.lastIndexOfCharacter(".") // 17

PS開発者String.Indexがプレーンではなく対処を余儀なくされたのは本当に奇妙ですIntString単純なsubstring()方法だけでなく、内部のメカニズムに悩む必要があるのはなぜですか?


あなたは使うべきではないencodedOffsetことを達成するために、このコメントと正しい方法を確認しstackoverflow.com/questions/34540185/...
レオDabus

18

下付き文字を使用してそれを行います(s[start..<end]):

スイフト3、4、5

let s = "www.stackoverflow.com"
let start = s.startIndex
let end = s.index(s.endIndex, offsetBy: -4)
let substring = s[start..<end] // www.stackoverflow

11

編集/更新:

、後スウィフト4または(Xcodeの10.0+)新しい使用することができますBidirectionalCollection 方法の(lastIndexのを:)

func lastIndex(of element: Element) -> Int?

let string = "www.stackoverflow.com"
if let lastIndex = string.lastIndex(of: ".") {
    let subString = string[..<lastIndex]  // "www.stackoverflow"
}

これは素晴らしい答えですが、Swift 2ではこのプロパティがNSUrlに移動されたようです。多分これは将来これを読む人々を助けるかもしれません...
チャッキー

8

これが私のやり方です。同じ方法で、またはこのコードをアイデアに使用できます。

let s = "www.stackoverflow.com"
s.substringWithRange(0..<s.lastIndexOf("."))

ここに私が使用する拡張機能があります:

import Foundation
extension String {

  var length: Int {
    get {
      return countElements(self)
    }
  }

  func indexOf(target: String) -> Int {
    var range = self.rangeOfString(target)
    if let range = range {
      return distance(self.startIndex, range.startIndex)
    } else {
      return -1
    }
  }

  func indexOf(target: String, startIndex: Int) -> Int {
    var startRange = advance(self.startIndex, startIndex)        
    var range = self.rangeOfString(target, options: NSStringCompareOptions.LiteralSearch, range: Range<String.Index>(start: startRange, end: self.endIndex))
    if let range = range {
      return distance(self.startIndex, range.startIndex)
    } else {
      return -1
    }
  }

  func lastIndexOf(target: String) -> Int {
    var index = -1
    var stepIndex = self.indexOf(target)
    while stepIndex > -1 {
      index = stepIndex
      if stepIndex + target.length < self.length {
        stepIndex = indexOf(target, startIndex: stepIndex + target.length)
      } else {
        stepIndex = -1
      }
    }
    return index
  } 

  func substringWithRange(range:Range<Int>) -> String {
    let start = advance(self.startIndex, range.startIndex)
    let end = advance(self.startIndex, range.endIndex)
    return self.substringWithRange(start..<end)
  }

}

Credit albertbori / Common Swift String Extensions

一般的に私は、特に文字列操作、検索、スライスなどのニーズに対して、拡張機能の強力な支持者です。


2
SwiftでStringのAPIを改善するために、これらは本当にAppleがすべきことだと思います。
skyline75489 2015年

2
このAPIを探しています。Apple Swiftには基本的なAPIがたくさんありません。これは非常に複雑な言語であり、完全ではありません。スウィフトはでたらめです!
Loc

:ここではあなたが参照するライブラリのスウィフト3バージョンにフォークと更新されていますgithub.com/iamjono/SwiftString
RenniePet

5

String 組み込みのサブストリング機能があります。

extension String : Sliceable {
    subscript (subRange: Range<String.Index>) -> String { get }
}

「文字の最初のインデックスに移動する」ことが必要な場合は、組み込みfind()関数を使用して部分文字列を取得できます。

var str = "www.stackexchange.com"
str[str.startIndex ..< find(str, ".")!] // -> "www"

最後のインデックスを見つけるために、を実装できますfindLast()

/// Returns the last index where `value` appears in `domain` or `nil` if
/// `value` is not found.
///
/// Complexity: O(\ `countElements(domain)`\ )
func findLast<C: CollectionType where C.Generator.Element: Equatable>(domain: C, value: C.Generator.Element) -> C.Index? {
    var last:C.Index? = nil
    for i in domain.startIndex..<domain.endIndex {
        if domain[i] == value {
            last = i
        }
    }
    return last
}

let str = "www.stackexchange.com"
let substring = map(findLast(str, ".")) { str[str.startIndex ..< $0] } // as String?
// if "." is found, substring has some, otherwise `nil`

追加:

多分、のBidirectionalIndexType特別バージョンfindLastはより高速です:

func findLast<C: CollectionType where C.Generator.Element: Equatable, C.Index: BidirectionalIndexType>(domain: C, value: C.Generator.Element) -> C.Index? {
    for i in lazy(domain.startIndex ..< domain.endIndex).reverse() {
        if domain[i] == value {
            return i
        }
    }
    return nil
}

Stringの添え字はSwift 2.0では使用できないようですが、置き換えはご存知ですか?
コリー

Swift 4以降でインデックスをBidirectionalIndexTypeに制約する方法を知っていますか?
レオダバス

5

次の拡張機能を使用できます。

Swift 2.3

 extension String
    {
        func substringFromIndex(index: Int) -> String
        {
            if (index < 0 || index > self.characters.count)
            {
                print("index \(index) out of bounds")
                return ""
            }
            return self.substringFromIndex(self.startIndex.advancedBy(index))
        }

        func substringToIndex(index: Int) -> String
        {
            if (index < 0 || index > self.characters.count)
            {
                print("index \(index) out of bounds")
                return ""
            }
            return self.substringToIndex(self.startIndex.advancedBy(index))
        }

        func substringWithRange(start: Int, end: Int) -> String
        {
            if (start < 0 || start > self.characters.count)
            {
                print("start index \(start) out of bounds")
                return ""
            }
            else if end < 0 || end > self.characters.count
            {
                print("end index \(end) out of bounds")
                return ""
            }
            let range = Range(start: self.startIndex.advancedBy(start), end: self.startIndex.advancedBy(end))
            return self.substringWithRange(range)
        }

        func substringWithRange(start: Int, location: Int) -> String
        {
            if (start < 0 || start > self.characters.count)
            {
                print("start index \(start) out of bounds")
                return ""
            }
            else if location < 0 || start + location > self.characters.count
            {
                print("end index \(start + location) out of bounds")
                return ""
            }
            let range = Range(start: self.startIndex.advancedBy(start), end: self.startIndex.advancedBy(start + location))
            return self.substringWithRange(range)
        }
    }

スウィフト3

extension String
{   
    func substring(from index: Int) -> String
    {
        if (index < 0 || index > self.characters.count)
        {
            print("index \(index) out of bounds")
            return ""
        }
        return self.substring(from: self.characters.index(self.startIndex, offsetBy: index))
    }

    func substring(to index: Int) -> String
    {
        if (index < 0 || index > self.characters.count)
        {
            print("index \(index) out of bounds")
            return ""
        }
        return self.substring(to: self.characters.index(self.startIndex, offsetBy: index))
    }

    func substring(start: Int, end: Int) -> String
    {
        if (start < 0 || start > self.characters.count)
        {
            print("start index \(start) out of bounds")
            return ""
        }
        else if end < 0 || end > self.characters.count
        {
            print("end index \(end) out of bounds")
            return ""
        }
        let startIndex = self.characters.index(self.startIndex, offsetBy: start)
        let endIndex = self.characters.index(self.startIndex, offsetBy: end)
        let range = startIndex..<endIndex

        return self.substring(with: range)
    }

    func substring(start: Int, location: Int) -> String
    {
        if (start < 0 || start > self.characters.count)
        {
            print("start index \(start) out of bounds")
            return ""
        }
        else if location < 0 || start + location > self.characters.count
        {
            print("end index \(start + location) out of bounds")
            return ""
        }
        let startIndex = self.characters.index(self.startIndex, offsetBy: start)
        let endIndex = self.characters.index(self.startIndex, offsetBy: start + location)
        let range = startIndex..<endIndex

        return self.substring(with: range)
    }
}

使用法:

let string = "www.stackoverflow.com"        
let substring = string.substringToIndex(string.characters.count-4)

5

スウィフト4:

extension String {

    /// the length of the string
    var length: Int {
        return self.characters.count
    }

    /// Get substring, e.g. "ABCDE".substring(index: 2, length: 3) -> "CDE"
    ///
    /// - parameter index:  the start index
    /// - parameter length: the length of the substring
    ///
    /// - returns: the substring
    public func substring(index: Int, length: Int) -> String {
        if self.length <= index {
            return ""
        }
        let leftIndex = self.index(self.startIndex, offsetBy: index)
        if self.length <= index + length {
            return self.substring(from: leftIndex)
        }
        let rightIndex = self.index(self.endIndex, offsetBy: -(self.length - index - length))
        return self.substring(with: leftIndex..<rightIndex)
    }

    /// Get substring, e.g. -> "ABCDE".substring(left: 0, right: 2) -> "ABC"
    ///
    /// - parameter left:  the start index
    /// - parameter right: the end index
    ///
    /// - returns: the substring
    public func substring(left: Int, right: Int) -> String {
        if length <= left {
            return ""
        }
        let leftIndex = self.index(self.startIndex, offsetBy: left)
        if length <= right {
            return self.substring(from: leftIndex)
        }
        else {
            let rightIndex = self.index(self.endIndex, offsetBy: -self.length + right + 1)
            return self.substring(with: leftIndex..<rightIndex)
        }
    }
}

次のようにテストできます。

    print("test: " + String("ABCDE".substring(index: 2, length: 3) == "CDE"))
    print("test: " + String("ABCDE".substring(index: 0, length: 3) == "ABC"))
    print("test: " + String("ABCDE".substring(index: 2, length: 1000) == "CDE"))
    print("test: " + String("ABCDE".substring(left: 0, right: 2) == "ABC"))
    print("test: " + String("ABCDE".substring(left: 1, right: 3) == "BCD"))
    print("test: " + String("ABCDE".substring(left: 3, right: 1000) == "DE"))

https://gitlab.com/seriyvolk83/SwiftExライブラリを確認してください。これらおよびその他の役立つメソッドが含まれています。


4

文字列の部分文字列を開始インデックスからその文字の最後のインデックスまで取得しますか?その場合は、次のSwift 2.0以降の方法のいずれかを選択できます。

必要なメソッド Foundation

文字の最後のインデックスを含む部分文字列を取得します。

import Foundation

let string = "www.stackoverflow.com"
if let rangeOfIndex = string.rangeOfCharacterFromSet(NSCharacterSet(charactersInString: "."), options: .BackwardsSearch) {
    print(string.substringToIndex(rangeOfIndex.endIndex))
}

// prints "www.stackoverflow."

文字の最後のインデックスを含まない部分文字列を取得します。

import Foundation

let string = "www.stackoverflow.com"
if let rangeOfIndex = string.rangeOfCharacterFromSet(NSCharacterSet(charactersInString: "."), options: .BackwardsSearch) {
    print(string.substringToIndex(rangeOfIndex.startIndex))
}

// prints "www.stackoverflow"

これらの操作を繰り返す必要がある場合は、拡張Stringすることをお勧めします。

import Foundation

extension String {
    func substringWithLastInstanceOf(character: Character) -> String? {
        if let rangeOfIndex = rangeOfCharacterFromSet(NSCharacterSet(charactersInString: String(character)), options: .BackwardsSearch) {
            return self.substringToIndex(rangeOfIndex.endIndex)
        }
        return nil
    }
    func substringWithoutLastInstanceOf(character: Character) -> String? {
        if let rangeOfIndex = rangeOfCharacterFromSet(NSCharacterSet(charactersInString: String(character)), options: .BackwardsSearch) {
            return self.substringToIndex(rangeOfIndex.startIndex)
        }
        return nil
    }
}

print("www.stackoverflow.com".substringWithLastInstanceOf("."))
print("www.stackoverflow.com".substringWithoutLastInstanceOf("."))

/*
prints:
Optional("www.stackoverflow.")
Optional("www.stackoverflow")
*/

必要としない方法 Foundation

文字の最後のインデックスを含む部分文字列を取得します。

let string = "www.stackoverflow.com"
if let reverseIndex = string.characters.reverse().indexOf(".") {
    print(string[string.startIndex ..< reverseIndex.base])
}

// prints "www.stackoverflow."

文字の最後のインデックスを含まない部分文字列を取得します。

let string = "www.stackoverflow.com"
if let reverseIndex = string.characters.reverse().indexOf(".") {
    print(string[string.startIndex ..< reverseIndex.base.advancedBy(-1)])
}

// prints "www.stackoverflow"

これらの操作を繰り返す必要がある場合は、拡張Stringすることをお勧めします。

extension String {
    func substringWithLastInstanceOf(character: Character) -> String? {
        if let reverseIndex = characters.reverse().indexOf(".") {
            return self[self.startIndex ..< reverseIndex.base]
        }
        return nil
    }
    func substringWithoutLastInstanceOf(character: Character) -> String? {
        if let reverseIndex = characters.reverse().indexOf(".") {
            return self[self.startIndex ..< reverseIndex.base.advancedBy(-1)]
        }
        return nil
    }
}

print("www.stackoverflow.com".substringWithLastInstanceOf("."))
print("www.stackoverflow.com".substringWithoutLastInstanceOf("."))

/*
prints:
Optional("www.stackoverflow.")
Optional("www.stackoverflow")
*/

4

インデックスがわかっている場合、部分文字列を取得する簡単で短い方法を次に示します。

let s = "www.stackoverflow.com"
let result = String(s.characters.prefix(17)) // "www.stackoverflow"

インデックスが文字列の長さを超えてもアプリはクラッシュしません。

let s = "short"
let result = String(s.characters.prefix(17)) // "short"

どちらの例もSwift 3に対応しています。


4

ガチャガチャと音がするのは stringVarます:

stringVar [ stringVar .index(stringVar .startIndex、offsetBy:...)

ではスウィフト4

拡張機能はその一部を減らすことができます:

extension String {

    func index(at: Int) -> String.Index {
        return self.index(self.startIndex, offsetBy: at)
    }
}

次に、使用法:

let string = "abcde"

let to = string[..<string.index(at: 3)] // abc
let from = string[string.index(at: 3)...] // de

tofromはタイプSubstring(またはString.SubSequance)であることに注意してください。新しい文字列は割り当てられず、処理がより効率的になります。

String型を戻すにSubstringは、にキャストする必要がありますString

let backToString = String(from)

これは、文字列が最終的に割り当てられる場所です。


3
func substr(myString: String, start: Int, clen: Int)->String

{
  var index2 = string1.startIndex.advancedBy(start)
  var substring2 = string1.substringFromIndex(index2)
  var index1 = substring2.startIndex.advancedBy(clen)
  var substring1 = substring2.substringToIndex(index1)

  return substring1   
}

substr(string1, start: 3, clen: 5)

3

スウィフト3

let string = "www.stackoverflow.com"
let first3Characters = String(string.characters.prefix(3)) // www
let lastCharacters = string.characters.dropFirst(4) // stackoverflow.com (it would be a collection)

//or by index 
let indexOfFouthCharacter = olNumber.index(olNumber.startIndex, offsetBy: 4)
let first3Characters = olNumber.substring(to: indexOfFouthCharacter) // www
let lastCharacters = olNumber.substring(from: indexOfFouthCharacter) // .stackoverflow.com

理解のための良い記事、なぜこれが必要なのか


2

2つの部分文字列メソッドで文字列を拡張しました。次のように、from / to範囲またはfrom / lengthを使用して部分文字列を呼び出すことができます。

var bcd = "abcdef".substring(1,to:3)
var cde = "abcdef".substring(2,to:-2)
var cde = "abcdef".substring(2,length:3)

extension String {
  public func substring(from:Int = 0, var to:Int = -1) -> String {
    if to < 0 {
        to = self.length + to
    }
    return self.substringWithRange(Range<String.Index>(
        start:self.startIndex.advancedBy(from),
        end:self.startIndex.advancedBy(to+1)))
  }
  public func substring(from:Int = 0, length:Int) -> String {
    return self.substringWithRange(Range<String.Index>(
        start:self.startIndex.advancedBy(from),
        end:self.startIndex.advancedBy(from+length)))
  }
}

self.lengthは使用できなくなりました。これを拡張機能に追加する必要があります(Andrewz、コードを更新してください。別の回答を投稿したくありません):public func length()-> Int {return self.lengthOfBytesUsingEncoding(NSUTF16StringEncoding)}
philippe

1

Swift 2.0 以下のコードはXCode 7.2でテストされています。下部にある添付のスクリーンショットを参照してください

import UIKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        var mainText = "http://stackoverflow.com"

        var range = Range(start: mainText.startIndex.advancedBy(7), end: mainText.startIndex.advancedBy(24))
        var subText = mainText.substringWithRange(range)


        //OR Else use below for LAST INDEX

        range = Range(start: mainText.startIndex.advancedBy(7), end: mainText.endIndex)
        subText = mainText.substringWithRange(range)
    }
}

1

Swift 5で

String.Index単純なInt値の代わりにIndexを表す必要があります。

また覚えて、我々はスウィフトから部分を取得しようとすると、String(値タイプ)、我々は実際に反復する必要がSequence戻りれ、プロトコルString.SubSequenceの代わりに入力しString

Stringから戻るにはString.SubSequenceString(subString)

以下の例:

    let string = "https://stackoverflow.com"
    let firstIndex = String.Index(utf16Offset: 0, in: string)
    let lastIndex = String.Index(utf16Offset: 6, in: string)
    let subString = String(string[firstIndex...lastIndex])

0

スウィフト2.0、それはこのようなものです:

var string1 = "www.stackoverflow.com"
var index1 = string1.endIndex.advancedBy(-4)
var substring1 = string1.substringToIndex(index1)

多分彼らはそれを変更したかもしれませんが、それは「advancedBy」です(「advanced」の末尾の「d」に注意してください)
ウェッジマーティン

0

andrewzの投稿を変更して、Swift 2.0(おそらくSwift 3.0)と互換性があるようにしました。私の控えめな意見では、この拡張機能は理解しやすく、他の言語(PHPなど)で使用可能な拡張機能に似ています

extension String {

    func length() -> Int {
        return self.lengthOfBytesUsingEncoding(NSUTF16StringEncoding)
    }
    func substring(from:Int = 0, to:Int = -1) -> String {
       var nto=to
        if nto < 0 {
            nto = self.length() + nto
        }
        return self.substringWithRange(Range<String.Index>(
           start:self.startIndex.advancedBy(from),
           end:self.startIndex.advancedBy(nto+1)))
    }
    func substring(from:Int = 0, length:Int) -> String {
        return self.substringWithRange(Range<String.Index>(
            start:self.startIndex.advancedBy(from),
            end:self.startIndex.advancedBy(from+length)))
    }
}

0

また、Swift 4の単純な文字列拡張を作成します。

extension String {
    func subStr(s: Int, l: Int) -> String { //s=start, l=lenth
        let r = Range(NSRange(location: s, length: l))!
        let fromIndex = self.index(self.startIndex, offsetBy: r.lowerBound)
        let toIndex = self.index(self.startIndex, offsetBy: r.upperBound)
        let indexRange = Range<String.Index>(uncheckedBounds: (lower: fromIndex, upper: toIndex))

        return String(self[indexRange])
     }
}

したがって、次のように簡単に呼び出すことができます。

"Hallo world".subStr(s: 1, l: 3) //prints --> "all"

0

このInt-based回避策を試してください:

extension String {
    // start and end is included
    func intBasedSubstring(_ start: Int, _ end: Int) -> String {
        let endOffset: Int = -(count - end - 1)
        let startIdx = self.index(startIndex, offsetBy: start)
        let endIdx = self.index(endIndex, offsetBy: endOffset)
        return String(self[startIdx..<endIdx])
    }
}

注:これは単なる練習です。境界はチェックしません。ニーズに合わせて変更します。


-1

これはSwiftで部分文字列を取得する簡単な方法です

import UIKit

var str = "Hello, playground" 
var res = NSString(string: str)
print(res.substring(from: 4))
print(res.substring(to: 10))

OPは、事前定義されたハードコードされた値を使用して部分文字列を取得する方法ではなく、「文字列の最初から最後までの部分文字列を取得する方法」を尋ねます。
Eric Aya

これは元の例でした:「www.stackoverflow.com」を「www.stackoverflow」に変換します。最初から最後まで部分文字列が必要な場合、何もする必要はありません。つまり、結果はそのまま入力文字列になります。上記の回答の有用性を復元してください。
ZagorTeNej 2017年
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.