Airbnb React/JSX 编码规范
Airbnb React/JSX 编码规范
算是最合理的React/JSX编码规范之一了
内容目录
- 基本规范
- Class vs React.createClass vs stateless
- 命名
- 声明模块
- 代码对齐
- 单引号还是双引号
- 空格
- 属性
- Refs引用
- 括号
- 标签
- 函数/方法
- 模块生命周期
- isMounted
Basic Rules 基本规范
- 每个文件只写一个模块.
- 但是多个无状态模块可以放在单个文件中. eslint:
react/no-multi-comp
.
- 但是多个无状态模块可以放在单个文件中. eslint:
- 推荐使用JSX语法.
- 不要使用
React.createElement
,除非从一个非JSX的文件中初始化你的app.
创建模块
Class vs React.createClass vs stateless
如果你的模块有内部状态或者是
refs
, 推荐使用class extends React.Component
而不是React.createClass
,除非你有充足的理由来使用这些方法.
eslint:react/prefer-es6-class
react/prefer-stateless-function
// badconst Listing = React.createClass({
// ...render() {
return <div>{this.state.hello}</div>;
}
}); // goodclass Listing extends React.Component {
// ...render() {
return <div>{this.state.hello}</div>;
}
}如果你的模块没有状态或是没有引用
refs
, 推荐使用普通函数(非箭头函数)而不是类:// badclass Listing extends React.Component {
render() {
return <div>{this.props.hello}</div>;
}
} // bad (relying on function name inference is discouraged)const Listing = ({ hello }) => (
<div>{hello}</div>
); // goodfunction Listing({ hello }) {
return <div>{hello}</div>;
}
Naming 命名
- 扩展名: React模块使用
.jsx
扩展名. - 文件名: 文件名使用驼峰式. 如,
ReservationCard.jsx
. 引用命名: React模块名使用驼峰式命名,实例使用骆驼式命名. eslint:
react/jsx-pascal-case
// badimport reservationCard from './ReservationCard'; // goodimport ReservationCard from './ReservationCard'; // badconst ReservationItem = <ReservationCard />; // goodconst reservationItem = <ReservationCard />;
模块命名: 模块使用当前文件名一样的名称. 比如
ReservationCard.jsx
应该包含名为ReservationCard
的模块. 但是,如果整个文件夹是一个模块,使用index.js
作为入口文件,然后直接使用index.js
或者文件夹名作为模块的名称:// badimport Footer from './Footer/Footer'; // badimport Footer from './Footer/index'; // goodimport Footer from './Footer';
高阶模块命名: 对于生成一个新的模块,其中的模块名
displayName
应该为高阶模块名和传入模块名的组合. 例如, 高阶模块withFoo()
, 当传入一个Bar
模块的时候, 生成的模块名displayName
应该为withFoo(Bar)
.为什么?一个模块的
displayName
可能会在开发者工具或者错误信息中使用到,因此有一个能清楚的表达这层关系的值能帮助我们更好的理解模块发生了什么,更好的Debug.// bad export default function withFoo(WrappedComponent) {
return function WithFoo(props) {
return <WrappedComponent {...props} foo />;
}
} // good export default function withFoo(WrappedComponent) {
function WithFoo(props) {
return <WrappedComponent {...props} foo />;
} const wrappedComponentName = WrappedComponent.displayName || WrappedComponent.name || 'Component'; WithFoo.displayName = `withFoo(${wrappedComponentName})`;
return WithFoo;
}属性命名: 避免使用DOM相关的属性来用作其他的用途。
为什么?对于
style
和className
这样的属性名,我们都会默认它们代表一些特殊的含义,如元素的样式,CSS class的名称。在你的应用中使用这些属性来表示其他的含义会使你的代码更难阅读,更难维护,并且可能会引起bug。// bad
<MyComponent style="fancy" /> // good
<MyComponent variant="fancy" />
Declaration 声明模块
不要使用
displayName
来命名React模块,而是使用引用来命名模块, 如 class 名称.// badexport default React.createClass({
displayName: 'ReservationCard',
// stuff goes here
}); // goodexport default class ReservationCard extends React.Component {
}
Alignment 代码对齐
遵循以下的JSX语法缩进/格式. eslint:
react/jsx-closing-bracket-location
// bad
<Foo superLongParam="bar"anotherSuperLongParam="baz" /> // good, 有多行属性的话, 新建一行关闭标签
<FoosuperLongParam="bar"anotherSuperLongParam="baz"
/> // 若能在一行中显示, 直接写成一行
<Foo bar="bar" /> // 子元素按照常规方式缩进
<FoosuperLongParam="bar"anotherSuperLongParam="baz"
>
<Quux />
</Foo>
Quotes 单引号还是双引号
对于JSX属性值总是使用双引号(
"
), 其他均使用单引号. eslint:jsx-quotes
为什么? JSX属性 不能包括转译的引号, 因此在双引号里包括像
"don't"
的属性值更容易输入. HTML属性也是用双引号,所以JSX属性也遵循同样的语法.// bad
<Foo bar='bar' /> // good
<Foo bar="bar" /> // bad
<Foo style={{ left: "20px" }} /> // good
<Foo style={{ left: '20px' }} />
Spacing 空格
总是在自动关闭的标签前加一个空格,正常情况下也不需要换行. eslint:
no-multi-spaces
,react/jsx-space-before-closing
// bad
<Foo/> // very bad
<Foo /> // bad
<Foo
/> // good
<Foo />不要在JSX
{}
引用括号里两边加空格. eslint:react/jsx-curly-spacing
// bad
<Foo bar={ baz } /> // good
<Foo bar={baz} />
Props 属性
JSX属性名使用骆驼式风格
camelCase
.// bad
<FooUserName="hello"phone_number={12345678}
/> // good
<FoouserName="hello"phoneNumber={12345678}
/>如果属性值为
true
, 可以直接省略. eslint:react/jsx-boolean-value
// bad
<Foohidden={true}
/> // good
<Foohidden
/><img>
标签总是添加alt
属性. 如果图片以presentation(感觉是以类似PPT方式显示?)方式显示,alt
可为空, 或者<img>
要包含role="presentation"
. eslint:jsx-a11y/img-has-alt
// bad
<img src="hello.jpg" /> // good
<img src="hello.jpg" alt="Me waving hello" /> // good
<img src="hello.jpg" alt="" /> // good
<img src="hello.jpg" role="presentation" />不要在
alt
值里使用如 "image", "photo", or "picture"包括图片含义这样的词, 中文也一样. eslint:jsx-a11y/img-redundant-alt
为什么? 屏幕助读器已经把
img
标签标注为图片了, 所以没有必要再在alt
里说明了.// bad
<img src="hello.jpg" alt="Picture of me waving hello" /> // good
<img src="hello.jpg" alt="Me waving hello" />使用有效正确的 aria
role
属性值 ARIA roles. eslint:jsx-a11y/aria-role
// bad - not an ARIA role
<div role="datepicker" /> // bad - abstract ARIA role
<div role="range" /> // good
<div role="button" />不要在标签上使用
accessKey
属性. eslint:jsx-a11y/no-access-key
为什么? 屏幕助读器在键盘快捷键与键盘命令时造成的不统一性会导致阅读性更加复杂.
// bad
<div accessKey="h" /> // good
<div />避免使用数组的index来作为属性
key
的值,推荐使用唯一ID. (为什么?)// bad
{todos.map((todo, index) =>
<Todo{...todo}key={index}
/>
)} // good
{todos.map(todo => (
<Todo{...todo}key={todo.id}
/>
))}
Refs
总是在Refs里使用回调函数. eslint:
react/no-string-refs
// bad
<Fooref="myRef"
/> // good
<Fooref={ref => { this.myRef = ref; }}
/>
Parentheses 括号
将多行的JSX标签写在
()
里. eslint:react/wrap-multilines
// badrender() {
return <MyComponent className="long body" foo="bar">
<MyChild />
</MyComponent>;
} // goodrender() {
return (
<MyComponent className="long body" foo="bar">
<MyChild />
</MyComponent>
);
} // good, 单行可以不需要render() {
const body = <div>hello</div>;
return <MyComponent>{body}</MyComponent>;
}
Tags 标签
对于没有子元素的标签来说总是自己关闭标签. eslint:
react/self-closing-comp
// bad
<Foo className="stuff"></Foo> // good
<Foo className="stuff" />如果模块有多行的属性, 关闭标签时新建一行. eslint:
react/jsx-closing-bracket-location
// bad
<Foobar="bar"baz="baz" /> // good
<Foobar="bar"baz="baz"
/>
Methods 函数
使用箭头函数来获取本地变量.
function ItemList(props) {
return (
<ul>
{props.items.map((item, index) => ( <Item key={item.key} onClick={() => doSomethingWith(item.name, index)} /> ))}
</ul>
);
}当在
render()
里使用事件处理方法时,提前在构造函数里把this
绑定上去. eslint:react/jsx-no-bind
为什么? 在每次
render
过程中, 再调用bind
都会新建一个新的函数,浪费资源.// badclass extends React.Component {
onClickDiv() {
// do stuff
} render() {
return <div onClick={this.onClickDiv.bind(this)} />
}
} // goodclass extends React.Component {
constructor(props) {
super(props); this.onClickDiv = this.onClickDiv.bind(this);
} onClickDiv() {
// do stuff
} render() {
return <div onClick={this.onClickDiv} />
}
}在React模块中,不要给所谓的私有函数添加
_
前缀,本质上它并不是私有的.为什么?
_
下划线前缀在某些语言中通常被用来表示私有变量或者函数。但是不像其他的一些语言,在JS中没有原生支持所谓的私有变量,所有的变量函数都是共有的。尽管你的意图是使它私有化,在之前加上下划线并不会使这些变量私有化,并且所有的属性(包括有下划线前缀及没有前缀的)都应该被视为是共有的。了解更多详情请查看Issue#1024, 和 #490 。// badReact.createClass({
_onClickSubmit() {
// do stuff
}, // other stuff
}); // goodclass extends React.Component {
onClickSubmit() {
// do stuff
} // other stuff
}在
render
方法中总是确保return
返回值. eslint:react/require-render-return
// badrender() {
(<div />);
} // goodrender() {
return (<div />);
}
Ordering React 模块生命周期
class extends React.Component
的生命周期函数:
- 可选的
static
方法 constructor
构造函数getChildContext
获取子元素内容componentWillMount
模块渲染前componentDidMount
模块渲染后componentWillReceiveProps
模块将接受新的数据shouldComponentUpdate
判断模块需不需要重新渲染componentWillUpdate
上面的方法返回true
, 模块将重新渲染componentDidUpdate
模块渲染结束componentWillUnmount
模块将从DOM中清除, 做一些清理任务- 点击回调或者事件处理器 如
onClickSubmit()
或onChangeDescription()
render
里的 getter 方法 如getSelectReason()
或getFooterContent()
- 可选的 render 方法 如
renderNavigation()
或renderProfilePicture()
render
render() 方法
如何定义
propTypes
,defaultProps
,contextTypes
, 等等其他属性...import React, { PropTypes } from 'react'; const propTypes = {
id: PropTypes.number.isRequired,
url: PropTypes.string.isRequired,
text: PropTypes.string,
}; const defaultProps = {
text: 'Hello World',
}; class Link extends React.Component {
static methodsAreOk() {
return true;
} render() {
return <a href={this.props.url} data-id={this.props.id}>{this.props.text}</a>
}
} Link.propTypes = propTypes;
Link.defaultProps = defaultProps; export default Link;React.createClass
的生命周期函数,与使用class稍有不同: eslint:react/sort-comp
displayName
设定模块名称propTypes
设置属性的类型contextTypes
设置上下文类型childContextTypes
设置子元素上下文类型mixins
添加一些mixinsstatics
defaultProps
设置默认的属性值getDefaultProps
获取默认属性值getInitialState
或者初始状态getChildContext
componentWillMount
componentDidMount
componentWillReceiveProps
shouldComponentUpdate
componentWillUpdate
componentDidUpdate
componentWillUnmount
- clickHandlers or eventHandlers like
onClickSubmit()
oronChangeDescription()
- getter methods for
render
likegetSelectReason()
orgetFooterContent()
- Optional render methods like
renderNavigation()
orrenderProfilePicture()
render
isMounted
不要再使用
isMounted
. eslint:react/no-is-mounted
为什么?
isMounted
反人类设计模式:(), 在 ES6 classes 中无法使用, 官方将在未来的版本里删除此方法.
Airbnb React/JSX 编码规范的更多相关文章
- 【转】Airbnb React编码规范
Airbnb的编码规范是在业界非常流行的一套规范,而且它一直都在进化,推出最新技术的规范 原文:https://zhuanlan.zhihu.com/p/20616464 用更合理的方式书写React ...
- 【转】JavaScript 风格指南/编码规范(Airbnb公司版)
原文转自:http://blog.jobbole.com/79484/ Airbnb 是一家位于美国旧金山的公司,本文是其内部的 JavaScript 风格指南/编码规范,在 Github 上有 11 ...
- JavaScript 编码规范(中文/Airbnb公司版)
Airbnb 是一家位于美国旧金山的公司,本文是其内部的 JavaScript编码规范,写得比较全面,在 Github 上有 16,686 + Star,3,080 + fork,前端开发人员可参考. ...
- HTML/CSS/JS编码规范
最近整理了一份HTML/CSS/JS编码规范,供大家参考.目录:一.HTML编码规范二.CSS编码规范三.JS编码规范 一.HTML编码规范 1. img标签要写alt属性 根据W3C标准,img标签 ...
- 【go】脑补框架 Express beego tornado Flux reFlux React jsx jpg-ios出品
http://goexpresstravel.com/ 今天 Express 的作者 TJ Holowaychuk 发了一篇文章,正式宣告和 Node.js 拜拜了,转向 Go 语言. Go vers ...
- python编码规范、js编码规范及IDE的检查插件pylint/eslint等
一.python规范 参考:https://zh-google-styleguide.readthedocs.io/en/latest/google-python-styleguide/的风格规范和语 ...
- Android的编码规范
一.Android编码规范 1.学会使用string.xml文件 在我看来,当一个文本信息出现的次数大于一次的时候就必须要使用string.xml 比如一个保存按钮 , 不规范写法: <Butt ...
- PHP 高级编程(1/5) - 编码规范及文档编写
PHP 高级程序设计学习笔记20140612 软件开发中的一个重要环节就是文档编写.他可以帮助未来的程序维护人员和使用者理解你在开发时的思路.也便于日后重新查看代码时不至于无从下手.文档还有一个重要的 ...
- 【原】JAVA SE编码规范
/* * 编码规范: * 1.所有的命名遵循"见名知意"的原则 * 2.所有的命名不允许使用汉字或拼音 * 3.Java的工程命名建议使用小写,比如:oa.crm.cms... * ...
随机推荐
- 搭建Linux C语言开发环境
1.操作系统 Windows操作系统:windows 7 and windows 10 2.开发工具和编译工具 开发工具:notpad++ 和 vim 编译工具:Cygwin64 Terminal 3 ...
- HTML-参考手册: HTML 全局属性
ylbtech-HTML-参考手册: HTML 全局属性 1.返回顶部 1. HTML 全局属性 New : HTML5 新属性. 属性 描述 accesskey 设置访问元素的键盘快捷键. clas ...
- 抓包工具tcpdump用法说明--1
本文目录: 1.1 tcpdump选项 1.2 tcpdump表达式 1.3 tcpdump示例 tcpdump采用命令行方式对接口的数据包进行筛选抓取,其丰富特性表现在灵活的表达式上. 不带任何选项 ...
- ArcMap基于Oracle出现sde.instances_util.check_instance_table_conflicts:: ORA-00942:表或视图不存在/table or view doesnot exist解决思路
SDE环境:Oracle12C+ArcMap10.7+WinServer2012 出现问题情况: 1.SDE可以连接正常打开,但就是无法新建要素.导入要素等: 1)在根目录新建或导入要素,弹出提示: ...
- VTemplate模板引擎的使用--进阶篇
1.<vt:template>与<vt:include>标签的不同 <vt:template>和<vt:include> 标签都包含file属性,如果这 ...
- js 中 !!的用法
!!是将表达式强制转化为bool值的运算,运算结果为true或false,表达式是什么值,结果就是对应的bool值,不再取非. 不是取非再取非的意思!!! !!false=false; 要注意f ...
- HashCode方法整理
哈希表这个数据结构想必大多数人都不陌生,而且在很多地方都会利用到hash表来提高查找效率.在Java的Object类中有一个方法: 1 public native int hashCode(); 根据 ...
- Apache的虚拟主机功能(基于IP、域名、端口号)
Apache虚拟主机就是在一个Apache服务器上配置多个虚拟主机,实现一个服务器提供多站点服务,其实就是访问同一个服务器上的不同目录. 主要有三种方法: 1.通过不同的IP地址 2.通过不同的域名 ...
- 03、python的基础-->str字符串的使用
1.字符串首字母大写 s = "aksjdjjhfhdhjaGGGGkkk" s1 = s.capitalize() print(s1) 2.字符串全部字母大写 s = " ...
- leetcode.数组.769最多能完成排序的块-Java
1. 具体题目 数组arr是[0, 1, ..., arr.length - 1]的一种排列,我们将这个数组分割成几个“块”,并将这些块分别进行排序.之后再连接起来,使得连接的结果和按升序排序后的原数 ...