Swift 字符串和字符

字符串字面量

字符串字面量是由双引号("")包裹着具有固定顺序的文本字符集,可以用来给常量或者变量提供初始值。在Swift 4中新增加了多行字符串字面量,由三个双引号(""")包裹着的具有固定顺序的文本字符集,在多行字符串字面量中,可以使用反斜杠(\)作为续行符。对于缩进风格,关闭引号之前的的空白字符决定有多少列被忽略,如下图所示。

关闭引号前有4列被忽略

初始化空字符串

和Java类似,可以直接将空字符串字面量赋值给变量,通过isEmpty属性判断字符串是否为空。

字符串是值类型

如果创建了一个新的字符串,那么当其进行常量、变量赋值操作,或在函数/方法中传递时,会进行值拷贝。在实际编译时,Swift 编译器会优化字符串的使用,使实际的复制只发生在绝对必要的情况下,这意味着将字符串作为值类型的同时可以获得极高的性能。

使用字符

字符串其实就是字符的有序集合,可以通过for-in循环来遍历字符串中的characters属性来获取每一个字符的值

for character in "Dog!🐶".characters {

    print(character)

}

// D

// o

// g

// !

// 🐶

标明一个Character类型并用字符字面量进行赋值,可以建立一个独立的字符常量或变量

let exclamationMark: Character = "!"

字符串可以通过传递一个值类型为Character的数组作为自变量来初始化:

let catCharacters: [Character] = ["C", "a", "t", "!", "🐱"]

let catString = String(catCharacters)

print(catString)

// 打印输出:"Cat!🐱"

字符串插值

字符串插值是一种构建新字符串的方式,可以在其中包含常量、变量、字面量和表达式。 插入的字符串字面量的每一项都在以反斜线为前缀的圆括号中

let multiplier = 3

let message = "\(multiplier) times 2.5 is \(Double(multiplier) * 2.5)"

// message 是 "3 times 2.5 is 7.5"

可扩展的字形群集

每一个Swift的Character类型代表一个可扩展的字形群。 一个可扩展的字形群是一个或多个可生成人类可读的字符 Unicode 标量的有序排列。

计算字符数量

如果想要获得一个字符串中Character值的数量,可以使用字符串的characters属性的count属性

let unusualMenagerie = "Koala 🐨, Snail 🐌, Penguin 🐧, Dromedary 🐪"

print("unusualMenagerie has \(unusualMenagerie.characters.count) characters")

// 打印输出 "unusualMenagerie has 40 characters"

访问和修改字符串

可以通过字符串的属性和方法来访问和修改它,当然也可以用下标语法完成。

字符串索引

每一个String值都有一个关联的索引(index)类型,String.Index,它对应着字符串中的每一个Character的位置。前面提到,不同的字符可能会占用不同数量的内存空间,所以要知道Character的确定位置,就必须从String开头遍历每一个Unicode标量直到结尾。因此,Swift的字符串不能用整数(integer)做索引。

使用startIndex属性可以获取一个String的第一个Character的索引。使用endIndex属性可以获取最后一个Character的后一个位置的索引。因此,endIndex属性不能作为一个字符串的有效下标。如果String是空串,startIndex和endIndex是相等的。

通过调用String的index(before:)或index(after:)方法,可以立即得到前面或后面的一个索引。还可以通过调用index(_:offsetBy:)方法来获取对应偏移量的索引,这种方式可以避免多次调用index(before:)或index(after:)方法。

let greeting = "Guten Tag!"

greeting[greeting.startIndex]

// G

greeting[greeting.index(before: greeting.endIndex)]

// !

greeting[greeting.index(after: greeting.startIndex)]

// u

let index = greeting.index(greeting.startIndex, offsetBy: 7)

greeting[index]

// a

使用characters属性的indices属性会创建一个包含全部索引的范围(Range),用来在一个字符串中访问单个字符

for index in greeting.characters.indices {

    print("\(greeting[index]) ", terminator: "")

}

// 打印输出 "G u t e n  T a g ! "

字符串/字符相等

字符串/字符可以用等于操作符(==)和不等于操作符(!=),如果两个字符串(或者两个字符)的可扩展的字形群集是标准相等的,那就认为它们是相等的。在这个情况下,即使可扩展的字形群集是有不同的 Unicode 标量构成的,只要它们有同样的语言意义和外观,就认为它们标准相等。

前缀/后缀相等

通过调用字符串的hasPrefix(_:)/hasSuffix(_:)方法来检查字符串是否拥有特定前缀/后缀,两个方法均接收一个String类型的参数,并返回一个布尔值。

推荐阅读更多精彩内容