基础类型

PHP中主要的基础类型可以在Hack中进行显式类型标注。包含:

  • bool
  • int
  • float
  • string
  • array
  • resource
<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\Primitive;

class A {
protected float $x;
public string $y; public function __construct() {
$this->x = 4.0;
$this->y = "Day";
}
public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
}
} function bar(): string {
// local variables are inferred, not explicitly typed
$a = new A();
if ($a->foo(true) > 8.0) {
return "Good " . $a->y;
}
return "Bad " . $a->y;
} var_dump(bar());

Output

string(8) "Good Day"

别名类型

Hack不支持别名基础类型,所以以下用于类型标准的写法是无效的:

  • boolean
  • integer
  • real
  • double
  • void

void 是一种特殊的基础类型,代表函数或方法没有可监听的返回值。在定义了void的函数体中,可以使用return;。

NOTE: void 只能在方法和函数的返回值处定义,不能用在属性或参数上。

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\Void;

class A {
protected float $x;
public string $y; public function __construct() {
$this->x = 4.0;
$this->y = "Day";
}
public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
}
} // void can only be used as a return types
function bar(): void {
// local variables are inferred, not explicitly typed
$a = new A();
if ($a->foo(true) > 8.0) {
echo "Good " . $a->y;
} else {
echo "Bad " . $a->y;
}
} bar();

Output

Good Day

In Async

It is relatively common for async functions to return Awaitable. This means that while the function itself is returning an awaitable, the result of that awaitable will have no value. This, in essence, means the async function did some operation asynchronously that did not require a return value to the caller.

noreturn

noreturn 是一种特殊的基础类型,代表函数或静态方法永不返回值。与void有点相似,但如果定义了noreturn,在函数体中连return;语句都不能出现。

noreturn 用于标识指定函数或静态方法总是抛出异常或以某种方式终止程序执行。

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\NoReturn;

class A {
protected float $x;
public string $y; public function __construct() {
$this->x = 4.0;
$this->y = "Day";
}
public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
} // no return cannot be on an instance method
// only functions and static class methods
public static function baz(bool $b): noreturn {
if ($b) {
throw new \Exception("No Return");
} else {
exit(1);
}
return; // Even this will cause type-errors
}
} // void can only be used as a return types
function bar(): void {
// local variables are inferred, not explicitly typed
$a = new A();
if ($a->foo(true) > 8.0) {
echo "Good " . $a->y;
} else {
echo "Bad " . $a->y;
}
A::baz(false);
} bar();

Output

Good Day

注意:

仅用于静态方法与函数

noreturn can only be used in function or static method returns.

Instance methods cannot be noreturn. This is due to the order in which the typechecker's analysis phases happen. The return type of an instance method call cannot be determined during control flow analysis because it needs to know the type on the left-hand side of the ->, and the results of type inference aren't available yet. This isn't an issue for calls to static methods, since those can be resolved before types have been inferred.

noreturn is not applicable to properties or parameters.

对象

You can use the name of any built-in or custom class or interface.

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\Obj;

class Z {
public function create_A(): A {
return new A();
}
} class A {
protected float $x;
public string $y; public function __construct() {
$this->x = 4.0;
$this->y = "Day";
}
public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
}
} // We are taking a Z and returning an object of type A
function baz(Z $z): A {
return $z->create_A();
} function bar(): string {
// local variables are inferred, not explicitly typed
$z = new Z();
$a = baz($z);
if ($a->foo(true) > 8.0) {
return "Good " . $a->y;
}
return "Bad " . $a->y;
} var_dump(bar());
Output
string(8) "Good Day"
mixed

mixed is essentially a catch-all type that represents any possible Hack value (including null and void).

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\Mixed;

class A {
public float $x;
protected string $y; public function __construct() {
$this->x = 4.0;
$this->y = "Day";
}
// mixed is the most lax type. Use it only when necessary
public function foo(bool $b): mixed {
return $b ? 2.3 * $this->x : $this->y;
}
} function bar(): string {
// local variables are inferred, not explicitly typed
$a = new A();
$v = $a->foo(false);
// Since A::foo() returns a mixed, we need to do various checks to make sure
// that we let the typechecker know understand what is coming back.
if (is_float($v)) {
return "No String";
}
invariant(is_string($v), "Something went wrong if this isn't true");
return "Good " . $v;
} var_dump(bar());
Output
string(8) "Good Day"

Use sparsely

There are valid uses for mixed, but generally you want to be as specific as possible with your typing since the typechecker can only do so much with mixed given its constraints are so lax.

this

this can only be used as a return type annotation on a method of a class. this signifies that the method returns an object of the same class on which the method is defined.

The primary purpose of return this is to allow chaining of method calls on the instance of the class itself or its subclasses.

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining;

class Vehicle {
private ?int $numWheels;
private ?string $make; public function setNumWheels(int $num): this {
$this->numWheels = $num;
return $this;
} public function setMake(string $make): this {
$this->make = $make;
return $this;
}
} class Car extends Vehicle {
private ?bool $autoTransmission; public function setAutomaticTransmission(bool $automatic): this {
$this->autoTransmission = $automatic;
return $this;
}
} class Hybrid extends Car {
private ?bool $pluggable; public function setPluggable(bool $pluggable): this {
$this->pluggable = $pluggable;
return $this;
} public function drive(): void {}
} function run(): void {
$h = new Hybrid();
// $h->NumWheels(4) returns the instance so you can immediately call
// setMake('Tesla') in a chain format, and so on. Finally culminating in an
// actionable method call, drive().
$h->setNumWheels(4)
->setMake('Tesla')
->setAutomaticTransmission(true)
->setPluggable(true)
->drive();
var_dump($h);
} run();
Output
object(Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining\Hybrid)#1 (4) {
["pluggable":"Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining\Hybrid":private]=>
bool(true)
["autoTransmission":"Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining\Car":private]=>
bool(true)
["numWheels":"Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining\Vehicle":private]=>
int(4)
["make":"Hack\UserDocumentation\Types\TypeSystem\Examples\ThisChaining\Vehicle":private]=>
string(5) "Tesla"
}

this on a static method means that a class method returns an object of the same class as the calling method. You can use it to return an instance of an object from a static class method where you are returning something like new static().

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\ThisStatic;

class A {
protected float $x;
public string $y; // typechecker error if constructor isn't final because new static() cannot
// be called to return an instance of a subclass
final protected function __construct() {
$this->x = 4.0;
$this->y = "Day";
} public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
} // The this type annotation allows you to return an instance of a type
public static function create(int $x): this {
$instance = new static();
if ($x < 4) {
$instance->x = floatval($x);
}
return $instance;
}
} function bar(): string {
// local variables are inferred, not explicitly typed
// There is no public constructor, so call A's create() method
$a = A::create(2);
if ($a->foo(true) > 8.0) {
return "Good " . $a->y;
}
return "Bad " . $a->y;
} var_dump(bar());
Output
string(7) "Bad Day"

num

num 是int 与 float的一种特殊联合类型。正常情况下,Hack里整型和浮点型是不兼容的类型。但对于很多数值操作的函数来说,传整型或浮点型差不多的情况下,就可以用num了。

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\Num;

class A {
protected num $x;
public string $y; public function __construct(num $x) {
$this->x = $x;
$this->y = "Day";
}
public function foo(bool $b): num {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
}
// The $x property can be either a float or int
public function setNum(num $x): void {
$this->x = $x;
}
} function check(A $a): string {
if ($a->foo(true) > 8.0) {
return "Good " . $a->y;
}
return "Bad " . $a->y;
} function bar(): string {
// local variables are inferred, not explicitly typed
// Setting the $x property in A to an int
$a = new A(4);
$ret = check($a);
// Now setting to a float
$a->setNum(0.4);
$ret .= "##" . check($a);
return $ret;
} var_dump(bar());
Output
string(17) "Good Day##Bad Day"

arraykey

arraykey is special union type of int and string. Arrays and collection types can be keyed by int or string. Suppose, for example, an operation was performed on an array to extract the keys, but you didn't know the type of the key. You were left with using mixed or doing some sort of duplicative code. arraykey resolves that issue.

<?hh

namespace Hack\UserDocumentation\Types\TypeSystem\Examples\ArrayKey;

class A {
protected float $x;
public string $y; public function __construct(float $x) {
$this->x = $x;
$this->y = "Day";
}
public function foo(bool $b): float {
return $b ? 2.3 * $this->x : 1.1 * $this->x;
}
} // This function can return either a string or an int since it is typed to
// return an arraykey
function bar(): arraykey {
// local variables are inferred, not explicitly typed
$a = new A(0.9);
if ($a->foo(true) > 8.0) {
return "Good " . $a->y;
}
return 5;
} var_dump(bar());
Output
int(5)

XHP

There are two XHP interfaces that are used when typing XHP objects: XHPChild and XHPRoot.

XHPRoot is any object that is an instance of an XHP class.

XHPChild is the set of valid types for echoing within an XHP context (e.g., echo

{$xhpobj}

Hack语言的类型系统的更多相关文章

  1. Hack语言类型化简介

    在typechecker的配合下,Hack语言的类型化能力是Hack其他功能特性的基石.开发Hack语言的主要动机也正是为代码提供显式类型标注以便对代码进行类型一致性和潜在错误分析. 这是用于对比Ha ...

  2. Facebook的Hack语言三大看点

    Hack语言主要有三大看点:类型化.异步.集合. Hack最基础的特性就是类型标注.PHP5已经开始支持对象的类型化,PHP7也提供了标量类型化声明.Hack提供了全面的类型标注支持,与其typech ...

  3. Facebook Hack 语言 简介

    1. Hack 是什么? Hack 是一种基于HHVM(HipHop VM 是Facebook推出的用来执行PHP代码的虚拟机,它是一个PHP的JIT编译器,同时具有产生快速代码和即时编译的优点.)的 ...

  4. Hack语言特性之类型化

    Hack最基础的特性就是类型标注.PHP5已经开始支持对象的类型化,PHP7也提供了标量类型化声明.Hack提供了全面的类型标注支持,与其typecher配合使用,还可以实现快速.前置静态类型验证. ...

  5. 【编程笔记】Unity3D语言的类型系统--C#的类型系统

    几乎所有的编程语言都有自己的类型系统. 而编程语言更是常常按照其类型系统而被分为强类型语言/弱类型语言.安全类型语言/不安全类型语言.静态类型语言/动态类型语言等. 而C#的类型系统是静态.安全,并且 ...

  6. Hack 语言学习/参考---1.3 Summary

    Summary Hack provides the following, non-exhaustive list of features: Ability to annotate function a ...

  7. Hack 语言学习/参考---1.2 Hack Background

    Hack Background Facebook was initially built with PHP. Parameters and return types were specified in ...

  8. Hack 语言学习/参考---1.1 What is Hack?

    What is Hack?¶ Hack is a language for HHVM that interopates seamlessly with PHP. The barrier to entr ...

  9. Hack 语言学习/参考---1.Hack 语言

    Table of Contents What is Hack? Hack Background Summary Hack is a language for HHVM that interopates ...

随机推荐

  1. swift 创建单例模式

    一.意图 保证一个类公有一个实例,并提供一个访问它的全局访问点. 二.使用场景 1.使用场景 当类只能有一个实例而且客户可以从一个众所周知的访问点访问它时 当这个唯一实例应该是通过子类化可扩展的,并且 ...

  2. _学生选课数据库SQL语句练习题

    1. 查询Student表中的所有记录的Sname.Ssex和Class列. select Sname,Ssex,t.sclass from STUDENT t 2. 查询教师所有的单位即不重复的De ...

  3. Atitit 图片 验证码生成attilax总结

    Atitit 图片 验证码生成attilax总结 1.1. 图片验证码总结1 1.2. 镂空文字  打散 干扰线 文字扭曲 粘连2 1.1. 图片验证码总结 因此,CAPTCHA在图片验证码这一应用点 ...

  4. Java项目——模拟电话薄联系人增删改查

    该项目模拟了电话本记录联系人的业务功能,用来练习对数据库的增删改查等操作. 菜单类:Menu -- 用来封装主菜单和个选项的子菜单 Person类: Person--联系人的实体类 TelNoteRe ...

  5. javascript_core_10之继承与数组API

    1.现有两对象间的继承:Object.setPrototypeOf(child,father): 2.基于现有父对象创建子对象:var child=Object.create(father,{新属性} ...

  6. Liferay7 BPM门户开发之44: 集成Activiti展示流程列表

    处理依赖关系 集成Activiti之前,必须搞清楚其中的依赖关系,才能在Gradle里进行配置. 依赖关系: 例如,其中activiti-engine依赖于activiti-bpmn-converte ...

  7. js设置自动刷新

    如何实现刷新当前页面呢?借助js你将无所不能. 1,reload 方法,该方法强迫浏览器刷新当前页面.语法:location.reload([bForceGet])   参数: bForceGet, ...

  8. KnockoutJS 3.X API 第三章 计算监控属性(5) 参考手册

    计算监控属性构造参考 计算监控属性可使用以下形式进行构造: ko.computed( evaluator [, targetObject, options] ) - 这种形式是创建一个计算监控属性最常 ...

  9. Java-集合-第三题 有如下Student 对象, private String name; private int age; private int score; private String classNum; 其中,classNum 表示学生的班号,例如“class05”。 有如下List List list = new ArrayList(); l

    第三题 有如下Student 对象, private String name; private int age; private int score; private String classNum; ...

  10. 关于json序列化和反序列的问题,没事写个案例,希望能帮到那些需要帮忙的朋友!

    现在关于json的读写问题,有许许多多的解决方法,因人而异,根据实际问题去选择自己想要的最容易方法.我觉得自带的Newtonsoft.Json是个不错的选择,随便写两个例子吧! 一:关于简单的json ...