19

In Swift 4, new method .split(separator:) is introduced by apple in String struct. So to split a string with whitespace which is faster for e.g..

let str = "My name is Sudhir"

str.components(separatedBy: " ")

//or 

str.split(separator: " ")
rmaddy
  • 314,917
  • 42
  • 532
  • 579
Sudhir kumar
  • 659
  • 5
  • 21
  • I don't think this has any performance differences. – dasdom Sep 21 '17 at 13:02
  • 1
    Why don't you test it and measure the performance? – Btw, the method is not new in Swift 4, see e.g. https://stackoverflow.com/a/25229901/1187415. – Martin R Sep 21 '17 at 13:04
  • @MartinR please see https://developer.apple.com/documentation/swift/string?changes=latest_minor mention by apple that its added by them in latest version – Sudhir kumar Sep 21 '17 at 13:10
  • 2
    Looking into code, `.components` delegates to `NSString` while `split` is implemented through `Collection` interface. According to `FIXME:`, `components` will be in future delegated to `split`, see https://github.com/apple/swift/blob/96a3b9709cda03499f79775426fbf67854283188/stdlib/public/SDK/Foundation/NSStringAPI.swift#L731 – Sulthan Sep 21 '17 at 13:14
  • @Sudhirkumar: We are both right in some sense :) – In Swift 3 it was `str.characters.split(...)` because a String itself was not a collection. – Martin R Sep 21 '17 at 13:16

3 Answers3

19

Performance aside, there is an important difference between split(separator:) and components(separatedBy:) in how they treat empty subsequences.

They will produce different results if your input contains a trailing whitespace:

    let str = "My name is Sudhir " // trailing space

    str.split(separator: " ")
    // ["My", "name", "is", "Sudhir"]

    str.components(separatedBy: " ")
    // ["My", "name", "is", "Sudhir", ""] ← Additional empty string

To have both produce the same result, use the omittingEmptySubsequences:false argument (which defaults to true):

    // To get the same behavior:
    str.split(separator: " ", omittingEmptySubsequences: false)
    // ["My", "name", "is", "Sudhir", ""]

Details here:

https://developer.apple.com/documentation/swift/string/2894564-split

Mark
  • 6,647
  • 1
  • 45
  • 88
13

I have made sample test with following Code.

  var str = """
                One of those refinements is to the String API, which has been made a lot easier to use (while also gaining power) in Swift 4. In past versions of Swift, the String API was often brought up as an example of how Swift sometimes goes too far in favoring correctness over ease of use, with its cumbersome way of handling characters and substrings. This week, let’s take a look at how it is to work with strings in Swift 4, and how we can take advantage of the new, improved API in various situations. Sometimes we have longer, static strings in our apps or scripts that span multiple lines. Before Swift 4, we had to do something like inline \n across the string, add an appendOnNewLine() method through an extension on String or - in the case of scripting - make multiple print() calls to add newlines to a long output. For example, here is how TestDrive’s printHelp() function (which is used to print usage instructions for the script) looks like in Swift 3  One of those refinements is to the String API, which has been made a lot easier to use (while also gaining power) in Swift 4. In past versions of Swift, the String API was often brought up as an example of how Swift sometimes goes too far in favoring correctness over ease of use, with its cumbersome way of handling characters and substrings. This week, let’s take a look at how it is to work with strings in Swift 4, and how we can take advantage of the new, improved API in various situations. Sometimes we have longer, static strings in our apps or scripts that span multiple lines. Before Swift 4, we had to do something like inline \n across the string, add an appendOnNewLine() method through an extension on String or - in the case of scripting - make multiple print() calls to add newlines to a long output. For example, here is how TestDrive’s printHelp() function (which is used to print usage instructions for the script) looks like in Swift 3
          """

    var newString = String()

    for _ in 1..<9999 {
        newString.append(str)
    }

    var methodStart = Date()

 _  = newString.components(separatedBy: " ")
    print("Execution time Separated By: \(Date().timeIntervalSince(methodStart))")

    methodStart = Date()
    _ = newString.split(separator: " ")
    print("Execution time Split By: \(Date().timeIntervalSince(methodStart))")

I run above code on iPhone6 , Here are the results

Execution time Separated By: 8.27463299036026 Execution time Split By: 4.06880903244019

Conclusion : split(separator:) is faster than components(separatedBy:).

Sudhir kumar
  • 659
  • 5
  • 21
  • 1
    Which optimization level have you used? – Sulthan Sep 21 '17 at 13:53
  • Sorry didn't understand. I simply test both methods on same set of input and check execution time taken by each method. – Sudhir kumar Sep 21 '17 at 13:56
  • 4
    The comparison does not make sense if you don't have full optimization enabled. – Sulthan Sep 21 '17 at 14:19
  • 2
    Then please help me , How do I test it. – Sudhir kumar Sep 21 '17 at 14:20
  • 9
    Use of `components` is returning a `[String]`, where `split` is returning `[Substring]`. Each `String` is a String allocation, `Substring` is only using references, does not allocate new String and is faster and cheaper. – bauerMusic Sep 23 '19 at 07:09
  • @Sudhirkumar he's referring to Swift optimization level in Xcode -> Target -> Build settings. If you run your app in debug mode, by default it's Onone; you can manually change it to product level "fast" optimization level. Check this out if you still can't find it: https://github.com/hyperoslo/BarcodeScanner/issues/106 – superarts.org Jun 15 '21 at 20:01
1

Maybe a little late to answer:

When you play with them, they behave a little bit different:

str.components(separatedBy: "\n\n")

This call can give you some interesting results

str.split(separator: "\n\n")

This leads to an compile error as you must provide a single character.

sprohaszka
  • 43
  • 5