Swift语法01

2017-02-26  本文已影响0人  mian小爬

Swift 简介

查看Swift当前版本

$ xcrun swift --version

简介

历史

从发布至今,苹果的每一个举措都彰显其大力推广 Swift 的决心

版本

Swift 特色

Swift 现状

为什么要学习 Swift?

  1. 从4月份开始,苹果提供的资料已经没有 OC 的了,这说明苹果推动 Swift 的决心
  2. OC 源自于 smalltack-c,迄今已经有 40 多年的历史,虽然 OC 的项目还会在未来持续一段时间,但是更换成 Swift 是未来必然的趋势
  3. 现在很多公司都注重人才储备,如果会Swift,就业会有很大的优势,简历中如果写上会 Swift,虽然面试中虽然不会怎么被问到,但对于薪资提升有很大帮助,同时可以从另外一个侧面证明我们是有自学能力的人,这是所有企业都需要的
  4. Swift 里面融合了很多其他面向对象语言的思想,不像OC那么封闭,学会 Swift,再转其他语言会轻松很多
  5. Swift 毕竟也是出身自苹果,整体程序开发思路和 OC 是一样的,等 Swift 项目讲完后,大家完全可以用同样的思路写出 OC 的来,而且在翻写的过程中,能够对很多原本忽略的 OC 基本功有很大的加强和改善

Swift 开发快速体验

目标

学习资源

基本语法

目标

变量和常量

需要掌握


定义

//: # 常量
//: 定义常量并且直接设置数值
let x = 20
//: 常量数值一经设置,不能修改,以下代码会报错
// x = 30

//: 使用 `: 类型`,仅仅只定义类型,而没有设置数值
let x1: Int
//: 常量有一次设置数值的机会,以下代码没有问题,因为 x1 还没有被设置数值
x1 = 30
//: 一旦设置了数值之后,则不能再次修改,以下代码会报错,因为 x1 已经被设置了数值
// x1 = 50

//: # 变量
//: 变量设置数值之后,可以继续修改数值
var y = 200
y = 300

自动推导

重要技巧:Option + Click 可以查看变量的类型

[图片上传失败...(image-1c3941-1536302339170)]

没有隐式转换!!!

如果要对不同类型的数据进行计算,必须要显式的转换

let x2 = 100
let y2 = 10.5

let num1 = Double(x2) + y2
let num2 = x2 + Int(y2)

let & var 的选择

Optional 可选值

需要掌握


介绍

//: num 可以是一个整数,也可以是 nil,注意如果为 nil,不能参与计算
let num: Int? = 10

注意:必须要确保解包后的值不是 nil,否则会报错

//: num 可以是一个整数,也可以是 nil,注意如果为 nil,不能参与计算
let num: Int? = 10

//: 如果 num 为 nil,使用 `!` 强行解包会报错
let r1 = num! + 100

//: 使用以下判断,当 num 为 nil 时,if 分支中的代码不会执行
if let n = num {
    let r = n + 10
}

常见错误

unexpectedly found nil while unwrapping an Optional value

翻译

在[解包]一个可选值时发现 nil

?? 运算符

let num: Int? = nil

let r1 = (num ?? 0) + 10
print(r1)

控制流

if

let num = 200
if num < 10 {
    print("比 10 小")
} else if num > 100 {
    print("比 100 大")
} else {
    print("10 ~ 100 之间的数字")
}

三目运算

var a = 10
var b = 20
let c = a > b ? a : b
print(c)

适当地运用三目,能够让代码写得更加简洁

可选项判断

单个可选项判断

let url = NSURL(string: "http://www.baidu.com")

//: 方法1: 强行解包 - 缺陷,如果 url 为空,运行时会崩溃
let request = NSURLRequest(URL: url!)

//: 方法2: 首先判断 - 代码中仍然需要使用 `!` 强行解包
if url != nil {
    let request = NSURLRequest(URL: url!)
}


可选项条件判断

//: 1> 初学 swift 一不小心就会让 if 的嵌套层次很深,让代码变得很丑陋
if let u = url {
    if u.host == "www.baidu.com" {
        let request = NSURLRequest(URL: u)
    }
}

//: 2> 使用 ',' 可以获取前面赋值对象
if let u = url , u.host == "www.baidu.com" {
    let request = NSURLRequest(URL: u)
}

多个可选项判断

//: 3> 可以使用 `,` 同时判断多个可选项是否为空
let oName: String? = "张三"
let oNo: Int? = 100

if let name = oName {
    if let no = oNo {
        print("姓名:" + name + " 学号: " + String(no))
    }
}

if let name = oName, let no = oNo {
    print("姓名:" + name + " 学号: " + String(no))
}

判断之后对变量需要修改

let oName: String? = "张三"
let oNum: Int? = 18

if var name = oName, num = oNum {

    name = "李四"
    num = 1

    print(name, num)
}

guard

let oName: String? = "张三"
let oNum: Int? = 18

guard let name = oName else {
    print("name 为空")
    return
}

guard let num = oNum else {
    print("num 为空")
    return
}

// 代码执行至此,name & num 都是有值的
print(name)
print(num)

switch

let score = "优"

switch score {
    case "优":
        let name = "学生"
        print(name + "80~100分")
    case "良": print("70~80分")
    case "中": print("60~70分")
    case "差": print("不及格")
    default: break
}

for 循环

var sum = 0
for var i = 0; i < 10; i++ {
    sum += I
}
print(sum)
sum = 0
for i in 0..<10 {
    sum += I
}
print(sum)
sum = 0
for i in 0...10 {
    sum += I
}
print(sum)
for _ in 0...10 {
    print("hello")
}

字符串

在 Swift 中绝大多数的情况下,推荐使用 String 类型

字符串演练

for s in str.characters {
    print(s)
}
// 返回以字节为单位的字符串长度,一个中文占 3 个字节
let len1 = str.lengthOfBytesUsingEncoding(NSUTF8StringEncoding)
// 返回实际字符的个数
let len2 = str.characters.count

let str1 = "Hello"
let str2 = "World"
let i = 32
str = "\(i) 个 " + str1 + " " + str2

我和我的小伙伴再也不要考虑 stringWithFormat 了 :D

let str1 = "Hello"
let str2 = "World"
let i: Int? = 32
str = "\(i ?? 0) 个 " + str1 + " " + str2
let h = 8
let m = 23
let s = 9
let timeString = String(format: "%02d:%02d:%02d", arguments: [h, m, s])
let timeStr = String(format: "%02d:%02d:%02d", h, m, s)

String & Range 的结合

let helloString = "我们一起飞"
(helloString as NSString).substringWithRange(NSMakeRange(2, 3))
let str = "爱笑的人运气不会太差哟"
        let startIndex = str.index(str.startIndex, offsetBy: 2)
        let endIndex = str.index(str.endIndex, offsetBy: -1)
        let subStr1 = str.substring(to: startIndex)
        print(subStr1)

        let subStr2 = str.substring(from: endIndex)
        print(subStr2)

        let subStr3 = str.substring(with: startIndex..<endIndex)
        print(subStr3)

        //String 和  NSString 之间可以相互转换 可以转换为 NSString 来截取子串
        let NStr = (str as NSString).substring(with: NSRange(location: 1, length: 2))
        print(NStr)

集合

数组

//: [Int]
let numbers = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
for num in numbers {
    print(num)
}
let num1 = numbers[0]
let num2 = numbers[1]
let array = ["zhangsan", "lisi"]
//: 不能向不可变数组中追加内容
//array.append("wangwu")
var array1 = ["zhangsan", "lisi"]

//: 向可变数组中追加内容
array1.append("wangwu")
//: array1 仅允许追加 String 类型的值
//array1.append(18)

var array2: [Any] = ["zhangsan", 18]
//: 在 Swift 中,数字可以直接添加到集合,不需要再转换成 `NSNumber`
array2.append(100)

var array3: [String]
//: 实例化之前不允许添加值
//array3.append("laowang")
//: 实例化一个空的数组
array3 = [String]()
array3.append("laowang")
array3 += array1

//: 必须是相同类型的数组才能够合并,以下两句代码都是不允许的
//array3 += array2
//array2 += array3
//: 删除指定位置的元素
array3.removeAtIndex(3)
//: 清空数组
array3.removeAll()

字典

//: [String : Any] 是最常用的字典类型
var dict = ["name": "zhangsan", "age": 18]
//: * 如果 key 不存在,会设置新值
dict["title"] = "boss"
//: * 如果 key 存在,会覆盖现有值
dict["name"] = "lisi"
dict
//: 遍历
for (k, v) in dict {
    print("\(k) ~~~ \(v)")
}
//: 合并字典
var dict1 = [String: NSObject]()
dict1["nickname"] = "大老虎"
dict1["age"] = 100

//: 如果 key 不存在,会建立新值,否则会覆盖现有值
for (k, v) in dict1 {
    dict[k] = v
}
print(dict)

错误处理

需要了解

  1. 知道错误处理机制的三种方式
  2. 开发中通常使用第二种 try?
  3. 在 Swfit 3.0 中 Foundation 的部分类去除了 NS 前缀

代码演练,网络访问反序列化数据

let url = URL(string: "http://www.weather.com.cn/data/sk/101010100.html")!
URLSession.shared.dataTask(with: url, completionHandler: { (data, _, error) in
    if error != nil {
        print(error)
        return
    }
}).resume()
// 方式1:强try.加载失败直接崩溃
let dict = try! JSONSerialization.jsonObject(with: data!, options: [])
print(dict)

// 方式2:可选try.反序列化失败返回nil
let dict = try? JSONSerialization.jsonObject(with: data!, options: [])
print(dict)

// 方式3:默认try.返回序列化失败可以捕捉详细信息
do {
    let dict = try JSONSerialization.jsonObject(with: data!, options: [])
    print(dict)
}catch {
    print(error)
}

函数

目标

函数

目标

代码实现

//Swfit 3.0
func sum(a: Int, b: Int) -> Int {
    return a + b
}

let result = sum(a:10, b: 20)


//Swfit 2.0
func sum(a: Int, b: Int) -> Int {
    return a + b
}

let result = sum(10, b: 20)
func demo(str: String) -> Void {
    print(str)
}
func demo1(str: String) -> () {
    print(str)
}
func demo2(str: String) {
    print(str)
}

demo("hello")
demo1("hello world")
demo2("olleh")
func sum1(num1 a: Int, num2 b: Int) -> Int {
    return a + b
}

sum1(num1: 10, num2: 20)
func demo() {

        //该内部函数的范围只能在 demo函数中可以被访问
        func sum() {
            print("哈哈哈")
        }

        //必须先声明内部函数才能调用调用
        sum()

    }

函数格式小结

// 格式:func 函数名(形参1: 类型 = 默认值, _ 形参2: 类型 = 默认值...) -> 返回值 { // 代码实现 }
// 说明:包含默认值的函数可以不用传递,并且可以任意组合
//
// 格式:func 函数名(形参1: 类型, _ 形参2: 类型...) -> 返回值 { // 代码实现 }
// 说明:_ 可以忽略外部参数,与其他语言的函数风格更加类似
//
// 格式:func 函数名(外部参数1 形参1: 类型, 外部参数2 形参2: 类型...) -> 返回值 { // 代码实现 }
// 说明:外部参数名供外部调用使用,形参 在函数内部使用
//
// 格式:func 函数名(形参列表) -> 返回值 { // 代码实现 }

闭包

与 OC 中的 Block 类似,闭包主要用于异步操作执行完成后的代码回调,网络访问结果以参数的形式传递给调用方

目标

OC 中 Block 概念回顾

闭包的定义

    func demo1A() {
        func sum() {
            print("哈哈哈")
        }
        sum()

    }

//使用闭包实现
    func demo1B() {
        //定义代码块   () -> ()
        //没有参数没有返回值的闭包
        let closure = { () -> Void in
            print("OK")
        }

        //执行代码块
        closure()
    }
    func demo3A() {
        func sum(num1 a: Int, num2 b: Int) -> Int{
            return a + b
        }

        let result = sum(num1: 10, num2: 20)
        print(result)
    }
    func demo3B() {
        let closure = { (num1 a: Int, num2 b: Int ) -> Int in

            return a + b
        }


        let result = closure(num1: 100, num2: 200)
        print(result)
    }
let closure = {
    print("hello")
}

尾随闭包

闭包的三种简写形式只需要了解即可,在实际开发中做到看到不陌生即可,直接使用系统提示的类型

基本使用

GCD 异步

func loadData() {
     DispatchQueue.global().async {
            //假装耗时
            Thread.sleep(forTimeInterval: 2)
    }
}

自定义闭包参数,实现主线程回调

override func viewDidLoad() {
    super.viewDidLoad()

    loadData {
        print("完成回调")
    }
}

//异步请求网络数据没在主队列中回调 更新UI

    //TODO: 待会解释该关键词
    //@escaping
    func loadData(a: Int,finished: @escaping (String) -> () ) {
        DispatchQueue.global().async {
            //假装耗时
            Thread.sleep(forTimeInterval: 2)
            //获取结果
            let res = "老王"
            //在主队列中回调数据
            DispatchQueue.main.async {
                //回调数据  只负责请求数据 不负责更新UI
                //从外界传递闭包
                finished(res)
            }
        }
    }
override func viewDidLoad() {
    super.viewDidLoad()

    //定义一个有参数没有返回值的闭包当做参数传递
        let closure = { (res: String) -> () in
            print("我是请求结果: \(res)")
        }
        loadData(a:20,finished: closure)

    //使用尾随闭包的方式滴啊用
        loadData(a: 20) { (res) in
            print(res)
        }
}

循环引用

class NetworkTools: NSObject {

    /// 加载数据
    ///
    /// - parameter finished: 完成回调
    func loadData(finished: () -> ()) {
        print("开始加载数据...")

        // ...
        finished()
    }

    deinit {
        print("网络工具 88")
    }
}
class ViewController: UIViewController {

    var tools: NetworkTools?

    override func viewDidLoad() {
        super.viewDidLoad()

        tools = NetworkTools()
        tools?.loadData() {
            print("come here \(self.view)")
        }
    }

    /// 与 OC 中的 dealloc 类似,注意此函数没有()
    deinit {
        print("控制器 88")
    }
}

运行不会形成循环引用,因为 loadData 执行完毕后,就会释放对 self 的引用

/// 完成回调属性
var finishedCallBack: (()->())?

/// 加载数据
///
/// - parameter finished: 完成回调
func loadData(finished: () -> ()) {

    self.finishedCallBack = finished

    print("开始加载数据...")

    // ...
    working()
}

func working() {
    finishedCallBack?()
}

deinit {
    print("网络工具 88")
}

运行测试,会出现循环引用

解除循环引用

/// 类似于 OC 的解除引用
func demo() {
    weak var weakSelf = self
    tools?.loadData() {
        print("\(weakSelf?.view)")
    }
}
loadData { [weak self] in
    print("\(self?.view)")
}
loadData { [unowned self] in
    print("\(self.view)")
}

闭包(Block) 的循环引用小结

面相对象

目标

构造函数基础

构造函数是一种特殊的函数,主要用来在创建对象时初始化对象,为对象成员变量设置初始值,在 OC 中的构造函数是 initWithXXX,在 Swift 中由于支持函数重载,所有的构造函数都是 init

构造函数的作用

必选属性

class Person: NSObject {

    /// 姓名
    var name: String
    /// 年龄
    var age: Int
}

提示错误 Class 'Person' has no initializers -> 'Person' 类没有实例化器s

原因:如果一个类中定义了必选属性,必须通过构造函数为这些必选属性分配空间并且设置初始值

/// `重写`父类的构造函数
override init() {

}

提示错误 Property 'self.name' not initialized at implicitly generated super.init call -> 属性 'self.name' 没有在隐式生成的 super.init 调用前被初始化

/// `重写`父类的构造函数
override init() {
    super.init()
}

提示错误 Property 'self.name' not initialized at super.init call -> 属性 'self.name' 没有在 super.init 调用前被初始化

/// `重写`父类的构造函数
override init() {
    name = "张三"
    age = 18

    super.init()
}

小结

子类的构造函数

/// 学生类
class Student: Person {

    /// 学号
    var no: String

    override init() {
        no = "001"

        super.init()
    }
}

小结

Optional 属性

class Person: NSObject {
    /// 姓名
    var name: String?
    /// 年龄
    var age: Int?
}

重载构造函数

/// `重载`构造函数
///
/// - parameter name: 姓名
/// - parameter age:  年龄
///
/// - returns: Person 对象
init(name: String, age: Int) {
    self.name = name
    self.age = age

    super.init()
}

注意事项

调整子类的构造函数

/// `重写`父类构造函数
///
/// - parameter name: 姓名
/// - parameter age:  年龄
///
/// - returns: Student 对象
override init(name: String, age: Int) {
    no = "002"

    super.init(name: name, age: age)
}
/// `重载`构造函数
///
/// - parameter name: 姓名
/// - parameter age:  年龄
/// - parameter no:   学号
///
/// - returns: Student 对象
init(name: String, age: Int, no: String) {
    self.no = no

    super.init(name: name, age: age)
}

注意:如果是重载的构造函数,必须 super 以完成父类属性的初始化工作

重载重写

KVC 字典转模型构造函数

/// `重写`构造函数
///
/// - parameter dict: 字典
///
/// - returns: Person 对象
init(dict: [String: Any]) {
    setValuesForKeysWithDictionary(dict)
}
/// 个人模型
class Person: NSObject {

    /// 姓名
    var name: String?
    /// 年龄
    var age: Int?

    /// `重写`构造函数
    ///
    /// - parameter dict: 字典
    ///
    /// - returns: Person 对象
    init(dict: [String: AnyObject]) {
        super.init()

        setValuesForKeysWithDictionary(dict)
    }
}

运行测试,仍然会报错

错误信息:this class is not key value coding-compliant for the key age. -> 这个类的键值 age 与 键值编码不兼容

/// 姓名
var name: String?
/// 年龄
var age: Int? = 0

/// `重写`构造函数
///
/// - parameter dict: 字典
///
/// - returns: Person 对象
init(dict: [String: Any]) {
    super.init()

    setValuesForKeysWithDictionary(dict)
}

提示:在定义类时,基本数据类型属性一定要设置初始值,否则无法正常使用 KVC 设置数值

KVC 函数调用顺序

init(dict: [String: Any]) {
    super.init()

    setValuesForKeysWithDictionary(dict)
}

override func setValue(value: Any?, forKey key: String) {
    print("Key \(key) \(value)")

    super.setValue(value, forKey: key)
}

// `NSObject` 默认在发现没有定义的键值时,会抛出 `NSUndefinedKeyException` 异常
override func setValue(value: Any?, forUndefinedKey key: String) {
    print("UndefinedKey \(key) \(value)")
}

子类的 KVC 函数

/// 学生类
class Student: Person {

    /// 学号
    var no: String?
}

convenience 便利构造函数

特点

/// `便利构造函数`
///
/// - parameter name: 姓名
/// - parameter age:  年龄
///
/// - returns: Person 对象,如果年龄过小或者过大,返回 nil
convenience init?(name: String, age: Int) {
    if age < 20 || age > 100 {
        return nil
    }

    self.init(dict: ["name": name, "age": age])
}

注意:在 Xcode 中,输入 self.init 时没有智能提示

/// 学生类
class Student: Person {

    /// 学号
    var no: String?

    convenience init?(name: String, age: Int, no: String) {
        self.init(name: name, age: age)

        self.no = no
    }
}

便利构造函数应用场景

构造函数小结

懒加载

在 iOS 开发中,懒加载是无处不在的

lazy var person: Person = {
    print("懒加载")
    return Person()
}()
lazy var demoPerson: Person = Person()
上一篇下一篇

猜你喜欢

热点阅读