教程

一、demo

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8"/>
<title>Hello React!</title>
<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
</head>
<body> <div id="example"></div>
<script type="text/babel">
ReactDOM.render(
<h1>Hello, world!</h1>,
document.getElementById('example')
);
</script> </body>
</html>

二、安装

官方CDN,生成环境不要用

<script src="https://cdn.staticfile.org/react/16.4.0/umd/react.development.js"></script>
<script src="https://cdn.staticfile.org/react-dom/16.4.0/umd/react-dom.development.js"></script>
<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>

需要3个库:

  • react.min.js React的核心库
  • react-dom.min.js react的dom相关库
  • babel.min.js 对不支持es6的浏览器的兼容

使用npm构建环境

教程

三、元素渲染

<div id="example"></div>
<script type="text/babel">
function tick() {
const element = (
<div>
<h2>now is {new Date().toLocaleTimeString()}</h2>
</div>
)
ReactDOM.render(element, document.getElementById('example'))
} setInterval(tick, 1000) </script>
  • 定义一个element,它是React的元素对象
  • 通过 ReactDOM.render,把元素渲染到dom里面
  • 渲染后,修改元素不会同步到dom,如果要修改dom,需要重新渲染,也就是调用render函数
  • element只能是const,也就是只读类型

每次修改dom,就要重新获取元素,所以把元素的获取封装起来,就比较方便了。可以通过元素函数,或者元素类来封装。名字必须大写开头。

元素函数

function Clock(props) {
return (
<div>
<h2>now is {props.date.toLocaleTimeString()}</h2>
</div>
);
} function tick() { ReactDOM.render(<Clock date={new Date()}/>, document.getElementById('example')); } setInterval(tick, 1000);
  • 定义了获取元素的函数Clock
  • 通过<Clock date={new Date()}/>,这个代码会执行Clock函数,然后把入参存在函数的props属性

元素类

除了可以封装为函数外,也可以封装为类。

class Clock extends React.Component {
render() {
return (
<div>
<h2>now is {this.props.date.toLocaleTimeString()}</h2>
</div>
);
}
} function tick() { ReactDOM.render(<Clock date={new Date()}/>, document.getElementById('example')); }
  • 定义Clock类,继承React.Component
  • 定义render返回元素对象
  • 通过this.props访问props属性
  • 调用方法和函数一样,都是<Clock date={new Date()}/>

四、JSX

React使用JSX来代替js

JSX是很像XML的js语法拓展

JSX优点:

  1. 执行更快
  2. 类型安全,会检查编译错误
  3. 使用模板更加快捷简单

JSX是用来在js语法中,声明React的元素的(也就是html代码),但是又不像字符串。


var arr = [
<h1>h111</h1>,
<h2>h222</h2>,
3
]
var style = {
fontSize: 100,
color : 'blue'
}
const element1 = <h1>
<p style={style} font='1px' data-myattribute="kevinlu">{1 + 1},{1 == 1 ? 2 : 3}</p>
element1
{/*注释...*/}
<div>{arr}</div>
</h1> function tick() {
ReactDOM.render(element1, document.getElementById('example')); }
  • 这样就声明了一个元素element1 ,但是又不需要用引号包裹。声明时可以换行。最外层只能是一个标签

    • 例如这样是不行的:const ele = <p>1</p><p>2</p>因为最外层有2个标签
    • 这样可以:const ele = <div><p>1</p><p>2</p></div>,通过外层加一个div标签。
  • 如果要定义自定义属性,可以:<p font='15pt' data-myattribute = "kevinlu"></p>data-前缀
  • 可以在元素中使用js表达式,用{}包裹
  • 不支持if else,但可以用三元运算符{i == 1 ? 'True!' : 'False'}
  • 支持内联样式,把样式定义在一个字典里面,然后再元素里引用字典
  • 支持数组,数据里面的元素可以是元素,元素会被逐一展示

五、组件

一个元素可以以组件形式被另一个元素引用,而且可以传参数。

function Con(props) {
return (
<div>
<h2>I am Con name={props.name}</h2>
</div>
);
} const ele = <div><Con name="kevinlu"></Con><h3>content</h3></div> ReactDOM.render(ele, document.getElementById('example'));
  • Con是一个组件,通过props.name获取入参。组件也可以用类的形式来定义
  • ele 是一个元素,引用了Con组件,然后传输了入参name="kevinlu"
  • 也可以创建复合组件,就是多个组件组合为一个新组件
  • 组件必须大写开头,区别于html默认标签都是小写开头

六、组件状态

为组件赋予状态属性,通过修改状态来动态更新DOM

class Clock extends React.Component {
constructor(props) { //构建函数
super(props);
this.state = {date: new Date()}
} componentDidMount() {//当DOM被创建(挂载),执行该函数
this.timerId = setInterval(() => this.tick(), 1000) //() => this.tick()相当于定义了一个箭头函数,()表示无入参
} componentWillUnmount() { // 当DOM被删除(卸载),执行改函数
clearInterval(this.timerId)
} tick() {//通过setState来更新state,如果直接更新,例如this.state.date=new Date(),不会同步到DOM
this.setState({date: new Date()})
} render() {
return <div>
<h1>Time</h1>
<h2>Now is {this.state.date.toLocaleTimeString()}</h2>
</div>
}
} ReactDOM.render(
<Clock></Clock>,
document.getElementById('example')
)
  • 执行ReactDOM.render 渲染Clock元素到

    • 生成一个Clock实例
    • 执行构建函数constructor
    • 执行render生成元素的html代码
    • 渲染元素的html到DOM
    • 执行componentDidMount钩子,里面设置了一个定时器,定时器ID存储在实例的变量this.timerId中
  • 定时器定时执行() => this.tick()
    • 这是一个箭头函数,直接传this.tick是不行的,因为会丢失this参数。
    • 箭头函数执行this.tick也就是Clock实例的tick函数,
    • tock函数通过setState更新Clock实例的state里面的date属性,更新为当前时间
    • setState函数会做:
      • 更新变量
      • 重新这些render生成html
      • 把html更新到DOM
      • 所以如果直接更新变量,例如this.state.date=new Date(),是不会同步到DOM的
  • 当DOM被删除componentWillUnmount函数会被执行,然后删除定时器

七、props

类型检查,元素组件支持入参,props支持类型检查,检查入参是否符合要求,如果不符合,或早console显示warning信息。

<script src="https://cdn.staticfile.org/babel-standalone/6.26.0/babel.min.js"></script>
function Clock(props) {
return <div>{props.name}</div>
} Clock.propTypes = {
name: PropTypes.string
}
ReactDOM.render(
<Clock name={123}></Clock>,
document.getElementById('example')
)
  • Clock要求的name入参类型是string
  • 要加入babel文件的引入
  • 传入了number,所以警告:
react.development.js:372 Warning: Failed prop type: Invalid prop `name` of type `number` supplied to `Clock`, expected `string`.
in Clock

支持的类型:

	// 可以声明 prop 为指定的 JS 基本数据类型,默认情况,这些数据是可选的
optionalArray: React.PropTypes.array,
optionalBool: React.PropTypes.bool,
optionalFunc: React.PropTypes.func,
optionalNumber: React.PropTypes.number,
optionalObject: React.PropTypes.object,
optionalString: React.PropTypes.string, // 可以被渲染的对象 numbers, strings, elements 或 array
optionalNode: React.PropTypes.node, // React 元素
optionalElement: React.PropTypes.element,

八、事件

元素里面的事件,需要调用函数来触发。

class Toggle extends React.Component{
constructor(props){
super(props);
this.state={isToggleOn:true}
this.handleClick=this.handleClick.bind(this) //需要绑定this
}
handleClick(){
this.setState({isToggleOn:!this.state.isToggleOn})
}
render(){
return <button onClick={this.handleClick}>
{this.state.isToggleOn?'ON':'OFF'}
</button>
} }
ReactDOM.render(
<Toggle></Toggle>,
document.getElementById('example')
)
  • 这里写了一个开关功能,点击变ON,再次点击变OFF,以此循环。
  • 设置buttom的 onClick={this.handleClick},表示当buttom被点击,会调用handleClick函数
  • 这里需要事先为handleClick函数绑定this,否则在handleClick函数里面访问this会返回undefined,而不是Toggle的实例

不需要绑定this的方法

方法1:

class ToggleTest extends React.Component {
handleClick = () => {
console.log('this is ', this)
} render() {
return <button onClick={this.handleClick}>xxx</button>
}
}
ReactDOM.render(<ToggleTest ></ToggleTest >,document.getElementById('example'))
  • 修改定义handleClick 的方法,加上() =>,这个叫:属性初始化器语法

方法2:

class ToggleTest extends React.Component {
handleClick () {
console.log('this is ', this)
} render() {
return <button onClick={(e)=>this.handleClick(e)}>xxx</button>
}
}
ReactDOM.render(<ToggleTest ></ToggleTest >,document.getElementById('example'))
  • onclick的时候,填入箭头函数。这里相当于重新定义了一个函数。如果向更低阶的元素组件传递函数,会重复构建函数,导致性能问题。

传递参数

class ToggleParam extends React.Component {
handleClick(name, e) { // e是React的事件实例,需要放在最后面
alert(name)
}
handleClick1(name) { // 也可以不要e参数
} render() {
return <button onClick={this.handleClick.bind(this, 'kevinlu')}>xxx</button>
}
}
ReactDOM.render(<ToggleParam></ToggleParam>,document.getElementById('example'))
  • e是React的事件实例,需要放在所有参数最后面,也可以不设置e参数。
  • 这里通过bind的方法,绑定this和name参数,bind会返回一个新函数,当按钮被点击,React会调用该函数
  • 可以通过e.preventDefault();来阻止默认的操作。例如点击链接会跳转到新页面,这个命令可以阻止跳转操作。

九、条件渲染

1. if条件判断

function If(props) {
if (props.isLogin) {
return <h1>is Login</h1>
} else {
return <h1>is not Login</h1>
}
} class Login extends React.Component {
constructor(props) {
super(props);
this.state = {isLogin: false}
} login() {
this.setState({isLogin: true})
} logout() {
this.setState({isLogin: false})
} render() {
if (this.state.isLogin)
return <div>now is login <button onClick={this.logout.bind(this)}>logout</button></div>
else
return <div>now is logout <button onClick={this.login.bind(this)}>login</button></div>
} }
ReactDOM.render(<If isLogin={false}></If>, document.getElementById('example'))
  • If和Login两个元素组件,都是根据属性,来判断render的返回html

2.&&运算符

function At(props) {
return <div>kkkkk {props.number > 0 && <h2>number is {props.number} </h2>}</div>
}
  • 如果props.number,显示元素<h2>number is {props.number} </h2>

3. 三目运算符

和IF类似,不过它可以在元素里面通过{}的方式定义:{props.number ?"large 0":'least 0'}

4. 阻止渲染

直接让元素组件返回null,就不会渲染这个元素。

function NoneTest(props) {
if (props.number > 0)
return <div>number is {props.number}</div>
else
return null
}

十、数组

var l = [1, 2, 3, 4, 5]
const ele = <ul>
{l.map(
(number) => <li key={number}>{number}</li>
)}
</ul>
ReactDOM.render(ele, document.getElementById('example'))
  • 通过JS的map方法,遍历l数组,对每个元素执行(number) => <li>{number}</li>函数,返回的是元素数组
  • 对于元素列表,React会遍历数组,逐个显示,所以url标签里面会有5个li标签
  • 数组构建的元素,需要包含一个唯一的key,不然会报错。key一般是数组元素的id,如果没有id可以用下标,即(number,index) => <li key={index}>{number}</li>

十一、组件API

  • setState(object nextState[, function callback]) 设置状态

    *callback可选,会在修改状态后,渲染DOM后被调用

    • 没有设置的key,不会被替换
  • replaceState(object nextState[, function callback]) 替换状态
    • 和setState类似,区别是没有设置的key,会被设置为undefined
  • setProps(object nextProps[, function callback]) 设置props
  • replaceProps(object nextProps[, function callback])
  • forceUpdate([function callback]) 通知组件重绘DOM
  • DOMElement findDOMNode() 获取组件对应的DOM元素,如果没有,返回null
  • bool isMounted() 判断组件挂载状态

十二、组件生命周期

生命周期有:

  • Mounting 已挂载,也就是已插入到真实的DOM
  • Updating 更新中,重新渲染
  • Unmounting 已移出真实的DOM

方法:

  • componentWillMount 组件将要挂载前调用
  • componentDidMount 组件挂载完成调用
  • componentWillReceiveProps 组件接收一个新的prop时被调用,第一次渲染不会调用。主要用户组件的更新
  • shouldComponentUpdate 返回bool值,在组件接收到新的props或者state时被调用,第一次渲染不调用
  • componentWillUpdate 在组件接收到新的props或者state但还没有render时被调用
  • componentDidUpdate 更新完成后调用,第一次渲染不调用
  • componentWillUnmount 组件被移除前调用

十三、AJAX

一般在componentDidMount 中通过AJAX获取数据,然后通过setState更新组件。

class AjaxTest extends React.Component{
constructor(props) {
super(props);
this.state={content:"test"}
} componentDidMount() {
this.req = $.get('/demo/react_demo/data', function (res) {
this.setState({content: res})
}.bind(this))
}
componentWillUnmount(){
this.req.abort()
}
render(){
return <div>{this.state.content}</div>
}
}
  • 这里用了jq的ajax方法,导入:`
  • get的回调函数需要bind this 不然回调函数里面访问this,不会执行组件实例
  • componentWillUnmount当组件被卸载,关闭http请求

    `

十四、表单

class Form extends React.Component {
constructor(props) {
super(props);
this.state = {name: "unknow"}
} onAttrChange(key, e) {
var state = {}
state[key] = e.target.value
this.setState(state, () => console.info('new name', this.state.name))
} render() {
return <div>名字:<input value={this.state.name} onChange={this.onAttrChange.bind(this, 'name')}></input></div>
}
}
  • 表单的值修改后,需要通过onChange事件,更新新的值到组件的变量

如果是嵌套组件,需要把父组件的this传给子组件

function Input(props) {
return <input value={props.this.state.name} onChange={props.this.onAttrChange.bind(props.this, 'name')}></input>
} class Form extends React.Component {
constructor(props) {
super(props);
this.state = {name: "unknow"}
} onAttrChange(key, e) {
var state = {}
state[key] = e.target.value
this.setState(state, () => console.info('new name', this.state.name))
} render() {
return <div>名字:<Input this={this}></Input></div>
}
}

Select

class Select extends React.Component {
constructor(props) {
super(props);
this.state = {value: "unknow"}
} onAttrChange(key, e) {
var state = {}
state[key] = e.target.value
this.setState(state, () => console.info('new value', this.state.value))
} render() {
return <div><select value={this.state.value} onChange={this.onAttrChange.bind(this, 'value')}>
<option value="gg">Google</option>
<option value="fb">Facebook</option>
</select></div>
}
}

十五、Ref

我们可以在元素里面的某个标签,设置一个id,然后通过ref,获取到这个标签。

class RefTest extends React.Component {
constructor(props) {
super(props);
} onClick() {
var name = this.refs.myRef //获取到input实例
console.info(name.value)//获取input的值
} render() {
return <div>
name: <input ref="myRef"></input>
<button onClick={this.onClick.bind(this)}>BTN</button>
</div>
}
}
  • 这里我们为input 加了一个ref属性
  • 在onClick里,通过this.refs.myRef可以获取到这个input的实例,然后获取它的value值

React教程的更多相关文章

  1. [转]React 教程

    本文转自:http://www.runoob.com/react/react-install.html React 可以直接下载使用,下载包中也提供了很多学习的实例. 本教程使用了 React 的版本 ...

  2. SAP官网发布的react教程

    大家学习React的时候,用的是什么教程呢?Jerry当时用的阮一峰博客上的入门教程,因为React使用的JSX语法并不是所有的浏览器都支持,所以还得使用browser.js在浏览器端将JSX转换成J ...

  3. 看完阮一峰的React教程后, 我写了一个TodoList

    看完阮一峰的React教程后,就自己做了这个TodoList,自己慢慢琢磨效率差了点但是作为入门小练习还是不错的. 以下是效果图:我的源码:todolistUI:bootstrap 4 一.组件化 我 ...

  4. react教程 — 性能优化

    参考:https://segmentfault.com/a/1190000007811296?utm_medium=referral&utm_source=tuicool  或  https: ...

  5. (私人收藏)React教程手册

    React教程手册 https://pan.baidu.com/s/1ka2PJ54HgqJ8lC6XgbvdLg pedx React 教程 含有3个附件,如下: react.js react-do ...

  6. React教程(一) React介绍与搭建

    React的介绍: React来自于Facebook公司的开源项目 React 可以开发单页面应用 spa(单页面应用) react 组件化模块化 开发模式 React通过对DOM的模拟(虚拟dom) ...

  7. React 教程

    React 入门实例教程 http://www.ruanyifeng.com/blog/2015/03/react.html React 测试入门教程http://www.ruanyifeng.com ...

  8. 学习react教程

    网址收藏: React官网,React的Github,React的中文文档 1.react是什么? React起源于Facebook的内部项目,因为该公司对市场上所有的Javascript MVC框架 ...

  9. React教程:4 个 useState Hook 示例

    摘要: React示例教程. 原文:快速了解 React Hooks 原理 译者:前端小智 到 React 16.8 目前为止,如果编写函数组件,然后遇到需要添加状态的情况,咱们就必须将组件转换为类组 ...

  10. react入门(1)之阮一峰react教程

    阮一峰老师的github地址:React Demos React 入门实例教程 2.ReactDOM.render() // ReactDOM.render() 将模板转化为 HTML 语言 // 参 ...

随机推荐

  1. SpringBoot整合mybatis-plus并实现代码自动生成

    1.引入maven <dependency> <groupId>org.springframework.boot</groupId> <artifactId& ...

  2. 【LeetCode】485. Max Consecutive Ones 解题报告(Java & Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 Java解法 Python解法 日期 [LeetCo ...

  3. 【LeetCode】971. Flip Binary Tree To Match Preorder Traversal 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 前序遍历 日期 题目地址:https://leetc ...

  4. 【LeetCode】380. Insert Delete GetRandom O(1) 解题报告(Python)

    [LeetCode]380. Insert Delete GetRandom O(1) 解题报告(Python) 作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxu ...

  5. 【LeetCode】846. Hand of Straights 解题报告(Python & C+)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  6. 【剑指Offer】扑克牌顺子 解题报告(Python)

    [剑指Offer]扑克牌顺子 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-interviews 题目描 ...

  7. 【LeetCode】831. Masking Personal Information 解题报告(Python)

    [LeetCode]831. Masking Personal Information 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzh ...

  8. 1686 第K大区间

    1686 第K大区间 时间限制:1 秒 空间限制:131072 KB   定义一个区间的值为其众数出现的次数.现给出n个数,求将所有区间的值排序后,第K大的值为多少. 众数(统计学/数学名词)_百度百 ...

  9. Python_time&datetime

    获取常用日常时间 # encoding: utf-8 import time import datetime # 当前时间 datetime_now_time = datetime.datetime. ...

  10. js 模块化 -- export 时 一个默认和多个默认的写法

    js文件 只有一个  food 类,一般写 export {food} 或者 export {food as default} 那么如果有多个呢? 这样 export {food, food2} 或者 ...