>[**swift进阶总汇**](https://www.jianshu.com/p/c00fa675d7d5) 本文主要介绍泛型及其底层原理  ## 泛型 泛型主要用于解决`代码的抽象能力` + `代码的复用性` 例如下面的例子,其中的T就是泛型 ``` func test
(_ a: T, _ b: T)->Bool{ return a == b } //经典例子swap,使用泛型,可以满足不同类型参数的调用 func swap
(_ a: inout T, _ b: inout T){ let tmp = a a = b b = tmp } ``` ### 类型约束 在一个类型参数后面放置协议或者是类,例如下面的例子,要求类型`参数T遵循Equatable协议` ``` func test
(_ a: T, _ b: T)->Bool{ return a == b } ``` 当传入的参数是没有遵循`Equatable`协议时,会报错 ### 关联类型 在定义协议时,使用`关联类型`给`协议中用到的类型`起一个`占位符名称` * 此时的数组中的类型是Int ``` struct CJLStack { private var items = [Int]() mutating func push(_ item: Int){ items.append(item) } mutating func pop() -> Int?{ if items.isEmpty { return nil } return items.removeLast() } } ``` * 如果想使用其他类型呢?可以`通过协议来实现` ``` protocol CJLStackProtocol { //协议中使用类型的占位符 associatedtype Item } struct CJLStack: CJLStackProtocol{ //在使用时,需要指定具体的类型 typealias Item = Int private var items = [Item]() mutating func push(_ item: Item){ items.append(item) } mutating func pop() -> Item?{ if items.isEmpty { return nil } return items.removeLast() } } ``` ### where语句 where语句主要用于 表明`泛型需要满足的条件`,即限制形式参数的要求,如下所示 ``` //***********3、where语句:表明泛型需要满足的条件 protocol CJLStackProtocol { //协议中使用类型的占位符 associatedtype Item var itemCount: Int {get} mutating func pop() -> Item? func index(of index: Int) -> Item } struct CJLStack: CJLStackProtocol{ //在使用时,需要指定具体的类型 typealias Item = Int private var items = [Item]() var itemCount: Int{ get{ return items.count } } mutating func push(_ item: Item){ items.append(item) } mutating func pop() -> Item?{ if items.isEmpty { return nil } return items.removeLast() } func index(of index: Int) -> Item { return items[index] } } /* where语句 - T1.Item == T2.Item 表示T1和T2中的类型必须相等 - T1.Item: Equatable 表示T1的类型必须遵循Equatable协议,意味着T2也要遵循Equatable协议 */ func compare
(_ stack1: T1, _ stack2: T2) -> Bool where T1.Item == T2.Item, T1.Item: Equatable{ guard stack1.itemCount == stack2.itemCount else { return false } for i in 0..
Item? func index(of index: Int) -> Item } struct CJLStack: CJLStackProtocol{ //在使用时,需要指定具体的类型 typealias Item = Int private var items = [Item]() var itemCount: Int{ get{ return items.count } } mutating func push(_ item: Item){ items.append(item) } mutating func pop() -> Item?{ if items.isEmpty { return nil } return items.removeLast() } func index(of index: Int) -> Item { return items[index] } } extension CJLStackProtocol where Item: Equatable{} ``` * 当希望`泛型指定类型时拥有特定功能`,可以像下面这么写(在上述写法二的基础上增加extension) ``` //当希望泛型指定类型时拥有特定功能,可以像下面这么写 extension CJLStackProtocol where Item == Int{ func test(){ print("test") } } var s = CJLStack() s.test() test ``` * 如果将where后的Int改成Double类型,是无法找到test函数的 ## 泛型函数 我们在上面介绍了泛型的基本语法,下面来分析下泛型的底层原理 以下面一个简单的泛型函数为例 ``` //简单的泛型函数 func testGenric
(_ value: T) -> T{ let tmp = value return tmp } class CJLTeacher { var age: Int = 18 var name: String = "Kody" } //传入Int类型 testGenric(10) //传入元组 testGenric((10, 20)) //传入实例对象 testGenric(CJLTeacher()) ``` 从上面的代码中可以看出,泛型函数可以接受任何类型 疑问:那么泛型是如何区分不同的参数,来管理不同类型的内存呢? * 查看SIL代码,并没有什么内存相关的信息 * 查看IR代码,从中可以得出`VWT`中存放的是 `size`(大小)、`alignment`(对齐方式)、`stride`(步长)、`destory`、`copy`(函数) 所以VWT+PWT的存储结构图示如下所示 ### 源码分析 * 在swift-source中搜索`valueWitnesses`(在Metadata.h中) 对于每一个类型(Int或者自定义),都在metadata中存储了一个`VWT`(用来管理当前类型的值) * 继续来到`Metadataimpl.h`文件,查看其中的`元组`的源码 然后回到刚开始的泛型函数`testGenric` ``` func testGenric
(_ value: T) -> T{ //tmp在栈上申请空间,如何知道申请多大呢?可以通过metadata中存储的vwt得知 //copy let tmp = value //destory return tmp } ``` 其IR代码的详细分析如下 ``` ; Function Attrs: argmemonly nounwind willreturn 泛型函数 declare void @llvm.lifetime.start.p0i8(i64 immarg, i8* nocapture) #1 ; %swift.type* %T 表示 传入类型的matadata define hidden swiftcc void @"$s4main10testGenricyxxlF"(%swift.opaque* noalias nocapture sret %0, %swift.opaque* noalias nocapture %1, %swift.type* %T) #0 { entry: %T1 = alloca %swift.type*, align 8 %tmp.debug = alloca i8*, align 8 %2 = bitcast i8** %tmp.debug to i8* call void @llvm.memset.p0i8.i64(i8* align 8 %2, i8 0, i64 8, i1 false) store %swift.type* %T, %swift.type** %T1, align 8 %3 = bitcast %swift.type* %T to i8*** %4 = getelementptr inbounds i8**, i8*** %3, i64 -1 ; valueWitnesses 值目录表,将其存入了 %swift.vwtable* 中 %T.valueWitnesses = load i8**, i8*** %4, align 8, !invariant.load !46, !dereferenceable !47 ; 做了一个类型转换 %5 = bitcast i8** %T.valueWitnesses to %swift.vwtable* ; 在valueWitnesses中获取当前这个类型的size大小 %6 = getelementptr inbounds %swift.vwtable, %swift.vwtable* %5, i32 0, i32 8 %size = load i64, i64* %6, align 8, !invariant.load !46 ; 然后根据获取的size,分配内存空间 %7 = alloca i8, i64 %size, align 16 call void @llvm.lifetime.start.p0i8(i64 -1, i8* %7) %8 = bitcast i8* %7 to %swift.opaque* ; 初始化tmp的内存空间 store i8* %7, i8** %tmp.debug, align 8 %9 = getelementptr inbounds i8*, i8** %T.valueWitnesses, i32 2 %10 = load i8*, i8** %9, align 8, !invariant.load !46 ; copy 拷贝 %initializeWithCopy = bitcast i8* %10 to %swift.opaque* (%swift.opaque*, %swift.opaque*, %swift.type*)* %11 = call %swift.opaque* %initializeWithCopy(%swift.opaque* noalias %8, %swift.opaque* noalias %1, %swift.type* %T) #6 %12 = call %swift.opaque* %initializeWithCopy(%swift.opaque* noalias %0, %swift.opaque* noalias %8, %swift.type* %T) #6 %13 = getelementptr inbounds i8*, i8** %T.valueWitnesses, i32 1 %14 = load i8*, i8** %13, align 8, !invariant.load !46 ; destory 销毁 %destroy = bitcast i8* %14 to void (%swift.opaque*, %swift.type*)* call void %destroy(%swift.opaque* noalias %8, %swift.type* %T) #6 %15 = bitcast %swift.opaque* %8 to i8* call void @llvm.lifetime.end.p0i8(i64 -1, i8* %15) ret void } ``` 所以,从IR代码中可以得知,当前`泛型`是通过`ValueWitnessTable`来进行`内存操作`的 ### 源码调试 调试分为两种,`值类型`和`引用类型` #### 引用类型调试 * 源码调试如下 * 在`retain`函数中加断点调试 * 通过lldb调试如下:`obj`中存储`CJLTeacher变量` 结论:对于引用类型,会调用`retain`进行引用计数`+1`,对于`destory`来说,就会调用release进行引用计数`-1` * 泛型类型使用`VWT`进行`内存管理`,VWT由编译器生成,其存储了该类型的size、alignment以及针对该类型的基本内存操作 * 当对泛型类型进行内存操作时(例如:内存拷贝)时,最终会调用对应泛型的VWT中的基本内存操作 * 泛型类型不同,其对应的VWT也不同 #### 值类型调试 * 在`initializeWithTake`方法中加断点 结论:值类型是`通过当前内存的copy、move来进行内存拷贝`。对于`destory`,内部`调用析构函数` #### 总结 >作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS开发交流群:[130 595 548](https://jq.qq.com/?_wv=1027&k=L3kztZno),不管你是小白还是大牛都欢迎入驻 ,让我们一起进步,共同发展!(群内会免费提供一些群主收藏的免费学习书籍资料以及整理好的几百道面试题和答案文档!) * 对于一个`值类型`,例如Integer, * 1、该类型的`copy`和`move`操作会`进行内存拷贝`, * 2、`destory`操作则不进行任何操作 * 对于一个`引用类型`,如class, * 1、该类型的`copy`操作会对`引用计数+1`, * 2、`move`操作会`拷贝指针`,而不会更新引用计数; * 3、`destory`操作会对`引用计数-1` ### 泛型函数传入函数的分析 上面都是对变量进行的分析,那么一问来了 如果泛型函数中传的是一个函数呢? 代码如下所示,此时传入的`m`,是传入的整个结构体吗? ``` //如果此时传入的是一个函数呢? func makeIncrement() -> (Int) -> Int{ var runningTotal = 10 return { runningTotal += $0 return runningTotal } } func testGenric
(_ value: T){} //m中存储的是一个结构体:{i8*, swift type *} let m = makeIncrement() testGenric(m) ``` * 分析IR代码 ``` define i32 @main(i32 %0, i8** %1) #0 { entry: %2 = alloca %swift.function, align 8 %3 = bitcast i8** %1 to i8* ; s4main13makeIncrementS2icyF 调用makeIncrement函数,返回一个结构体 {函数调用地址, 捕获值的内存地址} %4 = call swiftcc { i8*, %swift.refcounted* } @"$s4main13makeIncrementS2icyF"() ; 闭包表达式的地址 %5 = extractvalue { i8*, %swift.refcounted* } %4, 0 ; 捕获值的引用类型 %6 = extractvalue { i8*, %swift.refcounted* } %4, 1 ; 往m变量地址中存值 ; 将 %5 存入 swift.function*结构体中(%swift.function = type { i8*, %swift.refcounted* }) ; s4main1myS2icvp ==> main.m : (Swift.Int) -> Swift.Int,即全局变量 m store i8* %5, i8** getelementptr inbounds (%swift.function, %swift.function* @"$s4main1myS2icvp", i32 0, i32 0), align 8 ; 将值放入 f 这个变量中,并强转为指针 store %swift.refcounted* %6, %swift.refcounted** getelementptr inbounds (%swift.function, %swift.function* @"$s4main1myS2icvp", i32 0, i32 1), align 8 ; 将%2 强转为 i8*(即 void*) %7 = bitcast %swift.function* %2 to i8* call void @llvm.lifetime.start.p0i8(i64 16, i8* %7) ; 取出 function中 闭包表达式的地址 %8 = load i8*, i8** getelementptr inbounds (%swift.function, %swift.function* @"$s4main1myS2icvp", i32 0, i32 0), align 8 %9 = load %swift.refcounted*, %swift.refcounted** getelementptr inbounds (%swift.function, %swift.function* @"$s4main1myS2icvp", i32 0, i32 1), align 8 ; 将返回的闭包表达式 当做一个参数传入 方法,所以 retainCount+1 %10 = call %swift.refcounted* @swift_retain(%swift.refcounted* returned %9) #2 ; 创建了一个对象,存储了 <{ %swift.refcounted, %swift.function }>* %11 = call noalias %swift.refcounted* @swift_allocObject(%swift.type* getelementptr inbounds (%swift.full_boxmetadata, %swift.full_boxmetadata* @metadata, i32 0, i32 2), i64 32, i64 7) #2 ; 将 %swift.refcounted* %11 强转成了一个结构体类型 %12 = bitcast %swift.refcounted* %11 to <{ %swift.refcounted, %swift.function }>* ; 取出 %swift.function (最终的结果就是往 <{ %swift.refcounted, %swift.function }> 的%swift.function 中存值 ==> 做了间接的转换与传递) %13 = getelementptr inbounds <{ %swift.refcounted, %swift.function }>, <{ %swift.refcounted, %swift.function }>* %12, i32 0, i32 1 ; 取出
的首地址 %.fn = getelementptr inbounds %swift.function, %swift.function* %13, i32 0, i32 0 ; 将 i8* 放入 i8** %.fn 中(即创建的数据结构 <{ %swift.refcounted, %swift.function }> 的 %swift.function 中) store i8* %8, i8** %.fn, align 8 %.data = getelementptr inbounds %swift.function, %swift.function* %13, i32 0, i32 1 store %swift.refcounted* %9, %swift.refcounted** %.data, align 8 %.fn1 = getelementptr inbounds %swift.function, %swift.function* %2, i32 0, i32 0 ; 将 %swift.refcounted 存入 %swift.function 中 store i8* bitcast (void (%TSi*, %TSi*, %swift.refcounted*)* @"$sS2iIegyd_S2iIegnr_TRTA" to i8*), i8** %.fn1, align 8 %.data2 = getelementptr inbounds %swift.function, %swift.function* %2, i32 0, i32 1 store %swift.refcounted* %11, %swift.refcounted** %.data2, align 8 ; 将%2强转成了 %swift.opaque* 类型,其中 %2 就是 %swift.function内存空间,即存储的东西(函数地址 + 捕获值地址) %14 = bitcast %swift.function* %2 to %swift.opaque* ; sS2icMD ==> demangling cache variable for type metadata for (Swift.Int) -> Swift.Int 即函数的metadata %15 = call %swift.type* @__swift_instantiateConcreteTypeFromMangledName({ i32, i32 }* @"$sS2icMD") #9 ; 调用 testGenric 函数 call swiftcc void @"$s4main10testGenricyyxlF"(%swift.opaque* noalias nocapture %14, %swift.type* %15) ...... ``` 仿写泛型函数传入函数时的底层结构 仿写上述逻辑的结构 ``` //如果此时传入的是一个函数呢? struct HeapObject { var type: UnsafeRawPointer var refCount1: UInt32 var refCount2: UInt32 } struct FunctionData
{ var ptr: UnsafeRawPointer var captureValue: UnsafePointer
} struct Box
{ var refCounted: HeapObject var value: T } struct GenData
{ var ref: HeapObject var function: FunctionData
} func makeIncrement() -> (Int) -> Int{ var runningTotal = 10 return { runningTotal += $0 return runningTotal } } func testGenric
(_ value: T){ //查看T的存储 let ptr = UnsafeMutablePointer
.allocate(capacity: 1) ptr.initialize(to: value) /* - 将 %13的值给了 %2即 %swift.function* %13 = getelementptr inbounds <{ %swift.refcounted, %swift.function }>, <{ %swift.refcounted, %swift.function }>* %12, i32 0, i32 1 - 调用方法 %14 -> %2 %14 = bitcast %swift.function* %2 to %swift.opaque* call swiftcc void @"$s4main10testGenricyyxlF"(%swift.opaque* noalias nocapture %14, %swift.type* %15) */ let ctx = ptr.withMemoryRebound(to: FunctionData
>>.self, capacity: 1) { $0.pointee.captureValue.pointee.function.captureValue } print(ctx.pointee.value)//捕获的值是10 } //m中存储的是一个结构体:{i8*, swift type *} let m = makeIncrement() testGenric(m) 10 ``` 所以当是一个泛型函数传递过程中,会做一层包装,意味着并不会直接的将`m`中的函数值、type给`testGenric`函数,而是`做了一层抽象`,目的是解决不同类型在传递过程中的问题 ## 总结 >作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS开发交流群:[130 595 548](https://jq.qq.com/?_wv=1027&k=L3kztZno),不管你是小白还是大牛都欢迎入驻 ,让我们一起进步,共同发展!(群内会免费提供一些群主收藏的免费学习书籍资料以及整理好的几百道面试题和答案文档!) * 泛型主要用于解决`代码的抽象能力`,以及提升`代码的复用性` * 如果一个`泛型`遵循了某个协议,则在使用时,要求具体的类型也是必须遵循某个协议的 * 在定义`协议`时,可以使用`关联类型`给`协议中用到的类型`起一个`占位符名称` * `where`语句主要用于表明`泛型需要满足的条件`,即限制形式参数的要求 * 泛型类型使用`VWT`进行`内存管理`(即通过VWT区分不同类型),VWT由编译器生成,其存储了该类型的size、alignment以及针对该类型的基本内存操作 * 1、当对泛型类型进行内存操作时(例如:内存拷贝)时,最终会调用对应泛型的VWT中的基本内存操作 * 2、泛型类型不同,其对应的VWT也不同 * 当`希望泛型指定类型时拥有特定功能`,可以通过`extension`实现 * 对于泛型函数来说,有以下几种情况: * 1、该类型的`copy`操作会对`引用计数+1`, * 2、`move`操作会`拷贝指针`,而不会更新引用计数; * 3、`destory`操作会对`引用计数-1` * 1、该类型的`copy`和`move`操作会`进行内存拷贝`, * 2、`destory`操作则不进行任何操作 * 传入的是一个`值类型`,例如Integer, * 传入的是一个`引用类型`,如class, * 如果`泛型函数`传入的是一个`函数`,在传递过程中,会做一层包装,简单来说,就是不会直接将函数的`函数值+type`给泛型函数,而是做了一层抽象,主要是用于`解决不同类型的传递问题`
微信关注我们
原文链接:https://blog.51cto.com/u_15146321/2904185
转载内容版权归作者及来源网站所有!
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
相关文章
发表评论
资源下载
更多资源优质分享App
近一个月的开发和优化,本站点的第一个app全新上线。该app采用极致压缩,本体才4.36MB。系统里面做了大量数据访问、缓存优化。方便用户在手机上查看文章。后续会推出HarmonyOS的适配版本。
Oracle
Oracle Database,又名Oracle RDBMS,或简称Oracle。是甲骨文公司的一款关系数据库管理系统。它是在数据库领域一直处于领先地位的产品。可以说Oracle数据库系统是目前世界上流行的关系数据库管理系统,系统可移植性好、使用方便、功能强,适用于各类大、中、小、微机环境。它是一种高效率、可靠性好的、适应高吞吐量的数据库方案。
Apache Tomcat
Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目,由Apache、Sun 和其他一些公司及个人共同开发而成。因为Tomcat 技术先进、性能稳定,而且免费,因而深受Java 爱好者的喜爱并得到了部分软件开发商的认可,成为目前比较流行的Web 应用服务器。
Eclipse
Eclipse 是一个开放源代码的、基于Java的可扩展开发平台。就其本身而言,它只是一个框架和一组服务,用于通过插件组件构建开发环境。幸运的是,Eclipse 附带了一个标准的插件集,包括Java开发工具(Java Development Kit,JDK)。