Swift泛型Protocol对比C#泛型Interface
本篇纯属抬杠之作,之前我们提到了Swift的泛型Protocol使用associatedtype关键字,而不是使用<Type>语法的泛型参数。这其中有什么好处呢?
我就这个问题搜索了一些回答,大体上提到两点:
<Type>语法对Protocol没有意义,Protocol仅需要定义一个抽象的概念,具体的类型应该由实现的Class来明确,比如:
ClassWithInt<Int>: NumberProtocol
ClassWithDouble<Double>: NumberProtocol
associatedtype可以用来给Protocol中特定Func添加泛型约束,而不是限定整个Protocol
protocol GeneratorType {
associatedtype Element
public mutating func next() -> Self.Element?
}
听上去还是有一定道理的,然后实践是检验事实的唯一标准。下面我们通过代码实例来和C#进行对比。首先拿出网上多被引用解释上述两个观点的Swift代码:
public protocol Automobile {
associatedtype FuelType
associatedtype ExhaustType
func drive(fuel: FuelType) -> ExhaustType
}
public protocol Fuel {
associatedtype ExhaustType
func consume() -> ExhaustType
}
public protocol Exhaust {
init()
func emit()
}
public struct UnleadedGasoline<E: Exhaust>: Fuel {
public func consume() -> E {
print("...consuming unleaded gas...")
return E()
}
}
public struct CleanExhaust: Exhaust {
public init() {}
public func emit() {
print("...this is some clean exhaust...")
}
}
public class Car<F: Fuel,E: Exhaust>: Automobile where F.ExhaustType == E {
public func drive(fuel: F) -> E {
return fuel.consume()
}
}
public class Car1<F: Fuel>: Automobile {
public func drive(fuel: F) -> F.ExhaustType {
return fuel.consume()
}
}
具体的使用情况如下:
var car = Car<UnleadedGasoline<CleanExhaust>, CleanExhaust>()
car.drive(fuel: UnleadedGasoline<CleanExhaust>()).emit() var fusion = Car1<UnleadedGasoline<CleanExhaust>>()
fusion.drive(fuel: UnleadedGasoline<CleanExhaust>()).emit()
转换成C#代码的话,有两种思路,首先是把泛型参数放到Interface层面:
public interface Automobile<FuelType, ExhaustType>
{
ExhaustType Drive(FuelType fuel);
}
public interface Fuel<ExhaustType>
{
ExhaustType consume();
}
public interface Exhaust
{
void Emit();
} public class UnleadedGasoline<Exhaust> : Fuel<Exhaust> where Exhaust : new()
{
public Exhaust consume()
{
Console.WriteLine("...consuming unleaded gas...");
return new Exhaust();
}
}
public class CleanExhaust : Exhaust
{
public void Emit()
{
Console.WriteLine("...this is some clean exhaust...");
}
}
public class Car : Automobile<UnleadedGasoline<CleanExhaust>, CleanExhaust>
{
public CleanExhaust Drive(UnleadedGasoline<CleanExhaust> fuel)
{
return fuel.consume();
}
}
还可以模仿Swift对Automobile多做一层继承进行包装:
public interface Car1<T1> : Automobile<UnleadedGasoline<T1>, T1> where T1 : new()
{ } public class SimpleCar : Car1<CleanExhaust>
{
public CleanExhaust Drive(UnleadedGasoline<CleanExhaust> fuel)
{
return fuel.consume();
}
}
调用的时候没有什么太大的差别:
var gaso = new UnleadedGasoline<CleanExhaust>();
var car = new Car();
car.Drive(gaso).Emit(); var simpleCar = new SimpleCar();
simpleCar.Drive(gaso).Emit();
和Swift比较不同的是,我们在Interface就代入了泛型参数。但是由于我们不能直接实例化Interface,所以并不能直接使用Automobile来减少一层继承关系。
因为上述提到的使用associatedtype 的第一点理由见仁见智,这里不分高下。
C#还有第二种思路,就是我也把泛型约束下放到Func层级:
public interface Automobile
{
ExhaustType Drive<FuelType,ExhaustType>(FuelType fuel) where ExhaustType : new();
}
public interface Fuel
{
ExhaustType consume<ExhaustType>() where ExhaustType : new();
} public class UnleadedGasoline : Fuel
{
public Exhaust consume<Exhaust>() where Exhaust : new()
{
Console.WriteLine("...consuming unleaded gas...");
return new Exhaust();
}
} public class Car2 : Automobile
{
public CleanExhaust Drive<UnleadedGasoline, CleanExhaust>(UnleadedGasoline fuel) where CleanExhaust : new()
{
return (fuel as Fuel).consume<CleanExhaust>();
}
}
C#的接口并不能定义构造函数。强行模仿起来还真是有点累啊。最终的使用也很简单:
var fuel = new UnleadedGasoline();
var car2 = new Car2();
car2.Drive<UnleadedGasoline,CleanExhaust>(fuel).Emit();
通篇比较下来,应该说Swift通过associatedtype 关键字和<Type>的混用,使得泛型的定义更为复杂也更灵活了。
GitHub:
https://github.com/manupstairs/LearnSwift
https://github.com/manupstairs/LearnDotNetCore
Swift泛型Protocol对比C#泛型Interface的更多相关文章
- swift:入门知识之泛型
在尖括号里写一个名字来创建一个泛型函数或者类型 例如<T>.<Type> 可以创建泛型类.枚举和结构体 在类型后使用where来指定一个需求列表.例如,要限定实现一个协议的类型 ...
- Java泛型和集合之泛型介绍
在声明一个接口和类的时候可以使用尖括号带有一个或者多个参数但是当你在声明属于一个接口或者类的变量的时候或者你在创建一个类实例的时候需要提供他们的具体类型.我们来看下下面这个例子 List<Str ...
- Java基础----Java---集合框架---泛型、泛型方法、静态方法泛型、泛型接口、泛型限定、泛型类
泛型:jdk1.5后的新特性,用于解决安全问题,是一个安全机制. 好处: 1.将运行时的异常出现问题classcastException.转移到了编译时期.方便程序员调试解决问题,让运行事情问题减少, ...
- Java泛型三:Java泛型详解
原文地址https://www.cnblogs.com/lzq198754/p/5780426.html 1.为什么需要泛型 泛型在Java中有很重要的地位,网上很多文章罗列各种理论,不便于理解,本篇 ...
- Java泛型四:Java泛型总结
原文地址https://www.cnblogs.com/lwbqqyumidi/p/3837629.html 一. 泛型概念的提出(为什么需要泛型)? 首先,我们看下下面这段简短的代码: 1 publ ...
- 牛客网Java刷题知识点之泛型概念的提出、什么是泛型、泛型在集合中的应用、泛型类、泛型方法、泛型接口、泛型限定上限、泛型限定下限、 什么时候使用上限?泛型限定通配符的体现
不多说,直接上干货! 先来看个泛型概念提出的背景的例子. GenericDemo.java package zhouls.bigdata.DataFeatureSelection; import ja ...
- java 反射和泛型-反射来获取泛型信息
通过指定对应的Class对象,程序可以获得该类里面所有的Field,不管该Field使用private 方法public.获得Field对象后都可以使用getType()来获取其类型. Class&l ...
- Java 泛型 协变式覆盖和泛型重载
Java 泛型 协变式覆盖和泛型重载 @author ixenos 1.协变式覆盖(Override) 在JDK 1.4及以前,子类方法如果要覆盖超类的某个方法,必须具有完全相同的方法签名,包括返回值 ...
- java泛型基础、子类泛型不能转换成父类泛型
参考http://how2j.cn/k/generic/generic-generic/373.html 1.使用泛型的好处:泛型的用法是在容器后面添加<Type>Type可以是类,抽象类 ...
随机推荐
- java 调用 C# 类库搞定,三步即可,可以调用任何类及方法,很简单,非常爽啊
java 调用 C# 类库搞定,三步即可,可以调用任何类及方法,很简单,非常爽啊 java 调用 C# 类库搞定,可以调用任何类及方法,很简单,非常爽啊 总体分三步走: 一.准备一个 C# 类库 (d ...
- Xamarin.Android之UI Test简单入门
一.前言 相信Xamarin免费之后会有更多的人加入进来,这也是我一直以来最希望看到的事,更多的人加入到这个社区中,为这个社区贡献自己的一份力量,国内当前还没有一个比较正规或者说是名气比较大的Xama ...
- Mono+Jexus部署Web页面
一.web页面 二.Jexus默认站点的配置文件(只需修改站点路径) ###################### # Web Site: Default ###################### ...
- 开始VS 2012 中LightSwitch系列的第1部分:表中有什么?描述你的数据
[原文发表地址] Beginning LightSwitch in VS 2012 Part 1: What’s in a Table? Describing Your Data [原文发表时间] ...
- 可在广域网部署运行的QQ高仿版 -- GG叽叽V2.4,增加远程协助、桌面共享功能(源码)
QQ的远程协助.或者说桌面共享是一个非常实用的功能,所以,2.4版本的GG复制了它,而且,GG增强了桌面共享的功能,它可以允许指定要共享桌面的区域,这样,对方就只能看到指定区域的桌面,这对节省流量会非 ...
- Spring-Context之八:一些依赖注入的小技巧
Spring框架在依赖注入方面是非常灵活和强大的,多了解点一些注入的方式.方法,绝对能优化配置. idref idref属性可以传入一个bean的名称,虽然它是指向一个bean的引用,但是得到的是该b ...
- 细说.NET中的多线程 (四 使用锁进行同步)
通过锁来实现同步 排它锁主要用来保证,在一段时间内,只有一个线程可以访问某一段代码.两种主要类型的排它锁是lock和Mutex.Lock和Mutex相比构造起来更方便,运行的也更快.但是Mutex可以 ...
- C# 对包含文件或目录路径信息的 System.String 实例执行操作
在字符串操作中有一类比较特殊的操作,就是对包含文件或目录路径信息的 System.String 实例执行操作.比如根据一个表示路径的字符串获取其代表的文件名称.文件夹路径.文件扩展名等.在很多时候,我 ...
- NodeJS系列~第三个小例子,NodeJs与Redis实现高并发的队列存储
返回目录 众所周知 redis量个强大的缓存组件,可以部署在win32和linux环境之上,它有五大存储结构,其中有一种为列表list,它可以实现quene和stack的功能,即队列和堆栈的功能. r ...
- php做登录注册页面及加载
//SQL注入攻击 //1.过滤用户的输入 //2.使用预处理语句 //3.写代码的时候尽量避免 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1. ...