使用flow捕获js错误提高代码质量
admin
2024-01-29 01:51:53
0

本文将介绍Flow及其主特性。下面分别从如何安装设置,如何添加类型注释,如何在运行时自动去掉注释等方面来介绍。

安装

目前Flow兼容的操作系统有Mac OS X,Linux(64位),Windows(64位)。最简单的安装方式是通过 npm:

npm install --save-dev flow-bin

然后在package.json文件中的scripts 项添加:

"scripts": {
 "flow": "flow"
}

完成之后,就可以开始探索它的特性了。

启动

首先需要在项目的根目录下创建一个.flowconfig配置文件。我们可以通过以下命令创建一个空配置文件:

npm run flow init

完成设置之后,在终端输入以下命令可以在你的项目根目录以及任何子目录文件夹下进行专门的类型检查:

npm run flow check

但是,这并不是最高效的使用方式,因为每次Flow都会重新检查整个项目的所有文件。开发过程中,推荐启动Flow服务。

Flow服务的工作方式是增量检查,也就是说它只检查变化的部分。在终端输入以下命令来启动Flow 服务:

npm run flow

首次运行该命令时,服务启动并且显示最初类型检查结果。这保证了Flow更高效的增量式工作流。然后接下来每次想要知道检测结果,只要输入flow命令即可。开发结束之后,输入npm run flow stop停止服务。

Flow的类型检查是可选的,并不需要一次性检查所有代码。你可以选择你想要检查的文件,只要在对应的JavaScript文件最前面加上带有@flow标识的注释即可:

/*@flow*/

当你想在已有项目中加入Flow的时候,该特性特别有帮助。因为你可以一一选择并检测你要的文件,然后修正错误。

类型推断

通常,类型检查分为以下两种方式:

  • 通过注释:事先注释好我们期待的类型,Flow就会基于这些注释来评估

  • 通过代码推断:通过变量的使用上下文来推断出变量类型,然后根据这些推断来检查类型

第一种方式,我们需要额外编写只在开发阶段起作用的代码,最后在代码编译打包的阶段被剔除。显然,这种额外添加类型注释的方式增加了工作量。

第二种方式,不需要任何代码修改即可进行类型检查,最小化开发者的工作量。它不会强制你改变开发习惯,因为它会自动推断出变量的类型。这就是所谓的类型推断,Flow最重要的特性之一。

我们来通过一个例子来说明这个特性:

/*@flow*/

function foo(x) {
 return x.split(' ');
}

foo(34);

当你在终端运行npm run flow命令的时候,上述代码会报错,因为函数foo()的期待参数是字符串,而我们输入了数字。

错误信息类似如下:

index.js:4
 4:   return x.split(' ');
               ^^^^^ property `split`. Property not found in
 4:   return x.split(' ');
             ^ Number

上述信息清楚地指出了出错位置和错误原因。我们只要将参数变成字符串,即可修正错误,如下所示:

/*@flow*/

function foo(x) {
 return x.split(' ');
};

foo('Hello World!');

如上所述,以上代码不会报任何错。该例想说明的是,因为split()方法只适用于string类型的变量,所以x应该是string,这就是类型推断。

空类型

Flow处理null的方式与其他类型库不同。它不会忽略null,这样可以防止了因给变量传了null而导致程序崩溃的错误。

思考以下代码:

/*@flow*/

function stringLength (str) {
 return str.length;
}

var length = stringLength(null);

Flow会报错。为了防止出错,我们需要单独处理null。

/*@flow*/

function stringLength (str) {
 if (str !== null) {
   return str.length;
 }

 return 0;
}

var length = stringLength(null);

代码中我们引入对null的检查,确保代码能在任何情况下都正常且正确运行。上述代码可以通过Flow的类型检查。

类型注释

如上所述,类型推断是Flow最有用的特性之一,不需要编写类型注释就能获取有用的反馈。但在某些特定的场景下,添加类型注释可以提供更好更明确的检查依据。

考虑以下代码:

/*@flow*/

function foo(x, y){
 return x + y;
}

foo('Hello', 42);

Flow检查上述代码时检查不出任何错误,因为+即可以用在字符串上,也可以用在数字上,我们并没有明确指出add()的参数必须为数字。

在这种情况下,我们可以借助类型注释来指明期望的类型。类型注释是以冒号:开头,可以在函数参数,返回值,变量声明中使用。

如果我们在上段代码中添加类型注释,就会变成如下:

/*@flow*/

function foo(x : number, y : number) : number {
 return x + y;
}

foo('Hello', 42);

现在Flow就能检查出错误,因为函数参数的期待类型为数字,而我们提供了字符串。

Flow报错信息类似如下:

index.js:7
 7: foo('Hello', 42);
        ^^^^^^^ string. This type is incompatible with the expected param type of
 3: function foo(x : number, y : number) : number{
                     ^^^^^^ number

如果传入的参数是数字,就不会有错误。类型注释在大型复杂的JavaScript文件中也很有用,它能保证代码按照预期进行。

记住上一个例子,我们来看看Flow能支持的其他更多类型注释。

函数

/*@flow*/

/*--------- Type annotating a function --------*/
function add(x : number, y : number) : number {
 return x + y;
}

add(3, 4);

上述代码展示了变量类型注释以及函数类型注释。函数add()的参数,以及函数的返回值,期待类型为数字。如果传入其他类型参数,Flow就会检测到错误。

数组

/*-------- Type annotating an array ----------*/
var foo : Array = [1,2,3];

数组类型注释的格式是Array,T表示数组中每项的数据类型。在上述代码中,foo是每项均为数字的数组。

下面展示了类和对象的类型注释模型。唯一需要注意的是,可以在两个类型之间使用或逻辑,用|来间隔。变量bar1添加了必须为Bar类的类型注释。

/*-------- Type annotating a Class ---------*/
class Bar{
 x:string;           // x should be string      
 y:string | number;  // y can be either a string or a number
 constructor(x,y){
   this.x=x;
   this.y=y;
 }
}

var bar1 : Bar = new Bar("hello",4);

对象字面量

对象的类型注释类似于类,指定对象属性的类型。

/*--------- Type annonating an object ---------*/

var obj : {a : string, b : number, c: Array, d : Bar} = {
 a : "hello",
 b : 42,
 c : ["hello", "world"],
 d : new Bar("hello",3)
}

Null

若想任意类型T可以为null或者undefined,只需类似如下写成?T的格式即可。

/*@flow*/

var foo : ?string = null;

此时,foo可以为字符串,也可以为null。

目前我们只对Flow的类型注释做了很浅的探索。一旦你习惯了使用这些基本类型,建议在Flow官网上的类型文档深入了解所有的类型。

库定义

我们经常需要引入第三方库,Flow检查时就会抛出错误。但这并不是我们期待的错误。

庆幸的是,我们不需要修改库源码去防止这些报错。我们只需创建一个库定义(libdef)。libdef是包含第三方库声明的JS文件简称。

观察下面的例子:

/* @flow */

var users = [
 { name: 'John', designation: 'developer' },
 { name: 'Doe', designation: 'designer' }
];

function getDeveloper() {
 return _.findWhere(users, {designation: 'developer'});
}

Flow会检查出以下错误:

interfaces/app.js:9
 9:   return _.findWhere(users, {designation: 'developer'});
             ^ identifier `_`. Could not resolve name

由于Flow并不认识_,所以会报错。要解决这个问题,我们需要引入Underscore的库定义。

使用flow-typed

flow-typed仓库包含了众多流行的第三方库的libdef。只需在项目根目录下创建一个名为flow-typed的文件夹,并且下载相关的定义文件即可。

为了进一步简化,可以用npm的命令行方式一键获取和安装libdef 文件:

npm install -g flow-typed

安装成功之后, 运行flow-typed install来检查package.json文件,并且下载所有项目中用到的第三方库的libdef。

自定义libdef

如果你用的库并不在flow-typed仓库,你可以创建你自己的libdef。本文不会细谈自定义libdef,因为很少会有人遇到,感兴趣可以查看此文档。

剔除类型注释

由于额外添加的类型注释不是正确的JavaScript语法,打包编译的时候需要在源码中剔除。可以通过flow-remove-types来剔除,或者如果你已经用Babel来转译JS,你可以使用Babel preset来移除。我们只讨论第一种方法。

首先需要安装flow-remove-types作为项目依赖库:

npm install --save-dev flow-remove-types

然后在package.json文件中添加另一个script 入口:

"scripts": {
 "flow": "flow",
 "build": "flow-remove-types src/ -D dest/",
}

上述命令将剔除src文件夹下的所有类型注释,在dist文件夹中保存编译后的版本。编译后的文件就是普通的能运行于浏览器的JavaScript文件。

相关内容

热门资讯

消费还扛不起大旗 来源:虎嗅APP 出品 | 妙投APP 作者 | 段明珠 牛市走了一半,消费股却像被遗忘了。202...
多措并举破解部分农村“取暖贵”... 张贤达绘(人民视觉) 近来,我国北方一些地区农村“煤改气”后,有读者网友反映,用气价格高,导致出现有...
天域半导体携手青禾晶元,共同推... 2026年1月16日,天域半导体(02658.HK)宣布与青禾晶元半导体科技(集团)有限责任公司签署...
赵建:渤海银行“变形记”——“... 通过微观个例看宏观大势,以充分理解中国经济社会深度转型的底层逻辑,在中国资产重估中寻找可以长期战略级...
爱舍伦将于1月21日北交所上市... 海牛投研1月16日消息,爱舍伦(920050)将于1月21日在北交所上市,同日从新三板摘牌。 资料显...
科创引领 向新而行——南方基金... 来源:市场资讯 (来源:泡财经) 在科技自立自强加快推进、产业结构深度重塑的时代背景下,南方基金投教...
原创 伊... 伊朗刚发出战争警告,特朗普就宣布加税25%,在这个过程中,中国成为其首个针对目标。那么,美国会不会真...
企业安全管理综合信息系统 在企业运营过程中,安全管理是一项基础且至关重要的工作。随着组织规模的扩大和业务复杂性的增加,传统依赖...
两融资金大举入场,别被起伏迷惑 窗外的夜市已经安静下来,屏幕里的行情还在跳着数字。刚刷到一条市场数据,近期科创板的两融余额又往上走了...
“1200亿合同”遭质疑 容百... 1月19日,容百科技(688005.SH)开盘即迎来大跌。截至发稿前,股价报32.04元,下跌14....
读懂丨“全球上市板”要来了!新... 企业很快将有机会在美国和新加坡两地上市,这在新加坡交易所和纳斯达克之间尚属首例。 新加坡交易所-纳斯...
固收及黄金主题产品成银行理财“... 2026年开年,国内银行理财市场热度攀升,固收类产品稳居市场主力,黄金挂钩结构性存款异军突起成为“黑...
B站粉丝第一!罗翔连续6年成百... 快科技1月17日消息,B站2025百大UP主盛典将于明晚18点召开,届时百大UP主也将正式揭晓。 B...
视涯科技IPO通过注册:年度营... 瑞财经 王敏 1月15日,据上交所官网,视涯科技股份有限公司(以下简称“视涯科技”)科创板IPO注册...
马斯克强推FSD订阅制,特斯拉... 《商业内幕》为此采访了三位特斯拉车主。对于这一调整,他们有的感到沮丧,有的感到兴奋。 反感 “这很让...
万腾平台:特朗普关税威胁引发避... 周一,黄金和白银价格创下历史新高,因特朗普威胁对欧洲征收额外关税,引发市场避险情绪上升。投资者纷纷转...
华尔街错过的5万亿:当Dewu... 1月16日,香港尖沙咀的空气里,嗅不到一丝宏观经济寒冬的气息。 在美联储降息预期不明、全球资产收益率...
保时捷在华销量连跌4年,较高点... 保时捷销量正在大幅下滑,尤其在中国市场。 保时捷官方近日公布的数据显示,该公司2025年全球销量约2...
三大指数集体低开,航天系、AI... 1月19日,A股三大指数集体低开,上证指数下跌0.27%,深证成指跌0.41%,创业板指跌0.6%。...
绽妍生物拟北交所上市:70后副... 瑞财经 刘治颖 1月13日,绽妍生物科技股份有限公司(以下简称:绽妍生物)在四川证监局完成IPO辅导...