Swift之基础知识
- 出于对Swift3.0的学习,写下这篇基本语法的笔记。希望能帮助记忆 -0-
- 这边提供Swift3.0中文教材,资源链接: https://pan.baidu.com/s/1c2GDBtI 密码: ea4i
一. 第一个应用程序
0. 小技巧:> 'option + cmd + /': 增加文档注释> 'color + 回车': 弹出颜色选择> '图片名 + 回车': 图像视图> '// MARK: - XXX': mark> '// TODO: XXX'和'// FIXME: XXX': 临时标注(一般用后者)1. 和OC的区别:1.1 类和对象 OC: '[[UIView alloc] initWithXXX: ];' Swift: 'UIView(XXX: )' 'UIView()' 1.2 类方法 OC: '[UIColor redColor]' Swift: 'UIColor.red()' 2.0 : 'UIColor.redColor()' 1.3 访问当前对象属性,可以不使用'self' 建议都不用, 因为闭包(类似block)时需要使用'self' 1.4 没有';', 默认一行一句代码就不出错1.5 枚举类型 OC: 'UIButtonTypeContactAdd' Swift: '.contactAdd' 1.6 监听方法 OC: '@selector' Swift: '#selector',有参数不需要加':' 2.0: 直接使用"",有参数需要加':'
二. 变量与常量
1.变量: 值能被修改, var修饰 常量: 值不能被修改, let修饰2.自动推导: 变量/常量的类型会根据右侧的代码执行结果,推导对应的类型(option + click查看)3.注意点: > 在任何不同类型的数据之间, 不允许直接运算 > 不存在基本数据类型, 都是结构体4.optional可选项(等价于:类型+?) 4.1 可以有值, 可以为nil不是任何数据类型,不能参与计算 4.2 计算要用'!'强制解包, 如果有nil, 则直接报错:unexpectedly found nil wghile unwrapping an Optional value 4.3 let的可选值没有默认值 var的可选值默认为nil 4.4 空合运算符( a ?? b )将对可选类型 a 进行空判断:如果a为nil,则执行b,否则执行a
三. 分支语句
1. if语句
1.1 注意点 1) 条件不需要'()' 2) 语句必须有'{}'
2. ??可选项的判断
2.1 空合运算符( a ?? b )将对可选类型 a 进行空判断:如果a为nil,则执行b,否则执行a2.1 优先级比较低, 使用时最好加(), 例子如下: let aName: String? = "老王" print((aName ?? "") + "hello") // 打印:老王hello print(aName ?? "" + "hello")// 打印:老王
3. if let & if var
let aDog: String? = "laowang"let aAge: Int? = 10// 3.1 判断对象的值是否为nil, {}内一定有值,可以直接使用不需要解包// 3.2 if var连用, {}可以对值进行修改if var name = aDog, let age = aAge { // 进入分支后,name和age一定有值,不需要解包 // name和age的作用于仅在{}中 name = "laoxiao" print(name+String(age))}else { print("name或age有一个为nil")}
3. guard let
// 3.1. guard let 和 if let刚好相反// 3.2. guard let 守护一定有值,如果没有直接返回let aDog: String? = "laowang"let aAge: Int? = 10guard let name = aDog, let age = aAge else { print("name或age为nil") return}// 代码执行到这, name和age一定有值// 通常判断是否有值之后, 会做具体的逻辑实现,通常代码多// 如果用if let会多一层分支,guard是降低分支层次的办法print(name+String(age))
4. switch
// 1.值可以是任何类型// 2.作用域仅在 case 内部// 3.不需要 break// 4.如果要多值,使用','// 5.每一个 case 都要有代码let name = "nick"switch name {case "nick": let age = 18 print("one \(age)")case "fil": print("two")case "Davi": print("three")case "": break //相当于有一行代码case "tom","ningcol": print("tomAndNingcol")default: print("other")}
5. for循环
// 5.1 普通for循环: ..< 左闭右开; ...闭区间for i in 0..<10 { print(i)}// 5.2 使用 "_" 忽略不关心的值for _ in 0..<5{ print("hello")}// 5.3 递增(步数为2)for i in stride(from: 0, to: 12, by: 2) { print(i)}// 5.4 递减for i in stride(from: 12, to: 0, by: -2) { print(i)}// 5.5 反序循环let range = 0...10for i in range.reversed(){ print(i)}/***************************while循环*************************/print("----while循环----")var n = 2while n < 100 { n=n * 2}print(n)/***************************epeat-while循环*************************///它和 while 的区别是在判断循环条件之前,先执行一次循环的代码块。然后重复循环直到条件为 falsevar m = 2repeat { m = m * 2} while m < 100print(m)
四. 字符串
1) String 结构体,效率比对象高,一般推荐使用,支持遍历
2) NSString 继承NSObject
1. 遍历
for c in str.characters { print(c)}
2. 长度
// 2.1 放回指定编码对应的字节数// UTF8 的编码(0-4个), 每个汉字是3个字节str.lengthOfBytes(using: .utf8)// 2.2 返回字符个数(最常用)str.character.count// 2.3 使用NSString中转实现2.2let ocStr = str as NSStringocStr.length
3. 拼接
// 1."\(变量或常量)"// 但是要注意可选项, 用??语句let name:String? = "老王"let age = 80let location = "隔壁"print(location + (name ?? "a") + String(age) + "岁")// "隔壁老王80岁"print("\(location)\(name)\(age)岁")// "隔壁Optional("老王")80岁"let rect = CGRect(x: 0, y: 0, width: 100, height: 100)print("\(rect)")// "(0.0, 0.0, 100.0, 100.0)"// 2.格式化字符串let h = 13let m = 5let s = 9let timeStr = String(format: "%02d:%02d:%02d", arguments: [h,m,s])// "13:05:09"let timeStr1 = String(format: "%02d:%02d:%02d", h,m,s)// "13:05:09"
4. 子串
一般使用NSString作为中转, 因为Swift取子串一直在优化, swift方法忽略
五. 集合
1. 数组
和OC一样, 但是没有'@'
1.1 定义和初始化
let array1 = ["zhangsan","lisi"]let array2 = [1,2,3,4,5]var array3:[Int] // 定义一个数组(没有初始化)array3 = [Int]() //初始化//声明空数组,(必须初始化)let array4 = [String]() // 等价上面两行代码// Any任意对象let array5:[Any] = ["zhangsan","lisi",20]var arr3 = [Double](repeating: 0.0, count: 3) //[0.0, 0.0, 0.0]var arr4 = Array(repeating: 3.0, count: 3) //[3.0, 3.0, 3.0]
1.2 遍历
// 1.内容遍历for name in array1{ print(name)}// 2.下标遍历for i in 0..
1.3 增删改
// 增加arr.append("haha")// 修改arr[0] = "Tom"// 删除arr.removeFirst()arr.remove(at: 2)// 删除全部并保留空间arr.removeAll(keepingCapacity: true)print(arr.capacity) //数组容量/************容量*************/// 容量每次都会在原来基础上 * 2print("初始容量 \(array3.capacity)")for i in 0..<8{ array3.append(i) print("\(array3),容量:\(array3.capacity)")}
1.4 合并
// 直接相加var arr = ["haha", "nihao", "hello"]let arr1 = ["Evi","Tank"]arr += arr1// 注意点:类型必须一致var arr1:[NSObject] = ["Evi" as NSObject,"Tank" as NSObject]let arr2:[NSObject] = [1 as NSObject,2 as NSObject]arr1 += arr2;
2. 字典
OC中使用
{}
, swift中使用[]
2.1 定义和初始化
let dict1 = ["name":"lisi","age":"18"]// 不同类型必须指明为 anyvar dict2:[String:Any] = ["name":"lisi","age":18]// 字典数组let array = [ ["name":"lisi","age":"18"], ["name":"wangwu","age":8]]let array1:[[String:Any]] = [ ["name":"lisi","age":"18"], ["name":"wangwu","age":8]]
2.2 增删改
// 增加dict2["sex"] = "man"// 修改(通过key来取值,key不存在就是新增)dict2["name"] = "zhangsan"// 删除(直接给key进行删除)dict2.removeValue(forKey: "age")
2.3 遍历
for e in dict2{ //e 为元组 print("字典遍历:\(e) e.key:\(e.key) value:\(e.value)")}for (key,value) in dict2{ print("key:\(key), value:\(value)")}
2.4 合并
var dict3:[String:Any] = ["name":"zhangsan","age":18,"sex":"man"]let dict4:[String:Any] = ["name":"haha","height":50]// 如果key存在修改 不存在会增加for e in dict4{ dict3[e.key] = dict4[e.key]}print("合并dict:" + String(format: "%@", dict3))
六. 函数
1. 定义
// 格式: 函数名(形参列表) -> 返回值类型func sum(x: Int ,y: Int) -> Int{ return x + y}/** 1.外部参数是在形参前加一个名字 2.外部参赛不影响内部细节 3.外部参数让调用更加直观*/func sum(numX x: Int ,numY y: Int) -> Int{ return x + y}print(sum(numX: 30, numY: 40))// 外部参数使用 "_" 会忽略形参func sum(_ x: Int ,_ y: Int) -> Int{ return x + y}print(sum(40, 50))// 不指定参数的值就为默认值func sum1(x: Int = 1 ,y: Int = 2) -> Int{ return x + y}print(sum1())print(sum1(x: 10, y: 10))print(sum1(x: 20))print(sum1(y: 20))
2. 闭包
2.1 定义
提前准备好的代码,在需要的时候执行,类似于oc中的block
/** 1.参数,返回值,实现代码都是写在 {} 中 2.必须以 "in" 关键字分隔定义和实现 3.格式: (形参列表 -> 返回值类型 in) 实现代码*/// 无返回值,有参数let demo1 = { (x: Int) -> () in print(x)}demo1(50)// 有返回值,有参数let demo2 = { (x: Int) -> (Int) in return 40 + x}print(demo2(40))
2.2 实际应用
同OC中的 Block 一样
2.3 循环引用
- 类似于oc一样,用
weak var weakSelf = self
// 注意只能用var来创建 - 在闭包中加入
[weak self] in
,此时self为可选类型 - 在闭包中加入
[unknow self] in
,不推荐,此时self不是可选类型,当self对象释放了在调用会出现奔溃。
2.4 尾随闭包
当一个函数的最后一个参数是闭包时,可以写成尾随闭包格式,了解就好。
// 尾随GCD闭包DispatchQueue.global().async {}// 普通GCD闭包DispatchQueue.global().async(execute: {})