https://blog.csdn.net/weixin_34054931/article/details/88027728

swift 可以定义模板函数,如:

func testFunc<T>(datas: [T]) -> T{

//处理 

}

复制代码

这里有个T,使用指代类型的,这个方法定义出来,可以用来处理任意的数组:

 

let names = [String]()

testFunc(names)
let names2 = [Int]()

testFunc(names)

复制代码

传入String的数组,T就是String;传入Int的数组,T就是Int.这个方法就像是一个模板,有了它,可以复刻出许多个不同的版本。

问题1:我想要这个T是具有某个特定方法的

举个常用的例子:

比如写一个用来找最小值的方法,func min<T>(datas: [T]) -> T?,我肯定是希望它能处理数字、字符串、日期,甚至是自定义类的对象,那就需要这个被排序的数组里的对象,必须得实现一个比较的函数, 这个我才能知道那个大那个小,才能排序。而且我只需要这个比较函数就可以对所有类通用了。

比如可以写成:

 

func min<T>(datas: [T]) -> T?{

  if datas.count == 0 {
    return nil

  }

  var min: T = datas.first!

  for data in datas {

    if data.compareTo(min) < 0{

      min = data

    }

  }

  return min

}

里面用了compareTo方法。我要怎么确定这个T一定是有这个方法的呢?

祭出法宝:protocol

联想:一些人困惑协议有什么用?跟block/闭包有什么区别?这里的功能就是闭包无法替代协议的,其实协议和委托是可以不一起行动的。

定义一个协议:

 

protocol Comparable : NSObjectProtocol{

/**
和其他对象比较

- parameter other: 其他对象

- returns: 0 相等 -1 小于 1 大于

*/

func compareTo(other: Self) -> Int

}

复制代码

然后把方法定义修改为:

 

func min<T:Comparable>(datas: [T]) -> T?

复制代码

T后面限定了类型,指定了这个T是遵循类Comparable这个协议的,那么也就具有了compareTo这个方法。在我理解里,协议的核心就在这:表明某个类具有特定的方法/能力

问题2:多个协议怎么办?

假如我想处理的T类型是需要具有两个不同的能力,举个现实的例子:一个榨汁机,它接收的东西应该同时具有可被碾碎和出水两个特性,这两个特性是分开的,因为饼干不出水和椰子碾不碎。对应到代码,可悲碾碎是一个protocol里的一个方法,会出水是另一个protocol的方法。

多个协议只需要写成:

 

func min<T:protocol<testProtocol1,testProtocol2>>(datas: [T]) -> T?

复制代码

把两个协议用protocol关键字装起来就好了。

问题3:如果我还想这个T是某个特定的类呢?

比如我想T是class1这个类的对象,同时遵循testProtocol1和testProtocol2,怎么写?

祭出法宝:where关键字

方法写成:

 

func findMinTemplateFunc<T : testCalss

where T :protocol<testProtocol1,testProtocol2>>(datas: [T]) -> T?
复制代码

把协议的限定方法where里面去,where还有其他用法,我也没太用过,就不说了。

这个需求看起来好像很难发生,但只需要想一个东西就有了:抽象类。swift/OC里没强掉这个概念,但是这个东西是存在的,比如CoreData里面的NSManageredObject,你会直接使用这个类来构造对象吗?肯定不会,肯定要建自己的数据实体,也就是NSManageredObject的子类来操作了。

当有了抽象类做父类的时候,你处理的都是子类,如果你写一个针对子类的模板方法,有些子类实现了testProtocol1,有些实现了testProtocol2,有些没有。这时,就必须类、协议同时限定才能达到效果。

最后贴段例子:

加入找出数组里最小值,每个值根据value1 \ value2 和rate做一段算法后的值来比较:

 

protocol testProtocol1: NSObjectProtocol {

    func value1() -> Int;
}

protocol testProtocol2: NSObjectProtocol {

func value2() -> Int;

}

//类似虚类的东西,比如NSManagedObject这种类,是不可能直接使用它来构建对象的,肯定是要配合自己建的CoreData实体

class testCalss: NSObject {

var rate: Int? = 1

}

func findMinTemplateFunc<T : testCalss where T :protocol<testProtocol1,testProtocol2>>(datas: [T]) -> T?{

    if datas.count == 0 {

      return nil

    }

var min : T = datas.first!

var minRealValue = min.value1() * 10 + min.value2() * 100

  if let rate = min.rate {

minRealValue *= minRealValue * rate

}

  for data in datas {

var realValue = data.value1() * 10 + data.value2() * 100

    if let rate = data.rate {

   realValue *= realValue * rate

   }

  if realValue < minRealValue {

min = data

minRealValue = realValue

}

}

return min

}

//例子

class subClass1: testCalss, testProtocol1, testProtocol2 {

func value1() -> Int {

    return Int(arc4random() % 10)

}

func value2() -> Int {

    return Int(arc4random() % 20)

}

}

class subClass2: testCalss, testProtocol1, testProtocol2 {

func value1() -> Int {

    return Int(arc4random() % 100)

}

func value2() -> Int {

    return Int(arc4random() % 200)

}

}

func runTest(){

var array1 = [subClass1]()

    for _ in 0...99 {

    array1.append(subClass1())

    }

findMinTemplateFunc(array1)

   var array2 = [subClass2]()

    for _ in 0...99 {

   array2.append(subClass2())

   }

findMinTemplateFunc(array1)

}

runTest()

Swift泛型定义 同时限定T的类(class)和多协议(protocol)的更多相关文章

  1. Swift泛型协议的N种用法

    They said "you should learn a new language every year," so I  learned Swift. Now  I  learn ...

  2. Java泛型全解析【接口、类、封装类型】

    目录   1.导读  2.为何需要泛型?   3.泛型的定义格式   3.泛型的好处  4.什么时候使用泛型?   5.泛型的擦除   6.泛型的补偿  7.泛型的应用      7.1[泛型类]   ...

  3. 一文详解scala泛型及类型限定

    今天知识星球球友,微信问浪尖了一个spark源码阅读中的类型限定问题.这个在spark源码很多处出现,所以今天浪尖就整理一下scala类型限定的内容.希望对大家有帮助. scala类型参数要点 1. ...

  4. Swift 泛型和闭包结合使用

    通常在Swift中定义一个闭包来使用 typealias Closure= (Any?) -> () var tempClosure :Closure? /// 定义一个方法直接调用 func ...

  5. SpringCloud系列三:SpringSecurity 安全访问(配置安全验证、服务消费端处理、无状态 Session 配置、定义公共安全配置程序类)

    1.概念:SpringSecurity 安全访问 2.具体内容 所有的 Rest 服务最终都是暴露在公网上的,也就是说如果你的 Rest 服务属于一些你自己公司的私人业务,这样的结果会直接 导致你信息 ...

  6. C++之enum枚举量声明、定义、使用与枚举类详解

    C++之enum枚举量声明.定义.使用与枚举类详解 学习一个东西,首先应该指导它能做什么,其次去知道它怎么去做,最后知道为什么去这么做. 知其然知其所以然.不能冒进 ,一步一步的慢慢来.

  7. C++定义一个简单的Computer类

    /*定义一个简单的Computer类 有数据成员芯片(cpu).内存(ram).光驱(cdrom)等等, 有两个公有成员函数run.stop.cpu为CPU类的一个对象, ram为RAM类的一个对象, ...

  8. django-自定义文件上传存储类

    文件储存API:https://yiyibooks.cn/xx/django_182/ref/files/storage.html 编写自定义存储系统:https://yiyibooks.cn/xx/ ...

  9. 系统整理 精讲 swift 泛型

    泛型是一种非常领会的语法,让我很是膜拜! 真是让人又爱又恨,学不懂的时候很抓狂 允许程序在函数,枚举,结构体,类中定义类型形参(类型可以动态改变) 每次使用可以传入不同类型的形参! Array< ...

随机推荐

  1. php 增删改查范例(3)

    编辑页面edit.php: <?php$id=$_GET['id'];$db= new mysqli('localhost','root','root','db_0808');$sql=&quo ...

  2. python词云图与中文分词

    2019-12-12中文文本分词和词云图具体功能介绍与学习代码: import jiebaa="由于中文文本的单词不是通过空格或者标点符号来进行分割"#jieba.lcut()s是 ...

  3. 吴裕雄 Bootstrap 前端框架开发——Bootstrap 表单:表单控件状态

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  4. java反射初探

    java反射 反射是java的重要特性之一,java.lang.reflect 是jdk支持反射的重要包,我下面可能会对构造器Constructor,属性Filed,方法Method会用到.反射其实很 ...

  5. XML中报错

    错误描述如下: Multiple annotations found at this line: - cvc-complex-type.2.4.a: Invalid content was found ...

  6. spring事务传播属性和隔离级别

    猫咪咪的Java世界 spring事务传播属性和隔离级别 博客分类: Spring java编程   1 事务的传播属性(Propagation) 1) REQUIRED ,这个是默认的属性 Supp ...

  7. 学习 Ansible Playbook,有这篇文章就够了!

    https://mp.weixin.qq.com/s?__biz=MzAwNTM5Njk3Mw==&mid=2247487361&idx=1&sn=b50327df2949e4 ...

  8. Linux关于文件处理命令

    一.登陆用户和机器名称 示例:[root@hadoop01 ~]# root:表示用户名 @hadoop01表示机器名称 ~表示当前文件目录是家目录 #表示输入命令提示符,用户可以在其后输入命令:非r ...

  9. 二 SVN代码冲突的解决

    问题: A和B都是最新的代码,A修改了代码提交了,B也修改了代码,但是B提交的时候出现冲突的问题. 解决方案:编辑冲突 解决冲突: 方法一:将文件里面冲突的描述去掉,重新提交 方法二:软件解决冲突

  10. SciPy fftpack(傅里叶变换)

    章节 SciPy 介绍 SciPy 安装 SciPy 基础功能 SciPy 特殊函数 SciPy k均值聚类 SciPy 常量 SciPy fftpack(傅里叶变换) SciPy 积分 SciPy ...