一. 第一个应用程序
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,否则执行a
2.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? = 10
guard 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...10
for i in range.reversed(){
print(i)
}
/***************************while循环*************************/
print("----while循环----")
var n = 2
while n < 100 {
n=n * 2
}
print(n)
/***************************epeat-while循环*************************/
//它和 while 的区别是在判断循环条件之前,先执行一次循环的代码块。然后重复循环直到条件为 false
var m = 2
repeat {
m = m * 2
} while m < 100
print(m)
四. 字符串
- String 结构体,效率比对象高,一般推荐使用,支持遍历
- 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.2
let ocStr = str as NSString
ocStr.length
3. 拼接
// 1."\(变量或常量)"
// 但是要注意可选项, 用??语句
let name:String? = "老王"
let age = 80
let 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 = 13
let m = 5
let s = 9
let 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..<array2.count{
print(array2[i])
}
// 3.enum block遍历, 同时遍历下标和内容
for e in array2.enumerated(){
print(e)
print("元组 \(e.offset) \(e.element)")
}
// 4.同时遍历下标和内容 2
for (n,s) in array2.enumerated(){
print("\(n) \(s)")
}
// 4.反序遍历
for a in array2.reversed(){
print(a)
}
1.3 增删改
// 增加
arr.append("haha")
// 修改
arr[0] = "Tom"
// 删除
arr.removeFirst()
arr.remove(at: 2)
// 删除全部并保留空间
arr.removeAll(keepingCapacity: true)
print(arr.capacity) //数组容量
/************容量*************/
// 容量每次都会在原来基础上 * 2
print("初始容量 \(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"]
// 不同类型必须指明为 any
var 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: {
})
|
请发表评论