【Swift 3 && C++11】<第一部分> 概览(8): Swift 泛型 与 C++ 泛型

编程范式

|Swift|C++
:-:|:-:|:-:
有用的关键字和符号|<>, where|template,<>

先来介绍 Swift 的泛型.

Swift 中在尖括号里写一个名字来创建一个泛型函数或者类型:

func makeArray<Item>(repeating item: Item, numberOfTimes: Int) -> [Item] {
    var result = [Item]()
    for _ in 0..<numberOfTimes {
         result.append(item)
    }
    return result
}
makeArray(repeating: "knock", numberOfTimes:4)

你可以创建泛型函数, 方法, 类, 枚举和结构体:

enum OptionalValue<Wrapped> {
    case none
    case some(Wrapped)
}
var possibleInteger: OptionalValue<Int> = .none
possibleInteger = .some(100)

在类型名后面使用where来指定对类型的特殊需求, 比如, 限定类型实现某一个协议, 限定两个类型是相同的,或者限定某个类必须有一个特定的父类:

func anyCommonElements<T: Sequence, U: Sequence>(_ lhs: T, _ rhs: U) -> Bool
    where T.Iterator.Element: Equatable, T.Iterator.Element == U.Iterator.Element {
    for lhsItem in lhs {
        for rhsItem in rhs {
            if lhsItem == rhsItem {
                return true
            }
        }
    }
   return false
}
anyCommonElements([1, 2, 3], [3])

练习: 修改anyCommonElements(_:_:) 函数来创建一个函数,返回一个数组,内容是两个序列的共有元素。

注意, <T: Equatable><T where T: Equatable>是等价的.

** C++中**的泛型分为泛型函数和泛型类. 通过创建相应的模板实现: 函数模板和类模板.

类模板, 以 template开头, 尖括号中写类型参数名列表, 类型参数名以 classtypename开头, 再后面写类的定义:

#include <iostream>
using namespace std;

template <typename T>
class Class1 {
public:
    T doSomething(const T);
    T t;
};

int main() {

    Class1<int> class1;
    Class1<double> class2;

    return 0;
}

函数模板,和类模板类似:

#include <iostream>
using namespace std;

template <typename T>
void doSomething(T t) {
    cout << t+1 << endl;
}

int main() {

    doSomething(5);
    
    return 0;
}

注意到, 在使用函数模板的时候, 和使用一般函数一样doSomething(5);, 而使用类模板时候需要显式指定类型实参Class<int> class1;.

这是因为当我们使用函数模板来调用某一个具体的函数时, 编译器将为我们自动推断出模板参数,比如doSomething(5); 编译器将推断出类型参数Tint.

而使用类模板的时候, 就没有这样方便了, 我们必须在使用类模板的时候需要显式的指定模板实参.

另外在定义模板(不管是函数模板还是类模板)时, 尖括号中的模板参数列表,并非必须是需要classtypename开头标示的某一类型, 它们也可以是确定的类型,比如template<int N> ...:

#include <iostream>
using namespace std;

template <int T>
void doSomething() {
    cout << T << endl;
}

template <const char ch>
class Class {
    
public:
    void doSomething() {
        cout << ch << endl;
    }
};

int main() {


    doSomething<5>();
    
    Class<'k'> class1;
    class1.doSomething();
    
    return 0;
}

不过, 此时在使用模板的时候,尖括号的内容必须用常量表达式来提供给模板.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 本章将会介绍 泛型所解决的问题泛型函数类型参数命名类型参数泛型类型扩展一个泛型类型类型约束关联类型泛型 Where...
    寒桥阅读 648评论 0 2
  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 9,533评论 1 51
  • 136.泛型 泛型代码让你可以写出灵活,可重用的函数和类型,它们可以使用任何类型,受你定义的需求的约束。你可以写出...
    无沣阅读 1,508评论 0 4
  • 泛型的概念 泛型代码可根据自定义需求,写出适用于任何类型、灵活且可重用的函数和类型,避免重复的代码,用一种清晰和抽...
    伯wen阅读 408评论 0 2
  • 很多个日子以前,听了一场讲座分享,一个女孩子,有着小麦般健康的肤色,曼妙的身材,及腰的长发,说话的时候不急不缓,眼...
    旅行祈祷与爱阅读 672评论 0 3