Swift5.1教程(三)--字符串和字符

字符串是一系列的字符,比如说 “hello, world”或者 “likeyou”。Swift 的字符串用 String类型来表示。 String的内容可以通过各种方法来访问到,包括作为 Character值的集合。

字符串的可修改能力通过选择常量和变量来进行管理。Swift 的 String类型桥接到了基础库中的 NSString类。Foundation 同时也扩展了所有 NSString 定义的方法给 String 。也就是说,如果你导入 Foundation ,就可以在 String 中访问所有的 NSString 方法,无需转换格式。

初始化一个空字符串

var str1 = ""
var str2:String=""
var str3 = String()
//通过检查布尔量 isEmpty属性来确认一个 String值是否为空:
if(str1.isEmpty&&str2.isEmpty&&str3.isEmpty){
print("字符串是空的")
}
//打印结果:字符串是空的

字符串可变性

你可以通过把一个 String设置为变量(这里指可被修改),或者为常量(不能被修改)来指定它是否可以被修改(或者改变):

var variableString = "Horse"
variableString += " and carriage"
// variableString 的值是 "Horse and carriage"
let constantString = "Highlander"
constantString += " and another Highlander"
//编译报错 this reports a compile-time error - a constant string cannot be modified

这个功能与 Objective-C 和 Cocoa 中的字符串改变不同,通过选择不同的类( NSString和 NSMutableString)来明确字符串是否可被改变。

字符串是值类型

Swift 的 String类型是一种值类型。如果你创建了一个新的 String值, String值在传递给方法或者函数的时候会被复制过去,还有赋值给常量或者变量的时候也是一样。每一次赋值和传递,现存的 String值都会被复制一次,传递走的是拷贝而不是原本。

Swift 的默认拷贝 String行为保证了当一个方法或者函数传给你一个 String值,你就绝对拥有了这个 String值,无需关心它从哪里来。你可以确定你传走的这个字符串除了你自己就不会有别人改变它。

另一方面,Swift 编译器优化了字符串使用的资源,实际上拷贝只会在确实需要的时候才进行。这意味着当你把字符串当做值类型来操作的时候总是能够有用很棒的性能。

操作字符

你可以通过 for-in循环遍历 String 中的每一个独立的 Character值:

var name = "dapeng"
for character in name {
print(character)
}
输出
d
a
p
e
n
g

String值可以通过传入 Character值的字符串作为实际参数到它的初始化器来构造:

let catCharacters: [Character] = ["d", "a", "p", "e", "n", "g"]
let catString = String(catCharacters)
print(catString)
输出
dapeng

连接字符串和字符

String值能够被加起来(或者说连接),使用加运算符( +)来创建新的String值:

let string1 = "hello"
let string2 = "world"
var welcome = string1 + string2
print(welcome)//输出helloworld

你使用 String类型的 append()方法来可以给一个 String变量的末尾追加 Character值:

let exclamationMark: Character = "!"
welcome.append(exclamationMark)
print(welcome)//输出helloworld!

你不能把 String或者 Character追加到已经存在的 Character变量当中,因为 Character值能且只能包含一个字符。

字符统计

要在字符串中取回 Character值的总数,使用字符串的 count属性:

let name = "dapeng"
print(name.count)
//输出 6

通过 count属性返回的字符统计并不会总是与包含相同字符的 NSString中 length属性相同。 NSString中的长度是基于在字符串的 UTF-16 表示中16位码元的数量来表示的,而不是字符串中 Unicode 扩展字形集群的数量。

访问和修改字符串

你可以通过下标脚本语法或者它自身的属性和方法来访问和修改字符串。

  • 字符串索引

你可以使用下标脚本语法来访问 String索引中的特定 Character。

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

使用 indices属性来访问字符串中每个字符的索引。

for index in greeting.indices {
print("\(greeting[index]) ", terminator: "")
}
// Prints "G u t e n   T a g ! "
  • 插入和删除

要给字符串的特定索引位置插入字符,使用 insert(_:at:)方法,另外要冲入另一个字符串的内容到特定的索引,使用 insert(contentsOf:at:) 方法

var welcome = "hello"
//这个方法是插入Character类型
welcome.insert("!", at: welcome.endIndex)
//输出 "hello!"
//这个方法是插入S类型的,S类型是Collection类型或者是字符类型,详细见接口文档
welcome.insert(contentsOf: " there", at: welcome.index(before: welcome.endIndex))
//输出 "hello there!"

要从字符串的特定索引位置移除字符,使用 remove(at:)方法,另外要移除一小段特定范围的字符串,使用 removeSubrange(_:) 方法:

welcome.remove(at: welcome.index(before: welcome.endIndex))
//输出 "hello there"

let range = welcome.index(welcome.endIndex, offsetBy: -6)..<welcome.endIndex
welcome.removeSubrange(range)
//输出 "hello"

子字符串

当你获得了一个字符串的子字符串——比如说,使用下标或者类似 prefix(_:) 的方法——结果是一个 Substring 的实例,不是另外一个字符串。Swift 中的子字符串拥有绝大部分字符串所拥有的方法,也就是说你可以用操作字符串相同的方法来操作子字符串。总之,与字符串不同,在字符串上执行动作的话你应该使用子字符串执行短期处理。当你想要把结果保存得长久一点时,你需要把子字符串转换为 String 实例。比如说:

let greeting = "Hello, world!";
let index = greeting.index(of: ",") ?? greeting.endIndex;
let beginning = greeting[..<index];
// beginning is "Hello"
// Convert the result to a String for long-term storage.
let newString = String(beginning);

与字符串类似,每一个子字符串都有一块内存区域用来保存组成子字符串的字符。字符串与子字符串的不同之处在于,作为性能上的优化,子字符串可以重用一部分用来保存原字符串的内存,或者是用来保存其他子字符串的内存。(字符串也拥有类似的优化,但是如果两个字符串使用相同的内存,他们就是等价的。)这个性能优化意味着在你修改字符串或者子字符串之前都不需要花费拷贝内存的代价。如同上面所说的,子字符串并不适合长期保存——因为它们重用了原字符串的内存,只要这个字符串有子字符串在使用中,那么这个字符串就必须一直保存在内存里。

在上面的例子中, greeting 是一个字符串,也就是说它拥有一块内存保存着组成这个字符串的字符。由于 beginning 是 greeting 的子字符串,它重用了 greeting 所用的内存。不同的是, newString 是字符串——当它从子字符串创建时,它就有了自己的内存。下面的图例显示了这些关系:

[插入图片]

字符串比较

Swift 提供了三种方法来比较文本值:字符串和字符相等性,前缀相等性以及后缀相等性。

  • 字符串和字符相等性

let str1 = "h"
let str2 = "h"
if str1 == str2{
print("相等")
}else{
print("不相等")
}
//输出:相等
  • 前缀和后缀相等性

要检查一个字符串是否拥有特定的字符串前缀或者后缀,调用字符串的 hasPrefix(:)和 hasSuffix(:)方法,它们两个都会接受一个 String 类型的实际参数并且返回一个布尔量值。

let name = "dapeng"
if name.hasPrefix("da") {
print("yes")
}
if name.hasSuffix("g") {
print("yes")
}
//很显然,都会输出yes
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容