swift协议,内存处理,视图,泛型

协议

协议是方法的集合(计算属性相当于方法)
可以把看似不相关的对象的公共行为放到一个协议里

协议的三种作用

1.能力-遵循了协议就意味着具备了某种能力
2.约定-遵循了协议就一定要能实现协议中的方法
3.角色-一个类可以遵循多个协议,一个协议也可以被多个类遵循,遵循协议就意味着扮演某种角色,遵循多个协议就以为着可以扮演多种角色

说明:Swift中的继承是单一继承(一个类只能有一个父类),如果希望一个类具备多种能力可以使用多个协议来实现。(c++里面是通过多重继承来实现的,并不好)

依赖倒转原则(面向协议编程)

1.声明变量的类型时应尽可能使用协议类型
2.声明方法参数类型时应尽可能使用协议类型
3.声明方法返回类型时应尽可能使用协议类型

开闭原则

1.抽象关键:在设计系统的时候一定要设计好模式
2.封装可变性:桥梁模式-将不同可变因素封装到不同的继承结构中

接口隔离,协议隔离原则

协议的设计要小而专不要大而全
协议的设计也要高度内聚

复合开闭原则

协议中全是抽象概念(只有声明没有实现)遵循协议的类可以各自对协议中的计算属性和方法给出自己的实现版本,这样当我们面向协议编程时就可以把多态的优势发挥到淋漓尽致,可以写出更通用更灵活的代码

例子:打折协议

/**
* 打折策略协议
*/

protocol DiscountStrategy {
    
    
    /**
     - parameter price:原价
     - returns:折扣的金额
    */
    
    
    
    
    func discount(price:Double) ->Double
    


}

///百分比折扣策略
class PercentageDiscount:DiscountStrategy {
    var percentage:Double
    
    init(percentage:Double){
        self.percentage = percentage
    }
    
    
    func discount(price:Double)->Double{
        return price*(1-percentage)
    }
}


class FixedDiscount:DiscountStrategy {
    var fixedMoney:Double
    
    init(fixedMoney:Double){
        self.fixedMoney = fixedMoney
        
    }
    
    func discount(price: Double) -> Double {
        return price >= fixedMoney ? fixedMoney:0
    }
}

class SegmentedDiscount:DiscountStrategy {
    
    func discount(price: Double) -> Double {
        if price < 20{
            return 0
        }else if price<50{
            return 3
        }else if price < 100 {
            return 10
        }else{
            return 30
        }
    }
}
//图书
class Book {
    var name:String
    var price:Double
    var type:String
    //四人帮设计模式-策略模式
    var strategy:DiscountStrategy?
    /**
     初始化方法
     - parameter name: 书名
     - parameter price:间隔
     - parameter type:类型
    */
    init(name:String,price:Double,type:String){
        self.name = name
        self.price = price
        self.type = type
        
    }
 /// 减了多少钱
    var discountValue:Double{
        get{
            if let s = strategy{
                return s.discount(price)
            }else{
                return 0
            }
        }
    }
    
    ///折扣后的价格
    var discountedPrice:Double{
        get{return price - discountValue}
let booksArray = [
    Book(name: "C语言程序设计", price: 24.0, type: "计算机"),
Book(name: "名侦探柯南", price: 98.5, type: "漫画"),
Book(name: "Swift从入门到精通", price: 35.8, type: "计算机"),
Book(name: "黄冈教学密集卷", price: 34.2, type: "教材"),
Book(name: "中国股市探秘", price: 58.5, type: "金融")]


let discountDict:[String:DiscountStrategy] = [
    "计算机" : PercentageDiscount(percentage:0.78),
    "教材" : PercentageDiscount(percentage:0.85),
    "漫画" : SegmentedDiscount(),
    "科普" : FixedDiscount(fixedMoney:2)
]

var totalprice = 0.0
var totalDiscount = 0.0

for book in booksArray {
    if let strategy = discountDict[book.type]{
        book.strategy = strategy
    }
    
    print("《\(book.name)》原价:¥\(book.price)元")
    print("《\(book.name)》折扣后:¥\(book.discountedPrice)")
    totalprice += book.discountedPrice
    totalDiscount += book.discountValue
}

print(String(format:"总计:¥%.1f元",totalprice))
print(String(format:"总计:¥%.1f元",totalDiscount))

计算机的硬件构成

运算器,控制器,存储器,输入设备,输出设备
运算器+存储器=cpu(中央处理器)
存储器=内存(Ram - random access memory)

程序员可以使用的内存

1.堆(Stack)- 我们创建的对象都是放在堆上的
2.栈(heap)- 我们定义的局部变量临时都是放在栈上的
3.静态区域(static area)

  • 数据段--全局量
  • 只读数据段--常量
  • 代码段--函数和方法

值类型和引用类型的区别

1.结构的对象是值类型,类型的对象是引用类型

值类型在赋值的时候会在内存中进行对象拷贝
引用类型在赋值时的时候不会进行对象拷贝只是能加一个引用
2.结构中会自动生成初始化方法
3.结构中方法默认情况下是不允许改变结构中的属性的,除非加上Mutating关键字

自动内存处理机制

class Person {
    var name :String
    var age:Int
   // 指派构造器(designated)
    init(name:String,age:Int){
      print("创建人")
        self.name = name
        self.age = age
    }
    //将构造器转化为必要构造器
    //子类里必须遵循父类构造
//     required init(name:String,age:Int){
//        print("创建人")
//        self.name = name
//        self.age = age
//    }

    //默认赋值,便利构造器(convenience)
    //只能横向调用,不能跨类
    convenience init(){
        //下面的语句必须写在调用自己的初始化方法之后否则MAJOR属性会被附上不正确的值
        self.init(name:"无名",age:20)
    }
    
    
    deinit{
        print("销毁人")
    }
}

class Student:Person {
    var major:String
//    //重写
//    override init(){
//        self.major = major
//        print("创建学生")
//    }
    init(name:String,age:Int,major:String){
        //子类只能调用直接父类的构造器
        //只能调用非便利构造器(指派构造器)
    self.major = major//1初始化自己特有的属性
        super.init(name: name, age: age)//2初始化二阶段
        //此处可以调用对象方法,已完成初始化
    }
    
    func study(){
        
    }
    //释放时不需要传参
    deinit{
        print("销毁学生")
    }
}

class Teacher: Person {
//    override init(){
//        print("创建老师")
//    }
    override init(name:String,age:Int){
        super.init(name: name, age: age)
    }
    deinit{
        print("销毁老师")
    }
}

////创建一个学生对象,然后用stu去引用它,所以此时学生对象引用计数为1
////只要有一个对象引用,就不会删除,销毁(及时性销毁优于andrid)
//var stu:Student? = Student()
////取消全部引用就会销毁
////stu=nil
////此处没有创建新的学生对象,原来的学生对象的引用计数+1
//var stu2 = stu
////同上,原来的学生对象的引用计数+1
//var stu3 = stu2
////学生对象引用计数-1
//stu=nil
////学生对象引用计数-1
//stu2=nil
////学生对象引用计数-1
//stu3=nil
////当学生对象引用计数为0时ARC会自动清理内存释放学生对象
////ARC及时性的内存清理 优于JAVA中Garbage collection(垃圾回收)

//var stu1:Student? = Student()
////弱引用,虽然引用,但不会增加引用计数 默认是强引用(会增加引用计数)
//weak var stu2 = stu1
//
//weak var stu3 = stu2
//
//stu1=nil
////如果想释放内存,释放内存最好方式是手动将一个引用赋值为nil

//而这里是全局变量,函数调用不会使其消失
//var stu:Student?
//
//func foo(){
////    //局部变量,函数调用结束后局部变量就消失了
//    //所以学生对象的引用计数也就变成0了,所以会被ARC释放掉
////    let stu = Student()
//    print(stu)
//}
//
//foo()

//栈 -FILO 先进后出结构
//创建任何子类对象的时候一定是先创建父类对象
//var stu:Person = Student()
////如果一个引用转移了(会导致原来对象的引用计数-1 新对象引用计数+1)
////栈存储,先进后出
//stu = Teacher()
//stu = Person()


//超出自动释放池不会起作用
//var stu:Student?
//swift自动释放池
//通过向AUTOreleasepool函数中传入一个闭包来实现
//autoreleasepool { () -> () in
//    //自动释放池中的对象引用在池的边界引用计数会自动-1的消息
//    //将来做IOS开发时,如果某个地方会创建很多的临时对象
//    //那么最好在此处设置一个自动释放池避免内存顺时峰值过高
//    let stu = Student()
//    let stu2 = stu
//}
//// 离开自动释放池时,Stu会受到引用计数减1消息,Stu2也会受到引用计数减1得消息
//OC用法
//@autoreleasepool


//双向关联关系
//如果程序中出现了类与类之间双向关联,必须要将其中一端设置成weak引用,否
//则将会形成循环引用,导致ARC无法释放内存
    class Emp {
        //推荐使用
        //如果允许可控类型通常用WEAK来破除循环引用
        //如果员工关联的部门对象被释放了那么dept会被赋值为NIL
        //如果要继续给dept发消息不会崩溃
        //weak var dept:Dept?
        //谨慎使用
        //如果不允许使用可空类型就必须使用UNOWNED来破除循环引用
        //需要注意的是如果员工对象关联的部门对象被释放了
        //如果还要通过员工对象操作它所关联的部门对象将导致程序崩溃
        //错误提示EXE_BAD_ACCESS
    var dept:Dept?
    
    init() {
        print("创建一个员工")
    }
    
    deinit{
        print("销毁员工")
    }
    
}

class Dept {
    weak var manager : Emp?
    init(){
        print("创建一个部门")
    }
    deinit{
        print("销毁部门")
    }
}

//func bar() {
//    _ = Person()
//    let emp = Emp()
//    let dept = Dept()
//    //引用计数无限,破除环可解除环,方法WEAK
//    emp.dept = dept
//    dept.manager = emp
//}
//
//bar()

////视图闭包
//var x :(Int,Int)->Bool = {
//    [weak self](x,y) -> Bool in
//    return true
//}

泛型

//通用版 类型为虚拟类型(泛型)
func mySwap<T>(inout a:T,inout _ b:T){
    let temp = a
    a=b
    b=temp
}
//swift自动推断类型
var a="字符",b="坠马"
mySwap(&a, &b)
print(a,b)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容