You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

363 lines
16 KiB
Markdown

2 years ago
# 大咖助阵|海纳:聊聊语言中的类型系统与泛型
你好,我是海纳,是极客时间[《编程高手必学的内存知识》](https://time.geekbang.org/column/intro/100094901?tab=comment)专栏的作者。
我们知道,编程语言中有非常重要的一个概念,就是数据类型。类型的概念伴随着我们学习一门具体语言的全过程,也深入到了程序员的日常开发之中。所以对于现代程序员而言,了解语言中的类型系统是一项非常重要的技能。
这一节课我会简单地介绍什么是类型类型的作用以及由简单类型推导来的泛型编程的基本概念接着再比较C++和Java两种语言的泛型实现。很多新的编程语言的泛型实现都有它们的影子所以了解C++和Java泛型会有助于你理解泛型设计的基本概念。
通过这节课的学习,你会得到一种新的学习语言的视角,那就是从类型的角度去进行分析。
比如我们在学习一门新的语言的时候,可以考虑以下几个问题:
1. 这门语言是强类型的吗?
2. 这门语言是动态类型吗?
3. 它支持多少种内建类型呢?
4. 它支持结构体吗?
5. 它支持字典(Recorder)吗?
6. 它支持泛型吗?
7. ……
这样当我们拿到一门新的语言的规范Specification文档后就可以带着这些问题去文档中寻找答案。等你把这些问题搞明白了语言的很多特性也就掌握了。这是很多优秀程序员可以短时间内掌握一门新语言的秘技之一。
接下来,我们就从类型的基本概念开始讲起。
## 什么是类型
编程语言中的变量都是有类型的而且变量的类型不一定一致。例如Go语言中的int和float声明的变量它们的类型就不一致如果你直接对它们执行加操作Go的编译器就会报错很多隐式类型转换带来的问题在编译阶段就可以发现了。比如你可以看下面这个例子
```plain
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的例子
```plain
>>> a = 1
>>> a + "hello"
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unsupported operand type(s) for +: 'int' and 'str'
>>> a = "hello"
>>> a + "world"
'helloworld'
```
从上面的例子中我们可以看到Python的类型检查也是比较严格的在第2行将一个整型值和字符串值相加是会引发Traceback的。但是一个变量却可以先使用整数为它赋值再使用字符串为它赋值第6行然后再进行字符串值的加操作就没有问题了。这就说明了变量的类型是在程序运行的时候才去检查而不是编译期间进行的。我们把这种语言称为**动态类型语言**。
同样的例子如果使用Go来实现编译器就会报错
```plain
var a int = 1
a = "hello"
fmt.Print(a)
```
由此可见Go语言是一种**静态的强类型语言**。
动态类型不仅仅表现在变量的类型可以更改在面向对象的编程语言中动态类型往往还意味着类的定义也可以动态更改。我们仍然以Python为例来观察动态类型的特点
```plain
>>> class A():
...  pass
...
>>> A.a = 1
>>> a = A()
>>> a.a
1
```
从上述例子中我们可以看到类A的类属性是可以在运行时进行添加和修改的。这与静态编译的语言非常不同。
由此,我们可以得出结论,动态类型相比静态类型,它的优点在于:
1. 动态类型有更好的灵活性,在运行时可以修改变量的类型,也可以对类定义进行修改,所以针对动态类型语言的热更新就更容易设计;
2. 动态类型语言写起来很方便,非常适合用来编写小规模的脚本。
同时,动态类型也往往具有一些缺点(通常是这样,但并不绝对)。
首先动态类型语言的代码不容易阅读。据统计程序员的日常工作中90%的时间是在阅读别人写的代码只有不足10%的时间才是在开发新的功能。而动态类型语言,没有类型标注,代码会非常难懂,即使有一些动态类型语言有类型标注的,但因为可以运行时修改类型,往往会出现一个类的属性在不同的地方被修改的情况,这使得代码的阅读和维护变得困难;
第二点是动态类型语言的性能往往会差一些比如Python和JavaScript因为在编译期间缺少类型提示编译器无法为对象安排合理的内存布局你可以参考[内存课导学三](https://time.geekbang.org/column/article/431373)所以它们的对象布局相比Java/C++等静态类型语言会更加复杂,同时这也会带来性能的下降。
**由此可见,我们并不能简单地说,静态类型就比动态类型好,或者强类型就比弱类型好,还是要根据具体的场景来进行取舍。**
比如要求快速开发规模较小的工具人们常常会选择使用Python而多人合作的大型项目人们就会选择使用Java之类的静态强类型语言。
另外类似int、 String这种类型往往是语言的内建类型而语言的内建类型在表达力上经常是不够的这就需要人们通过将简单内建类型组合起来实现相应的功能这就是**复合类型**。
典型的复合类型包括枚举、结构、列表、字典等。这些类型在Go语言中都有相应的定义你可以参考Go语言专栏进行学习。
在讲完了类型的基本概念以后,我们再讲解一个类型系统中非常常见,同时也是比较困难的一个话题,那就是泛型。
## 为什么要使用泛型
我们使用一个实际的例子来讲一下为什么要使用泛型。比如这里有一个栈的C++实现,栈里可以存放的变量是整型的,它的代码如下所示:
```plain
#include <iostream>
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++为例,泛型的栈可以这么实现:
```plain
#include <iostream>
using namespace std;
template <typename T>
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<int> stack(3);
   stack.push(1);
   stack.push(2);
   cout << stack.pop() + stack.pop() << endl;
   Stack<string> 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系统上我们可以使用以下命令对这个文件进行编译然后查看它的编译结果
```plain
$ g++ -o stack -g stack.cpp
$ objdump -d stack
```
从这个结果中可以看到C++编译器生成了两个push方法其参数类型分别是整型和字符串类型。也就是说在C++中,泛型类型在被翻译成机器码的时候,是真的创建了两种不同的类型。
泛型使用最广泛的场景就是容量类例如vector、list、map等等。C++ STL中定义的容器类都是以模板的形式提供的。
我们可以再使用一种新的视角来理解泛型,那就是可以将泛型声明看作是类型之间的转换关系,或者换种说法,就是我们可以使用一种类型(甚至是值)得到另外一种新的类型。
## 泛型:使用类型得到新的类型
现在我们就用这个新视角来理解泛型,**把泛型声明看成是一个输入参数是类型,返回值也为类型的函数**。我使用一个vector的例子来说明这一点
```plain
int main() {
vector<int> vi;
vector<double> vd;
vector* p = &v1;
return 0;
}
```
这里程序的第4行会报错报错的信息显示“vector”并不是一个有效的类型。而“vector<int> ”和“vector<double>”则是有效的类型。从这个例子中我们观察到vector类型必须指定一个类型参数才能变成一个有效的类型。所以我们可以把
```plain
template <typename T> class vector;
```
看成是一个函数它接受一个类型int或者double得到一种新的类型vector<int>或者vector<double>
在C++中,更神奇的是,泛型的类型参数不仅仅可以是一种类型,还可以是一个具体的值,例如:
```plain
template <int n> class A;
int main() {
A<0> a;
A<1> b;
return 0;
}
```
在上述代码中A<0>和A<1>分别是两个不同的类型。使用这种办法,我们可以在编译期间通过模板让编译器帮我们做一些计算,例如:
```plain
#include <iostream>
using namespace std;
template <int n>
struct fib {
   static const int v = fib<n-2>::v + fib<n-1>::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的泛型设计是使用了一种叫做“泛型擦除”的办法来实现的。
我举一个例子来说明“泛型擦除”是怎么一回事。请看下面的代码:
```plain
import java.util.ArrayList;
class Playground {
    public static void main(String[ ] args) {
        ArrayList<Integer> int_list = new ArrayList<Integer>();
        ArrayList<String> str_list = new ArrayList<String>();
        System.out.println(int_list.getClass() == str_list.getClass());
    }
}
```
这段代码的输出是true。
如果按照上一小节中关于C++泛型的实现ArrayList<Integer>和ArrayList<String>应该是不同的两种类型。但这里的结果却是true这是因为Java会把这两种ArrayList的泛型都擦除掉从而导致整个程序中只有一种类型。
我们这里再举一个例子帮你理解一下Java的泛型
```plain
import java.util.ArrayList;
class Playground {
    public static void main(String[ ] args) {
        System.out.println("Hello World");
    }
    public static void sayHello(ArrayList<String> list) {
    }
    public static void sayHello(ArrayList<Integer> list) {
    }
}
```
这里第8行定义的sayHello方法和第12行定义的sayHello方法是方法重载。我们知道方法的重载的基本条件是两个同名方法的参数列表并不相同。
从字面上看第一个sayHello方法的参数类型是ArrayList<String>第二个方法的参数类型是ArrayList<Integer>,所以可以实现方法的重载。但是当我们尝试编译上述程序的时候,却会得到这样的错误提示:
```plain
Playground.java:12: error: name clash: sayHello(ArrayList<Integer>) and sayHello(ArrayList<String>) have the same erasure
public static void sayHello(ArrayList<Integer> list) {
^
1 error
```
这是因为当对泛型进行擦除以后两个sayHello方法的参数类型都变成了ArrayList从而变成了同名方法所以就会出现命名冲突报错。
通过上面两个例子我们就能感觉到C++泛型和Java泛型的不同之处了。它们之间最核心的区别是C++不同的泛型参数会得到一种新的类型而Java则不会它会进行类型擦除从而导致表面上不同的类型参数实际上指代的是同一种类型。
## 总结
在这节课里我们先了解到什么是类型系统并介绍了什么是强类型和弱类型什么是静态类型和动态类型。然后我们通过举例来说明PythonJavaScriptGo和C++各自的类型系统的特点。
从这些例子中,我们看到静态强类型语言更容易阅读和维护,但灵活性不如动态弱类型语言。所以动态弱类型语言往往都是脚本语言,不太适合构建大型程序。
接下来我们简单介绍了泛型的概念。我们使用了一个栈的例子来说明了使用泛型可以提高编程效率节省代码量。Go语言从1.18开始也支持泛型编程。
然后我们又提供了一个新的视角来理解泛型,这种新的视角是把泛型类看成是一种函数,它的输入参数可以是类型,也可以是值,它的返回值是一种新的类型。
最后我们介绍了C++的泛型实现和Java的泛型实现。C++不同的泛型参数会得到一种新的类型这个过程我们也会称它为泛型的实例化。而Java则会进行类型擦除从而导致表面上不同的类型参数实际上指代的是同一种类型。