《Hack与HHVM权威指南》——1.6.3 推理局限在函数内

本节书摘来自华章出版社《Hack与HHVM权威指南》一书中的第1章,第1.6.3节,作者 Owen Yamauchi,更多章节内容可以访问“华章计算机”公众号查看。

1.6.3 推理局限在函数内

关于Hack的类型推理的一个最基本限制就是:当分析一个函数的时候,它绝对不会看另外一个函数或方法的代码体。举例来说,假设以下是你的全部代码:

function f($str) {
  return 'Here is a string: ' . $str;
}
function main() {
  echo f('boo!');
}
main();

对于读者来说,如下两个事实是非常清晰的:$str总是一个字符串类型,并且函数 f() 总是返回一个字符串。然而,Hack的类型检查器并不能推断出上述事实。
当对函数f()进行类型推理的时候,它并不会检查函数f()的调用者以找出调用者实际传递的参数值的类型是什么。在函数main()内部进行类型推理时,它也不会查看函数f()的代码体来找到它实际上返回的是什么类型。它只会检查函数f()的签名用于返回类型标注。然后关于函数的返回类型标注,它什么也没有发现,所以它就认为函数f()的返回类型为“any”(具体参见1.4.2节的内容)。
这种限制的存在是基于性能的原因。对一个函数的推理限制在这个函数的内部,是为了对这个函数的分析所需要的计算量设置一个严格上限。 扩展开来的话,就是对整个代码库的所需计算量设置一个上限。在计算复杂度方面,类型推理算法在复杂度上是超线性(superlinear)的。所以给出多次小的输入而不是一个巨大的输入对于保持运行总时间的可管理性是非常重要的。
对于大型的代码库而言(比如Hack语言原生设计的Facebook),这个属性非常重要。当一个函数体内部变化后(但是它的函数签名并没有变化),类型检查服务端只需要重新分析这个函数,然后保持相关的知识点是及时的即可。并且这个重新分析的过程几乎是实时处理的。当一个函数签名发生了变化后,类型检查器的服务端将会重新分析这个函数和它的所有调用者,并不是它们的调用者,这就对需要的工作量提出了一个比较低的上限。
但是对于此限制也有特例:闭包。虽然一个闭包从定义上来说是个独立的函数,但是对一个内部有闭包的函数进行类型推理的时候,允许对闭包内部进行检查。详细考虑一下如下示例:
并推断出返回
var_dump($doubler(10)); // int(20)
var_dump($doubler(3.14)); // float(6.28)
虽然这个闭包没有任何的返回类型标注(这在严格模式下也是合法的),类型检查器依然能够推断出$doubler(10)是个整型。这是因为它分析了闭包的函数体内部的内容,在$x是个整型的假设下,并且在两个整型上应用加号操作符,仍然会得到一个整型结果注5。同理,它可以推断出$doubler(3.14)是浮点数类型。
顺便提一下,正是因为类型推理能够对闭包内部进行推理,所以即使在严格模式下,也允许闭包没有类型标注。

时间: 2024-09-08 04:32:10

《Hack与HHVM权威指南》——1.6.3 推理局限在函数内的相关文章

《Hack与HHVM权威指南》——第1章 类型检查

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 第1章 类型检查 类型检查器是Hack语言的标志特性,它对Hack程序静态地进行分析(不用运行它们),并且能够检查很多种错误.这就能够在程序开发初期尽量避免bug,并且使程序更容易阅读和理解.为了增强类型检查器的能力,Hack语言允许编程人员显式地在程序体中标注某些变量值的类型,比如函数参数.函数返回值和属性值,类型检

《Hack与HHVM权威指南》——1.7.1 提炼nullable类型到non-nullable类型

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.7.1 提炼nullable类型到non-nullable类型 null检查语句在从空值(nullable)的类型到非空值(non-nullable)类型的转变中经常用到.下面是个通过了类型检查器检查的示例. function takes_string(string $str) { // ... } functio

《Hack与HHVM权威指南》——1.5.1 使用超级全局变量

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.5.1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.5.1 使用超级全局变量 超级全局变量是在每个代码范围内都存在的全局变量,不需要使用global语句即可使用.这些在运行环境中特殊存在的变量一共有9个,分别是: $GLOBALS $_SERVER $_GET $_POST $_FILES $_COOKIE $_SESSION $_REQUEST $_ENV

《Hack与HHVM权威指南》——1.5.3 属性值初始化

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.5.3节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.5.3 属性值初始化 为了维护类型安全,类型标注过的属性在初始化时,无论是严格模式还是局部模式,类型检查器会强加一些规则.首要目标就是确保属性值在没有初始化为正确类型的值之前,不能被读取. 对于静态的属性值,规则非常简单:任何不可为空的(non-nullable)的静态属性值都必须有一个初始化值.没有显式初

《Hack与HHVM权威指南》——1.4 Hack的类型系统

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.4节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.4 Hack的类型系统 Hack提供了一系列强有力的方法来描述类型,在PHP最基本的布尔型.整型.字符串型.数组等类型系统的基础上,添加了很多新的方式来结合它们,并且使之更富有表现力. 原始类型 这里有和PHP一样的原始类型:bool.int.float.string.array和resource,这些都是合法

《Hack与HHVM权威指南》——第1章 类型检查1.7 类型提炼

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.7节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.7 类型提炼 假设你有个?string类型的值,而且准备把这个值传递给一个参数类型为string的函数.那么你怎么把一个类型(?string)转化为另外一个类型(string)呢?或者假设你有个object类型的值,它可能实现或没有实现Polarizable接口.同时,如果它实现了这个接口,你还希望调用这个ob

《Hack与HHVM权威指南》——第2章 泛型2.1 入门实例

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第2章,第2.1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 第2章 泛型 泛型在Hack的类型系统里面是个非常强大的特性,泛型可以允许你在不知道流程中传入的具体类型的情况下,写出类型安全的代码.一个类或者函数都可以是泛型的,这意味着它可以让调用者来选择传入的参数类型. 泛型结构体最好的例子就是数组和集合类(关于集合类的更多内容请参见第5章).不具备明确指出数组内容具体类型的

《Hack与HHVM权威指南》——1.2.1 自动加载一切

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.2.1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.2.1 自动加载一切 类型检查器做出的一个关键假设就是,你的项目经过设置后,代码库中的任何类.函数或者常量都能够在你代码库的其他地方使用.不会尝试去分析任何include或者require语句,确保当前文件在使用时已经加载了其他文件.相反,它认为你已经完成了"自动加载"的相关设置.这就简

《Hack与HHVM权威指南》——1.4.1 类型检查器模式

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.4.1节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.4.1 类型检查器模式 Hack的类型检查器有三种模式:严格(strict).局部(partial)和耦合(decl).这些模式都基于一个个单独的文件,不同模式下的单独文件可以无缝地进行对接.每个文件可以单独声明它的类型检查模式,语法就是在文件的第一行使用一个双斜线的注释.如下面的代码所示:<?hh

《Hack与HHVM权威指南》——1.5.6 在switch语句中的越界

本节书摘来自华章出版社<Hack与HHVM权威指南>一书中的第1章,第1.5.6节,作者 Owen Yamauchi,更多章节内容可以访问"华章计算机"公众号查看. 1.5.6 在switch语句中的越界 在switch语句中有从一个分支不小心越界到另外一个分支的常见错误.Hack添加了一条规则来捕获这种错误.下面将展示一个案例,从一个case分支越界到另外一个分支将会引发一个错误,除非第一个分支是空的: switch ($day) { case 'sun': echo '