类型系统是在计算机科学中,类型系统用于定义如何将编程语言中的
数值和
表达式归类为许多不同的
类型,如何操作这些类型,这些类型如何互相作用。类型可以
确认一个值或者一组值具有特定的意义和目的(虽然某些类型,如抽象类型和函数类型,在程序运行中,可能不表示为值)。类型系统在各种语言之间有非常大的不同,也许,最主要的差异存在于编译时期的语法,以及运行时期的操作实现方式。
解析
编译器可能使用值的静态类型以优化所需的存储区,并选取对值运算时的较佳算法。例如,在许多
C编译器中,“
浮点数”
数据类型是以 32
比特表示,与IEEE 754 规格一致的
单精度浮点数。因此,在数值运算上,C 应用了浮点数规范(浮点数加法、乘法等等)。
类型的约束程度以及评估方法,影响了语言的类型。更进一步,编程语言可能就类型多态性部分,对每一个类型都对应了一个极度个别的算法的运算。
类型理论研究类型系统,尽管实际的编程语言类型系统,起源于电脑架构的实际问题、编译器实现,以及语言设计。
基础
定型(typing,又称类型指派)赋予一组
比特某个意义。类型通常和
存储器中的数值或
对象(如
变量)相联系。因为在电脑中,任何数值都是以一组比特简单组成的,硬件无法区分
存储器地址、
脚本、
字符、
整数、以及
浮点数。类型可以告知程序和程序设计者,应该怎么对待那些比特。
类型系统提供的主要功能有:
使用类型可允许
编译器整数。
强类型提供更多的安全性,但它并不能保证绝对安全(详情请见
类型安全)
静态类型检查可提供有用的信息给编译器。例如,如果一个类型指明某个值必须以 4 的倍数对齐,编译器就有可以使用更有效率的
机器指令。
在更具表现力的类型系统中,若其可以阐明程序设计者的意图的话,类型就可以充当为一种文件形式。例如,时间戳记可以是整数的子类型;但如果程序设计者声明一个函数为返回一个时间戳记,而不只是一个整数,这个函数就能表现出一部分文件的阐释性。
类型允许程序设计者对程序以较高层次的方式思考,而不是烦人的低层次实现。例如,程序设计者可以将字符串想成一个值,以此取代仅仅是字节的数组。或者类型允许程序设计者表达两个子系统之间的
接口。将子系统间交互时的必要定义加以定位,防止子系统间的通信发生冲突。
程序通常对每一个值关系一个特定的类型(尽管一个类型可以有一个以上的
子类型)。其它的实体,如
对象、
模块、通信频道、依赖关系,或者纯粹的类型自己,可以和一个类型关系。例如:
一个数值的类型
一个对象的类型
一个类型的类型
在每一个编程语言中,都有一个特定的类型系统,保证程序的表现良好,并且排除违规的行为。作用系统对类型系统提供更多细微的控制。
类型检查
类型检查所进行的检验处理以及实行类型的约束,可发生在编译时期(
静态检查)或运行时期(动态检查)。静态类型检查是在
编译器所进行
语义分析中进行的。如果一个语言强制实行类型规则(即通常只允许以不丢失信息为前提的
自动类型转换)就称此处理为
强类型,反之称为弱类型。
动态检查
如果一个编程语言的类型检查,可在不测试运行时期表达式的等价性的情况下进行,该语言即为静态类型的。一个静态类型的编程语言,是在运行时期和编译时期之间的处理阶段下重视这些区别的。如果程序的独立模块,可进行各自的类型检查(独立编译),而无须所有会在运行时出现的模块的那些信息,该语言即具有一个编译时期阶段。如果一个编程语言支持运行时期(动态)调度已标记的数据,该语言即为动态类型的。如果一个编程语言破坏了阶段的区别,因而类型检查需要测试运行时期的表达式的等价性,该语言即为依存类型的。
在动态类型中,经常在运行时期进行类型标记的检查,因为变量所约束的值,可经由
运行路径获得不同的标记。在静态类型编程语言中,类型标记使用辨识联合类型表示。
动态类型经常出现于
脚本语言和RAD语言中。动态类型在解译语言中极为普遍,
编译语言则偏好无须运行时期标记的静态类型。对于类型和隐式类型语言较完整的列表参见类型和隐式类型语言。
术语推断类型(
鸭子类型,duck typing)指的是动态类型在语言中的应用方式,它会“推断”一个数值的类型。
某些静态语言有一个“后门”,在这些编程语言中,能够编写一些不被静态类型所检查的代码。例如,Java 和 C-风格的语言有“
转型”可用。在静态类型的编程语言中,不必然意味着缺乏动态类型机制。例如 Java 使用静态类型,但某些运算需要支持运行时期的类型测试,这就是动态类型的一种形式。更多静态和动态类型的讨论,请参阅编程语言。
检查
静态类型在编译时期时,就能可靠地发现类型错误。因此通常能增进最终程序的可靠性。然而,有多少的类型错误发生,以及有多少比例的错误能被静态类型所捕捉,仍有争论。静态类型的拥护者认为,当程序通过类型检查时,它才有更高的可靠性。虽然动态类型的拥护者指出,实际流通的软件证明,两者在可靠性上并没有多大差别。可以认为静态类型的价值,在于增进类型系统的强化。
强类型语言(如 ML 和
Haskell)的拥护者提出,几乎所有的臭虫都可以看作是类型错误,如果编写者以足够恰当的方式,或者由编译器推断来声明一个类型。
静态类型通常可以编译出速度较快的代码。当编译器清楚知道所要使用的数据类型,就可以产生优化过后的
机器码。更进一步,静态类型语言中的编译器,可以更轻易地发现较佳捷径。某些
动态语言(如
Common Lisp)允许任意类型的声明,以便于优化。以上理由使静态类型更为普及。参阅
优化。
相较之下,动态类型允许编译器和解译器更快速的运作。因为源代码在动态类型语言中,变更为减少进行检查,并减少解析代码。这也可减少编辑-编译-测试-除错的周期。
静态类型语言缺少类型推断(如 Java),而需要编写者声明所要使用的方法或函数的类型。编译器将不允许编写者忽略,这可为程序起附加性说明文件的作用。但静态类型语言也可以无须类型声明,所以与其说是静态类型的代价,倒不如说是类型声明的报酬。
静态类型允许构造函数库,它们的用户不太可能意外的误用。这可作为传达库开发者意图的额外机制。
动态类型允许建构一些静态类型系统所做不出来的东西。例如,eval 函数,它使得运行任意数据作为代码成为可能(不过其代码的类型仍是静态的)。此外,动态类型容纳过渡代码和原型设计,如允许使用字符串代替数据结构。静态类型语言最近的增强(如 Haskell 一般化代数数据类型)允许 eval 函数以类型安全的方式撰写。
动态类型使元程序设计更为强大,且更易于使用。例如
C++ 模板的写法,比起等价的
Ruby 或
Python 写法要来的麻烦。更高度的运行时期构成物,如元类型(metaclass)和
内观(Introspection),对静态类型语言而言通常更为困难。
强类型弱类型
强类型的基本定义即为,禁止错误类型的参数继续运算。C语言的
类型转换即为缺乏强类型的证例;如果编写者用 C 语言对一个值转换类型,不仅令编译器允许这个代码,而且在运行时期中也同样允许。这使得 C 代码可更为紧密和快速,不过也使除错变的更为困难。
部分学者使用术语
存储器安全语言(或简称为安全语言)形容禁止未定义运算发生的语言。例如,某个存储器安全语言将会检查数组边界。
设计精巧的语言也允许语言显现出弱类型(借由类型推断之类的技术)的特性以方便使用,并且保留了
强类型语言所提供的类型检查和保护。例子包括 VBNet、
C# 以及
Java。
运算符重载所带来的简化,像是不以算术运算中的加法来使用“+”,可以减少一些由动态类型所造成的混乱。例如,部分语言使用“.”或“&”来串连字符串。
安全性
编程语言的类型系统的第三种分类方法,就是类型运算和转换的安全性。如果它不允许导致不正确的情况的运算或转换,计算机科学就认为该语言是“
类型安全”的。
多态性和类型
术语“
多态性”指的是:代码(尤其是函数和类型)对各种类型的值能够动作,或是相同数据结构的不同实体能够控制不同类型的元素。为了提升复用代码的
潜在价值,类型系统逐渐允许多态性:在具有多态性的语言中,程序设计者只需要实现如列表或词典的数据结构一次,而不是对使用到它的元素的每一个类型都规划一次。基于这个原因,电脑学家也称使用了一定的多态性的方法为泛型程序设计。类型理论的多态性基础与
抽象化、
模块化和(偶尔)
子类型有相当密切的联系关系。
推断类型
推断类型(
鸭子类型,Duck typing)最初是由
Dave Thomas 在
Ruby 社区中提出的,推断类型用了这个论证法“如果它像什么,而且其它地方也像什么,那么它就是什么。”
在某些程序设计环境中,两个对象可以有相同的类型,即使它们没有什么交集。一个例子是
C++ 在
迭代器和指针之间的双重性。两者皆以不甚相同的机制实现并提供一个 * 运算。
这个技术之所以常被称作“
鸭子类型”,是基于这句格言:“如果它摇摇摆摆的走法很像鸭子,而且它的嘎嘎叫声也像鸭子,那它就是一只鸭子!”
显示声明
许多静态类型系统,如 C 和 Java,要求要声明类型:编写者必须以指定类型明确地关系到每一个变量上。其它的,如 Haskell,则进行类型推断:编译器根据编写者如何运用这些变量,以草拟出关于这个变量的类型的结论。例如,给定一个函数 f(x,y),它将 x 和 y 加起来,编译器可以推断出 x 和 y 必须是数字——因为加法仅定义给数字。因此,任何在其它地方以非数值类型(如字符串或
链表)作为参数来调用 f 的话,将会发出一个错误。
隐式暗示
在代码中数值、字符串常数以及表达式,经常可以在详细的前后文中暗示类型。例如,一个表达式 3.14 可暗示
浮点数类型;而 [1, 2, 3] 则可暗示一个整数的
链表;通常是一个
数组。
类型的类型
类型的类型是一种种类。在类型程序设计中有明确的种类,如 Haskell 编程语言的类型
构造函数,在申请比较简单的类型之后,其返回一个简单的类型。例如,类型构造函数 二选一 有这些种类 * -> * -> *(* 代表种类),而且它的申请 二选一 字符串 整数 是一个简单的类型。然而,大多数编程语言的类型,是由编写者来暗示或
硬编码,这就并未将种类的概念用作为首选层。
类型可分为几个大类:
全部是数字的类型,例如:整数和自然数
由基本类型组合成的类型,例如:
数组或记录单元。
抽象数据类型具有复合类型和界面两种属性,这取决于你提及哪一个。
例如:变量类型
例如:双参函数
如参数化类型、类型变量
识别其它类型的子集的类型
取决于运行时期的数值的类型
描述或约束面向对象系统结构的类型
兼容性
对于静态类型语言的类型检查器,必须检验所有
表达式的类型,是否与前后文所期望的类型一致。例如
指派语句x := e,推断表达式 e 的类型,必定与声明或推断的变量类型 x 一致。这个一致性的概念,就称为兼容性,是每一个编程语言所特有的。
很明显,如果 e 和 x 的类型相同,就允许指派,然后这是一个有效的表达式。因此在最简单的类型系统中,问题从两个类型是否兼容,简化为两个类型是否相等(或等价)。然而不同的语言对于两个类型表达式是否理解为表示了相同类型,有着不同的标准。类型的相等理论的差异相当巨大,两个极端的例子是结构类型系统(Structural type system),任两个以相同结构所描述的值的类型都是等价的,且在标明类型系统(Nominative type system)上,没有两个独特的语法构成的类型表达式表示同一类型,(即类型若要相等,就必须具有相同的“名字”)。
在
子类型的语言中,兼容关系更加复杂。特别是如果 A 是 B 的子类型,那么类型 A 的值可用于类型 B 也属意料之中,但反过来就不是这样。如同等价性,对每一个编程语言而言,子类型的关系的定义是不同的,可能存在各种变化。在语言中出现的参数或者特定的
多态性,也可能意味着具有对类型的兼容性。
争议
在
强类型、静态类型语言的支持者,和动态类型、自由形式的支持者之间,经常发生争执。前者主张,在编译的时候就可以较早发现错误,而且还可增进运行时期的性能。后者主张,使用更加动态的类型系统,分析代码更为简单,减少出错机会,才能更加轻松快速的编写程序。与此相关的是,考虑到在类型推断的编程语言中,通常不需要手动声明类型,这部分的额外开销也就自动降低了。