大咖助阵 海纳:聊聊语言中的类型系统与泛型 你好,我是海纳,是极客时间《编程高手必学的内存知识》专栏的作者。

我们知道,编程语言中有非常重要的一个概念,就是数据类型。类型的概念伴随着我们学习一门具体语言的全过程,也深入到了程序员的日常开发之中。所以对于现代程序员而言,了解语言中的类型系统是一项非常重要的技能。

这一节课,我会简单地介绍什么是类型,类型的作用,以及由简单类型推导来的泛型编程的基本概念,接着再比较C++和Java两种语言的泛型实现。很多新的编程语言的泛型实现都有它们的影子,所以了解C++和Java泛型,会有助于你理解泛型设计的基本概念。

通过这节课的学习,你会得到一种新的学习语言的视角,那就是从类型的角度去进行分析。

比如我们在学习一门新的语言的时候,可以考虑以下几个问题:

  • 这门语言是强类型的吗?
  • 这门语言是动态类型吗?
  • 它支持多少种内建类型呢?
  • 它支持结构体吗?
  • 它支持字典(Recorder)吗?
  • 它支持泛型吗?
  • ……

这样,当我们拿到一门新的语言的规范(Specification)文档后,就可以带着这些问题去文档中寻找答案。等你把这些问题搞明白了,语言的很多特性也就掌握了。这是很多优秀程序员可以短时间内掌握一门新语言的秘技之一。

接下来,我们就从类型的基本概念开始讲起。

什么是类型

编程语言中的变量都是有类型的,而且变量的类型不一定一致。例如Go语言中的int和float声明的变量,它们的类型就不一致,如果你直接对它们执行加操作,Go的编译器就会报错,很多隐式类型转换带来的问题,在编译阶段就可以发现了。比如,你可以看下面这个例子: func main() { var a int = 1 var b float64 = 1000.0 fmt.Print(a + b) }

这种情况下,Go的编译器会报这样的错误:invalid operation: a + b (mismatched types int and float64)。这就说明Go语言不支持整型和浮点型变量的加操作。

相比Go语言,JavaScript在类型上的要求就宽松很多,比如整数与字符串的加法操作,JavaScript会把整数转换成字符串,然后再与目标字符串进行拼接操作。显然Go语言会对语言类型进行严格检查,我们就说它的类型强度高于JavaScript。

Go语言的类型系统还有一个特点,那就是一个变量声明成什么类型的,就不能再更改了。与之形成鲜明对比的是Python。它们都具有比较高的类型强度,但是类型检查的时机不同。Go是在编译期,而Python则是在运行期。我们看一个Python的例子:

a = 1 »> a + “hello” Traceback (most recent call last):   File “", line 1, in TypeError: unsupported operand type(s) for +: 'int' and 'str' >>> a = "hello" >>> a + "world" 'helloworld'

从上面的例子中,我们可以看到,Python的类型检查也是比较严格的,在第2行,将一个整型值和字符串值相加是会引发Traceback的。但是一个变量却可以先使用整数为它赋值,再使用字符串为它赋值(第6行),然后再进行字符串值的加操作就没有问题了。这就说明了变量的类型是在程序运行的时候才去检查,而不是编译期间进行的。我们把这种语言称为动态类型语言

同样的例子如果使用Go来实现,编译器就会报错: var a int = 1 a = “hello” fmt.Print(a)

由此可见,Go语言是一种静态的强类型语言

动态类型不仅仅表现在变量的类型可以更改,在面向对象的编程语言中,动态类型往往还意味着类的定义也可以动态更改。我们仍然以Python为例来观察动态类型的特点:

class A(): …  pass … »> A.a = 1 »> a = A() »> a.a 1

从上述例子中,我们可以看到,类A的类属性是可以在运行时进行添加和修改的。这与静态编译的语言非常不同。

由此,我们可以得出结论,动态类型相比静态类型,它的优点在于:

  • 动态类型有更好的灵活性,在运行时可以修改变量的类型,也可以对类定义进行修改,所以针对动态类型语言的热更新就更容易设计;
  • 动态类型语言写起来很方便,非常适合用来编写小规模的脚本。

同时,动态类型也往往具有一些缺点(通常是这样,但并不绝对)。

首先,动态类型语言的代码不容易阅读。据统计,程序员的日常工作中90%的时间是在阅读别人写的代码,只有不足10%的时间才是在开发新的功能。而动态类型语言,没有类型标注,代码会非常难懂,即使有一些动态类型语言有类型标注的,但因为可以运行时修改类型,往往会出现一个类的属性在不同的地方被修改的情况,这使得代码的阅读和维护变得困难;

第二点是,动态类型语言的性能往往会差一些,比如Python和JavaScript,因为在编译期间缺少类型提示,编译器无法为对象安排合理的内存布局(你可以参考内存课导学三),所以它们的对象布局相比Java/C++等静态类型语言会更加复杂,同时这也会带来性能的下降。

由此可见,我们并不能简单地说,静态类型就比动态类型好,或者强类型就比弱类型好,还是要根据具体的场景来进行取舍。

比如要求快速开发,规模较小的工具,人们常常会选择使用Python,而多人合作的大型项目,人们就会选择使用Java之类的静态强类型语言。

另外,类似int、 String这种类型往往是语言的内建类型,而语言的内建类型在表达力上经常是不够的,这就需要人们通过将简单内建类型组合起来实现相应的功能,这就是复合类型

典型的复合类型包括枚举、结构、列表、字典等。这些类型在Go语言中都有相应的定义,你可以参考Go语言专栏进行学习。

在讲完了类型的基本概念以后,我们再讲解一个类型系统中非常常见,同时也是比较困难的一个话题,那就是泛型。

为什么要使用泛型

我们使用一个实际的例子来讲一下为什么要使用泛型。比如,这里有一个栈的C++实现,栈里可以存放的变量是整型的,它的代码如下所示: /#include using namespace std; class Stack { private:    int _size;    int _top;    int/* _array; public:    Stack(int n) {       _size = n;       _top = 0;       _array = new int[_size];    }    void push(int t) {       if (_top < _size) {          _array[_top++] = t;       }    }    int pop() {       if (_top > 0) {          return _array[--_top];       }       return -1;    } };   int main() {    Stack stack(3);    stack.push(1);    stack.push(2);    cout << stack.pop() + stack.pop() << endl;    return 0; }

运行这个程序,一切看上去都还不错。但是,假如我们需要一个管理浮点数栈,或者管理字符串的栈,就不得不再将上述逻辑重新实现一遍。除了_array的类型不一样之外,整数栈和浮点数栈的逻辑都是相同的。这就会带来大量的重复代码,不利于工程代码的维护。

为了解决这个问题,很多带有类型的语言都引入了泛型。以C++为例,泛型的栈可以这么实现: /#include using namespace std; template class Stack { private:    int _size;    int _top;    T/* _array; public:    Stack(int n) {       _size = n;       _top = 0;       _array = new T[_size];    }    void push(T t) {       if (_top < _size) {          _array[_top++] = t;       }    }    T pop() {       if (_top > 0) {          return _array[--_top];       }       return T();    } };   int main() {    Stack stack(3);    stack.push(1);    stack.push(2);    cout << stack.pop() + stack.pop() << endl;    Stack sstack(3);    sstack.push("hello ");    sstack.push("world!");    cout << sstack.pop() + sstack.pop() << endl;    return 0; }

执行这段代码,可以看到,控制台上可以成功打印出来3和”hello world”。

这段代码的巧妙之处在于,栈的核心逻辑,我们只写了一遍(第4行至第30行),然后只需要使用一行简单的代码就可以创建用于存储整数的栈(第33行)和用于存储字符串的栈(第37行)。

使用这种方式,可以帮助我们节约大量的时间和代码篇幅。接下来,我们看一下C++编译器是如何处理这段代码的。在Linux系统上,我们可以使用以下命令对这个文件进行编译,然后查看它的编译结果: $ g++ -o stack -g stack.cpp $ objdump -d stack

从这个结果中,可以看到,C++编译器生成了两个push方法,其参数类型分别是整型和字符串类型。也就是说,在C++中,泛型类型在被翻译成机器码的时候,是真的创建了两种不同的类型。

泛型使用最广泛的场景就是容量类,例如vector、list、map等等。C++ STL中定义的容器类都是以模板的形式提供的。

我们可以再使用一种新的视角来理解泛型,那就是可以将泛型声明看作是类型之间的转换关系,或者换种说法,就是我们可以使用一种类型(甚至是值)得到另外一种新的类型。

泛型:使用类型得到新的类型

现在我们就用这个新视角来理解泛型,把泛型声明看成是一个输入参数是类型,返回值也为类型的函数。我使用一个vector的例子来说明这一点: int main() { vector vi; vector vd; vector/* p = &v1; return 0; }

这里程序的第4行会报错,报错的信息显示“vector”并不是一个有效的类型。而“vector ”和“vector”则是有效的类型。从这个例子中,我们观察到,vector类型必须指定一个类型参数才能变成一个有效的类型。所以我们可以把

template class vector;

看成是一个函数,它接受一个类型int或者double,得到一种新的类型vector,或者vector。

在C++中,更神奇的是,泛型的类型参数不仅仅可以是一种类型,还可以是一个具体的值,例如: template class A; int main() { A<0> a; A<1> b; return 0; }

在上述代码中,A<0>和A<1>分别是两个不同的类型。使用这种办法,我们可以在编译期间通过模板让编译器帮我们做一些计算,例如:

/#include using namespace std; template struct fib {    static const int v = fib::v + fib::v; }; template <> struct fib<1> {    static const int v = 1; }; template <> struct fib<0> {    static const int v = 1; }; int main() {   cout << fib<10>::v << endl;   return 0; }

在这个例子中,编译以后的结果,fib<10>::v会被直接替换成55。这个计算的过程是由编译器完成的。

编译器会把fib<10>,fib<9>等等都看成一种类型。当编译器要计算fib<10>的值的时候就会先求解fib<9>和fib<8>的值,这样一直递归下去,就会找到fib<1>和fib<0>这里。而这两个值,我们已经提供了(第9行到第18行),递归就会结束。

在这个例子中,我们就看到了类型依赖于值的情况。

了解了C++的泛型设计以后,我们再来看一下Java语言的泛型实现。

Java中的泛型实现

Java语言的库的分发,往往采用这种形式:Java的源代码会先被翻译成字节码文件,然后这些文件又会被打包进jar文件。jar文件可以在网络上进行发布。

Java的一个特性是,相同的字节码文件在不同的体系结构和平台上的行为都是相同的,再加上要做到对低版本代码的兼容,所以Java的泛型设计和C++的差异就很大。总的来说,Java的泛型设计是使用了一种叫做“泛型擦除”的办法来实现的。

我举一个例子来说明“泛型擦除”是怎么一回事。请看下面的代码: import java.util.ArrayList; class Playground {     public static void main(String[ ] args) {         ArrayList int_list = new ArrayList();         ArrayList str_list = new ArrayList();         System.out.println(int_list.getClass() == str_list.getClass());     } }

这段代码的输出是true。

如果按照上一小节中关于C++泛型的实现,ArrayList和ArrayList应该是不同的两种类型。但这里的结果却是true,这是因为Java会把这两种ArrayList的泛型都擦除掉,从而导致整个程序中只有一种类型。

我们这里再举一个例子帮你理解一下Java的泛型: import java.util.ArrayList; class Playground {     public static void main(String[ ] args) {         System.out.println(“Hello World”);     }     public static void sayHello(ArrayList list) {     }     public static void sayHello(ArrayList list) {     } }

这里,第8行定义的sayHello方法和第12行定义的sayHello方法是方法重载。我们知道,方法的重载的基本条件是两个同名方法的参数列表并不相同。

从字面上看,第一个sayHello方法的参数类型是ArrayList,第二个方法的参数类型是ArrayList,所以可以实现方法的重载。但是当我们尝试编译上述程序的时候,却会得到这样的错误提示: Playground.java:12: error: name clash: sayHello(ArrayList) and sayHello(ArrayList) have the same erasure public static void sayHello(ArrayList list) { ^ 1 error

这是因为当对泛型进行擦除以后,两个sayHello方法的参数类型都变成了ArrayList,从而变成了同名方法,所以就会出现命名冲突报错。

通过上面两个例子,我们就能感觉到C++泛型和Java泛型的不同之处了。它们之间最核心的区别是C++不同的泛型参数会得到一种新的类型;而Java则不会,它会进行类型擦除,从而导致表面上不同的类型参数实际上指代的是同一种类型。

总结

在这节课里,我们先了解到什么是类型系统,并介绍了什么是强类型和弱类型,什么是静态类型和动态类型。然后我们通过举例来说明Python,JavaScript,Go和C++各自的类型系统的特点。

从这些例子中,我们看到静态强类型语言更容易阅读和维护,但灵活性不如动态弱类型语言。所以动态弱类型语言往往都是脚本语言,不太适合构建大型程序。

接下来,我们简单介绍了泛型的概念。我们使用了一个栈的例子来说明了使用泛型可以提高编程效率,节省代码量。Go语言从1.18开始也支持泛型编程。

然后我们又提供了一个新的视角来理解泛型,这种新的视角是把泛型类看成是一种函数,它的输入参数可以是类型,也可以是值,它的返回值是一种新的类型。

最后,我们介绍了C++的泛型实现和Java的泛型实现。C++不同的泛型参数会得到一种新的类型,这个过程我们也会称它为泛型的实例化。而Java则会进行类型擦除,从而导致表面上不同的类型参数实际上指代的是同一种类型。

参考资料

https://learn.lianglianglee.com/%e4%b8%93%e6%a0%8f/Tony%20Bai%20%c2%b7%20Go%e8%af%ad%e8%a8%80%e7%ac%ac%e4%b8%80%e8%af%be/%e5%a4%a7%e5%92%96%e5%8a%a9%e9%98%b5%20%e6%b5%b7%e7%ba%b3%ef%bc%9a%e8%81%8a%e8%81%8a%e8%af%ad%e8%a8%80%e4%b8%ad%e7%9a%84%e7%b1%bb%e5%9e%8b%e7%b3%bb%e7%bb%9f%e4%b8%8e%e6%b3%9b%e5%9e%8b.md