初探Swift,Swift协议,Swift接口,它是一种约定,实现这个接口的类必须要实现这个接口中所定义的所有方法。协议(Protocol),同PHP的接口(interface)。

/**

 * 协议(Protocol),同PHP的接口(interface)

 */

protocol MyProtocol{

    //变量可读可写

    var myName:String {

        get set

    }

    //定义变量,只读

    var myAge:Int{

        get

    }

    func getMyName()->String

}

//继承的父类一定要写在最前面,然后用,分割,后面跟协议,多个协议用,分割

class MyClass : MyProtocol{

    var m_name:String = ""

    var myName:String {

        get{

            return "lane"

        }set{

            m_name = newValue

        }

    }

    var myAge:Int{

        get{

            return 24

        }

    }

    

    func getMyName()->String{

        return "hello Lane"

    }

}


/**

* 协议类型

*/

var varProtocol:MyProtocol = MyClass()

varProtocol.myName


/**

* 协议继承

* 要实现子协议和父协议的全部内容才可以

*/

protocol ChildMyProtocol:MyProtocol{

    func getMyAge()->Int

}

class MyClass2 : ChildMyProtocol{

    var m_name:String = ""

    var myName:String {

        get{

            return "lane"

        }set{

            m_name = newValue

        }

    }

    var myAge:Int{

        get{

            return 24

        }

    }

    

    func getMyName()->String{

        return "hello Lane"

    }

    

    func getMyAge() -> Int {

        return 20;

    }

}


/**

* 协议合并

*/

func hello(s:protocol<MyProtocol, ChildMyProtocol>)


    初探Swift,Swift结构体是一种数据类型,与C语言的结构体类型,成员可以是各种数据类型,甚至是函数。Swift泛型是广泛的类型,即函数的参数中一个变量声明为Int,它也可以接受String并处理。


    一、结构体,声明与使用:

/**

 * 结构体

 */

//声明一个结构体,结构体带有默认值

struct People {

    var name = "小明"

    var age = 0

    func getKeyNumber()->Int{

        return age

    }

}

var xiaoming = People()

xiaoming.getKeyNumber()

var lane = People(name: "Lane", age: 24)

lane.getKeyNumber()


    二、泛型:在一个函数中,比如在交换两个变量值的函数中,声明时定义了Int型,那如果我想交换两个String时,怎么办?

/**

* 泛型

* 比如在交换两个变量值的函数中,声明时定义了Int型,那如果我想交换两个String时,怎么办?

*/

func swapValue(inout a:Int, inout b:Int){

    let temp = a

    a = b

    b = temp

}

//引入泛型

func newSwapValue<T>(inout a:T, inout b:T){

    let temp = a

    a = b

    b = temp

}

//交换两个String

var a = "hello"

var b = "world"

a

b

newSwapValue(&a, &b)

a

b

//交换两个Int

var c = 1

var d = 2

c

d

newSwapValue(&c, &d)

c

d


/**

* 类中定义泛型

*/

class MyT<T>{

    func getName(s:T)->Void{           

        println(s)

    }

}


var myTObj = MyT<Int>()

myTObj.getName(123)

var myTObj2 = MyT<String>()

myTObj2.getName("Hello world")

    初探Swift,Swift类,Swift类的实例化,Swift类的操作等是本文的内容。这是一门强类型、完全面相对象的语言。类可以更好的封装、抽象、复用模块和代码,使得代码更加的优雅简洁。

/**

 * 类

 */

//定义一个类

class TV{

    //定义属性

    var price = 1999

    var name = "乐TV"

    //定义方法

    func open()->String{

        return "Success"

    }

    func close()->Bool{

        return true

    }

}

//创建对象,不需要new这个关键字的

var myTV = TV()

//调用类的属性

println(myTV.name)

//调用类的方法

myTV.open()

myTV.close()

//类的对象是值调用还是引用调用。

var myTV2 = myTV

//改变myTV2的价格

myTV2.price = 2000

//打印myTV2的价格,是2000,修改成功

myTV2.price

//打印myTV1的价格,是2000,说明是引用传递的

myTV.price


//可以用“===”来判断使用来自同一个类

myTV === myTV2


//swift是没有指针概念的:因为虽然引用传递用的还是指针,但是swift并没有给我们指针的语法


/**

 * 类的构造函数、析构函数

 */

class People{

    var name:String = "小明"

}

var peopleObj = People()

//结果是小明

peopleObj.name

//增加析构函数后

class People2{

    var name:String = "小明"

    //析构函数,系统特殊函数,不需要加func

    init(name:String){

        self.name = name

    }

    //析构函数,系统特殊函数,不需要加func和()

    deinit {

        name = ""

    }

}

var peopleObj2 = People2(name: "小红")

peopleObj2.name



/**

 * 类的继承

 */

//学生类继承人类类

class Student : People{}

var studentObj = Student()

studentObj.name


class Student2 : People2{

    override init(name:String){

        //父类的构造函数需要手动调用,不会自动执行

        super.init(name: "小红")

        super.name = "hello: \(super.name)"

        self.name = name

    }

    //重写父类的函数需要加关键词override

    

    //重写属性

    override var name:String{

        get{

            return super.name

        }

        set{

            super.name = "hi \(name)"

        }

    }

    

    //不希望被重写,再父类的方法前面加@final。可以加在方法、属性、设置类前面

}

var studentObj2 = Student2(name: "小白")

println(studentObj2.name)

studentObj2.name = "lane"

println(studentObj2.name)

    初探Swift,Swift枚举。Swift枚举不同于C语言的枚举,C的枚举只能关联一个整形,而Swift枚举不但可以关联一个整形,更可以关联任意的数据类型。

    声明用enum开头,case 后面跟一个元素。可以不像C语言那样和整数关联。如下,ExamScore不是一个变量,而是一个数据类型,这个数据类型叫做ExamScore,同Int,String一样。

import UIKit

/**

 * 枚举(Enumerations)

 */

//声明用enum开头,case 后面跟一个元素。可以不像C语言那样和整数关联。如下,ExamScore不是一个变量,而是一个数据类型,这个数据类型叫做ExamScore,同Int,String一样。

enum ExamScore{

    case score_a

    case score_b

    case score_c

}

var myScore:Int = 40

//声明一个变量是刚才声明的ExamScore类型

var rank:ExamScore

if myScore >= 90 {

    //使用枚举:“枚举变量.枚举元素”

    rank = ExamScore.score_a

}else if myScore >= 60 {

    //使用枚举:枚举变量可以省略,只要“.枚举元素”

    rank = .score_b

}else{

    rank = .score_c

}

rank

switch rank{

    case .score_a:

        println("A等成绩")

    case .score_b:

        println("B等成绩")

    case .score_c:

        println("C等成绩")

}


/**

 * 枚举 - 关联一个值

 */

enum Month:Int{

    case Jan=1, Feb, Mar

}

//声明了Jan=1,Feb和Mar系统会自动赋值为2和3

//使用

//此时值为1

Month.Jan.rawValue

//另一种好似用方式

var myMonth:Month

myMonth = .Jan

myMonth.rawValue

//寻找关联值为2的那个枚举元素,返回一个可选型,因为如果寻找1000的那个元素,是找不到。

var nextMonth = Month(rawValue: 2)

nextMonth!.rawValue

//不仅仅,还可以关联字符串

enum MonthString:String{

    case Jan="一月"

    case Feb="二月"

    case Mar="三月"

}

//一月

MonthString.Jan.rawValue



/**

 * 复杂的枚举

 */

enum UserInfo{

    case birthday(Int, Int, Int)

    case name(String)

}

let xiaomingBirthday = UserInfo.birthday(1990, 1, 10)

let xiaomingName:UserInfo = .name("小明")


switch xiaomingBirthday{

    case .birthday(let year, let month, let day):

        println("\(year)年\(month)月\(day)")

    case .name(let name):

        println("大家号,我叫\(name)")

}


    初探Swift,本文关于Swift闭包、Swift闭包特性(比如结尾闭包和捕获参数)、Swift闭包引用类型的相关介绍与实践。swift闭包一个很重要的特性,可以简化代码,优化流程。使代码更加简洁。

    一、闭包(Closure),将函数的声明和逻辑代码都放在参数的位置,闭包的本质还是函数。

import UIKit

/**

 *闭包(Closure),将函数的声明和逻辑代码都放在参数的位置,闭包的本质还是函数。

 */

var arr = [2, 1, 5, 3, 9, 0, 4]

//使用闭包将数组从大到小排列,sorted不穿第二个参数时默认为从下到大,闭包放在{}中,用()来声明2个变量,->来声明返回值,后面跟一个in,逻辑代码放在in后面。

var result = sorted(arr, {(a:Int, b:Int)->Bool in

    return a > b

})

result

//闭包的简化操作,闭包中2个参数的类型可以省略,返回值声明可以省略

result = sorted(arr, {a, b in return a > b})

result

//闭包进一步简化,省略return

result = sorted(arr, {a, b in a>b})

result

//闭包再进一步优化,省略a,b两个参数声明,用$0, $1来代替,省略in关键字

result = sorted(arr, {$0 > $1})

//再省,因为“>”就是函数,是运算符函数

result = sorted(arr, >)

//注:必须是单行的才可以省略return, in已经ab的声明

result

    二、结尾闭包,闭包的参数里面{},看起来别扭,而且开发时可能会出现括号的对应问题。当传入闭包是最后一个参数的时候,可以使用如下的方式。仅仅是最后一个的时候可以!

/**

 *结尾闭包,闭包的参数里面{},看起来别扭,而且开发时可能会出现括号的对应问题。当传入闭包是最后一个参数的时候,可以使用如下的方式。仅仅是最后一个的时候可以!

 */

result = sorted(arr){a, b in

    return a > b

}

result

    三、捕获参数(capturing values)我们可以在闭包中使用闭包外的变量

/**

 *捕获参数(capturing values)我们可以在闭包中使用闭包外的变量

 */

var num = 3

result = sorted(arr){

    return fabs(Float($0-num)) < fabs(Float($1-num))

}

result

    四、引用类型:将函数(闭包)赋值给函数类型的变量A,函数类型的变量A再赋值给函数类型的变量B,那么,A和B是指向同一块地址的,调用A引起的变化再变量B中将会体现

/**

 *引用类型

 */

func calcTotalMiles(todayMiles:Int)->()->Int{

    var totalMiles = 0

    return {totalMiles += todayMiles; return totalMiles;}

}

var dailyTwoMiles = calcTotalMiles(2)

//结果:2

dailyTwoMiles()

//结果:4

dailyTwoMiles()

//把函数复制给另一个变量

varmyPlan =dailyTwoMiles

//调用这个变量

//结果:6

myPlan()

//结果:8

myPlan()

//回过头来调用之前被赋值的变量(函数类型的变量),看值是否改变

//结果:10

dailyTwoMiles()

//说明这个是引用传递的,函数类型的变量myPlan和函数类型的变量dailyTwoMiles是指向同一个地址的。