flow不是React Native必会的技能,但是作为正式的产品开发优势很有必要掌握的技能之一。所以,算是RN填坑之旅系列的番外篇。

Flow是一个静态的检查类型检查工具,设计之初的目的就是为了可以发现JavaScript脚本里不容易被发现的错误。在js开发的过程中,总会遇到一些问题。小的还可以,比如用alert或者console等输出一些信息可以debug,并解决。但是如果项目比较大的时候,这些手法只能起到一定的辅助作用。更有甚者,有些问题不运行到那段代码,根本不会发现错误。Facebook的兄弟们就是为了解决这个问题,于是开发了flow。

首先,跳转的你的项目目录下。然后就开始正文了。

安装&配置

Flow就从安装开始。

npm install --save-dev flow-bin

创建配置文件。

touch .flowconfig

先不管空白的.flowconfig配置文件。在package.json文件里flow脚本。

your project/package.json

  "scripts": {
"flow": "flow; test $? -eq 0 -o $? -eq 2",
},

然后给需要flow检查的文件里加上//@flow或者/*@flow*/。然后就可以检查了。(也可以在命令中加上--all, 这样就会检查所有文件)。

在根目录下运行命令:

npm run flow

用Flow检查

现在就把flow用起来。Flow绝不是上面几个命令而已,而是一套类型体系。下面通过一个例子来了解一下flow和flow的类型体系。

export default function(state = initialState, action) {
switch (action.type) {
case actionTypes.TRACKS_SET:
return setTracks(state, action);
case actionTypes.TRACK_PLAY:
return setPlay(state, action);
}
return state;
} function setTracks(state, action) {
const { tracks } = action;
return { ...state, tracks };
}

第一次检查会出很多的错,因为上面的写法没有按照flow的定义添加类型声明。下面来添加类型声明。

// @flow

import * as actionTypes from './actionTypes';

const initialState = {
tracks: [],
activeTrack: null
}; export default function(state = initialState, action) {
switch (action.type) {
case actionTypes.TRACKS_SET:
return setTracks(state, action);
case actionTypes.TRACK_PLAY:
return setPlay(state, action);
}
return state;
} function setTracks(state, action) {
const { tracks } = action;
return { ...state, tracks };
} function setPlay(state, action) {
const { track } = action;
return { ...state, activeTrack: track };
}

运行命令check命令之后会显示错误的内容:

test/track.js:10
10: export default function(state = initialState, action) {
^^^^^ parameter `state`. Missing annotation test/track.js:10
10: export default function(state = initialState, action) {
^^^^^^ parameter `action`. Missing annotation

flow的错误提示我们,需要给出方法的参数类型。

Flow: Any类型

修改代码:

export default function(state: any = initialState, action: any) {
switch (action.type) {
case actionTypes.TRACKS_SET:
return setTracks(state, action);
case actionTypes.TRACK_PLAY:
return setPlay(state, action);
}
return state;
}

这样修改之后就没有什么错误提示了。我们给参数指定了any类型。这个类型是所有类型的父类型,也是所有类型的子类型。所以,任何类型都可以用any代表了。但是这样并不能发挥类型检查的优势。

Flow:类型别名

使用flow的类型别名可以解决上面的问题。输出的默认方法的第一个参数其实是一个State类型的实例。在本例中使用的State是一个对象,其中tracks是一个数组,activeTrack是一个可以为空的对象。为State定义一个类型别名:

type State = {
tracks: Array<any>,
activeTrack: ?any
}; const initialState = {
tracks: [],
activeTrack: null
};

正好之前定义的initialState就是State类型的一个实例。同理,我们也可以为initialStateactiveTrack定义一个类型。

type Track = {
//这里给出定义
}

然后State类型就是这样的了:

type Track = {
//这里给出定义
} type State = {
tracks: Array<any>,
activeTrack: ?Track
};

注意,这里我们用到了一个特殊的类型:Maybe Type(可能类型或者可空类型)。这个类型的定义方式就是在类型的前面放一个问号。

下面也为两个方法setTrackssetPlay定义返回的类型,并应用到对应的方法上:

// @flow

import * as actionTypes from './actionTypes';

type Track = {
trackName: string
}; type State = {
tracks: Array<any>,
activeTrack: ?Track
}; type SetTrackAction = {
type: string,
tracks: Array<Track>
}; type PlayTrackAction = {
type: string,
track: Track
}; const initialState = {
tracks: [],
activeTrack: null
}; export function setTracks(tracks: Array<Track>): SetTrackAction {
return {
type: actionTypes.TRACKS_SET,
tracks
};
} export function setPlay(track: Track): PlayTrackAction {
return {
type: actionTypes.TRACK_PLAY,
track: track
};
}

Flow: Type Union

SetTrackActionPlayTrackAction可以使用Type Union的方式统一起来:

type Action = SetTrackAction | PlayTrackAction;

修改代码:

export function setTracks(tracks: Array<Track>): Action {
return {
type: actionTypes.TRACKS_SET,
tracks
};
} export function setPlay(track: Track): Action {
return {
type: actionTypes.TRACK_PLAY,
track: track
};
}

Flow: 模块处理

这里主要说明一种情况。如果引入的另外一个模块和本模块定义了一个同名的类型别名,但是里面包含的内容不同,那么Flow会检查出来并报错。

比如,现在我们的模块里有了Track这个类型,是这样的:

type Track = {
trackCode: string
};

如果在引入的actionTypes.js文件中也包含一个Track类型,但是定义的有些不同:

type Track = {
trackCode: number
};

两个Track的不同就在于trackCode的类型,一个是string,一个是number

运行flow之后就会显示出来具体的报错:

test/actionTypes.js:13
13: trackCode: 123
^^^ number. This type is incompatible with the expected return type of
5: trackCode: string
^^^^^^ string

Flow: 声明类型

上面的问题解决起来很简单,把两个类型的定义保持一致就可以。但是,我们不可能在任何一个需要Track类型的文件中都定义一个一模一样的类型。

Flow提供了一种特殊的类型声明方式,可以一次声明到处使用。

.flowconfig文件中的[lib]下添加如下内容。如果这个文件为空的话,运行flow init命令。

[libs]
decls

在根目录下:

mkdir decls
cd decls
touch flowTypes.js

在文件flowType.js中:

declare type Track = {
trackCode: string;
};

把其他的Track类型声明全部都删掉,然后运行命令:

npm run flow

最后

Flow对React的支持与上文所述的基本上大同小异。各位可以移步官网细看。

React Native填坑之旅--Flow篇(番外)的更多相关文章

  1. React Native填坑之旅--class(番外篇)

    无论React还是RN都已经迈入了ES6的时代,甚至凭借Babel的支持都进入了ES7.ES6内容很多,本文主要讲解类相关的内容. 构造函数 定义侦探类作为例子. ES5的"类"是 ...

  2. React Native填坑之旅--布局篇

    代码在这里: https://github.com/future-challenger/petshop/tree/master/client/petshop/src/controller 回头看看RN ...

  3. React Native填坑之旅--Navigation篇

    React Native的导航有两种,一种是iOS和Android通用的叫做Navigator,一种是支持iOS的叫做NavigatorIOS.我们这里只讨论通用的Navigator.会了Naviga ...

  4. React Native填坑之旅--ListView篇

    列表显示数据,基本什么应用都是必须.今天就来从浅到深的看看React Native的ListView怎么使用.笔者写作的时候RN版本是0.34. 最简单的 //@flow import React f ...

  5. React Native填坑之旅--Button篇

    从React过来,发现React Native(以下简称RN)居然没有Button.隔壁的iOS是有UIButton的,隔壁的隔壁的Android里也是有的.没有Button,就没有点击效果啊.这还真 ...

  6. React Native填坑之旅--LayoutAnimation篇

    比较精细的动画可以用Animated来控制.但是,在一些简单的界面切换.更新的时候所做的动画里再去计算开始值.结束值和插值器如何运作绝对是浪费时间. RN正好给我们提供了LayoutAnimation ...

  7. React Native填坑之旅--与Native通信之iOS篇

    终于开始新一篇的填坑之旅了.RN厉害的一个地方就是RN可以和Native组件通信.这个Native组件包括native的库和自定义视图,我们今天主要设计的内容是native库方面的只是.自定义视图的使 ...

  8. React Native填坑之旅--重新认识RN

    如同黑夜里的一道光一样,就这么知道了F8. F8是每年一次Facebook每年一次的开发者大会.每次大会都会release相应的APP,iOS.Android都有.之前都是用Native开发的,但是2 ...

  9. React Native填坑之旅--组件生命周期

    这次我们来填React Native生命周期的坑.这一点非常重要,需要有一个清晰的认识.如果你了解Android或者iOS的话,你会非常熟悉我们今天要说的的内容. 基本上一个React Native的 ...

随机推荐

  1. Android笔记之——事件的发生(2)

    Java: package com.example.helloworld;import android.os.Bundle;import android.view.KeyEvent;import an ...

  2. HashTable 简述

    1.解释:使用了映射函数,把值映射到对应的位置,key-> address, address是表中的存储位置,不是实际的地址:   2.Hash 函数设计, 分布合理,冲突少,利用率平衡,利用率 ...

  3. Python中的range函数用法

    函数原型:range(start, end, scan): 参数含义:start:计数从start开始.默认是从0开始.例如range(5)等价于range(0, 5); end:技术到end结束,但 ...

  4. 关于有偿提供拼图响应式后台的通知---------pintuer ui的官方通知(www.pintuer.com)

    拼图响应式前端框架版响应式后台正式发布. 考虑到目前拼图的状况,我们不打算免费开放下载,但也不会收各位朋友1分钱,该版后台将有偿提供给各位给予拼图贡献的朋友. 废话不多说,一切皆以有图有真相,下面上图 ...

  5. jquery 时间控件怎么能禁止输入只能选择日期?

    jsp一个input输入框用的是easyui时间控件,现在问题是如何是这个input只能点击选择日期,而禁止手动输入 解决方法::: 在日期的input标签里面添加::::editable=" ...

  6. js对象的继承以及公有私有属性的定义和读写

    最近想写一些js工具,有些方面需要用到面向对象的方法,比如继承父类属性和方法.通过私有化隐藏某些对象的属性等,因为没有系统的学习js,所以不知道怎么做,觉得很伤脑筋. 今天受到技术群里朋友的提示,并查 ...

  7. dojo布局(layout)

    使用BorderContainer和ContentPane实现布局 1.效果图如下: 2.HTML代码: <div id="appLayout" class="de ...

  8. python之socket 网络编程

    提到网络通信不得不复习下osi七层模型: 七层模型,亦称OSI(Open System Interconnection)参考模型,是参考模型是国际标准化组织(ISO)制定的一个用于计算机或通信系统间互 ...

  9. [OC笔记] Category分类之见解

    用过别的语言做过开发的同学都知道,如果你想扩充一个类,就应该去继承这个类.但是OC里面有更好的方法,那就是分类. 那什么是分类呢?就是在不改变原先类,我们可以在其中添加咱们自定义的方法,这样和同事合作 ...

  10. Request header is too large

    当request url过长的时候,经常tomcat后台就会包 "Request header is too large"错误. 解决方法:修改server.xml文件,linux ...