React 快速上手 - 04 基础特性 JSX、Props、State、Lifecycle、Event、Style

前端之家收集整理的这篇文章主要介绍了React 快速上手 - 04 基础特性 JSX、Props、State、Lifecycle、Event、Style前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

目录

React 快速上手 - 04 基础特性 JSX、Props、State、Lifecycle、Event、Style

目标

掌握 react 的基础特性

  • 语法 JSX
  • 属性 Props
  • 状态 State
  • 生命周期 Lifecycle
  • 事件 Event
  • 样式 Style

基础特性

react.js 本身只是一个精简的类库,提供了几个特性或者说是工具,每个话题深入都可以长篇大论。

我这里只关注使用,毕竟轮子造出来还是用的,而不是观赏。

1. JSX 语法

我的理解 jsx 就是 html + 表达式 的混合体

1.1 请用 ( ... ) 把 jsx 代码包含起来

  1. const Element1 = () => (<h2>组件1 - 常量</h2>)

这样写的理由

  • 低版本的兼容性
  • 多行书写不会报错
当然新版里如果你单行可以省略

1.2 必须有个顶级标签

  1. 错误
  2. let Element4 = () => {
  3. return (
  4. <h2>组件4 - es6 箭头函数</h2>
  5. <h2>组件4 - es6 箭头函数</h2>
  6. )
  7. }
  8.  
  9. 正确
  10. let Element4 = () => {
  11. return (
  12. <div>
  13. <h2>组件4 - es6 箭头函数</h2>
  14. <h2>组件4 - es6 箭头函数</h2>
  15. </div>
  16. )
  17. }
如果你只有一个标签,自己本身就是顶级标签,无需加

1.3 { ... } 开始你的js表达式

  1. function ElementShow(props) {
  2. return (
  3. <div>
  4. <p>字符串: {props.name} </p>
  5. <p>日期变量: {props.date.toLocaleTimeString()}</p>
  6. </div>
  7. )
  8. }

分别打印 属性值、时间函数

1.4 对于没有子元素的标签来说,请关闭标签

  1. <div>
  2. <ElementProps />
  3. </div>

结束符 /

@H_301_101@codepen

https://codepen.io/ducafecat/...

2. 属性 Props

2.1 属性值 都是只读的

  1. function ElementShow(props) {
  2. props.isShow = true // 只读不能修改
  3. ...

打印截图

2.2 特殊的几个 属性

记忆就行,和我们之前写 html 有些差异

我看着都是因为 js 中保留字关系

jsx html
tabIndex index
className class
htmlFor for

示例

  1. const ElementProps = () => (
  2. <div tabIndex="0" className="divbg" >
  3. JSX 属性 tabIndexclassName
  4. </div>
  5. )

2.3 默认使用驼峰格式 camelCase

  1. 错误
  2. <Foo
  3. UserName="hello"
  4. phone_number={12345678}
  5. />
  6.  
  7. 正确
  8. <Foo
  9. userName="hello"
  10. phoneNumber={12345678}
  11. />

2.4 如果属性值为 true,可以直接省略

  1. 错误
  2. <Foo
  3. hidden={true}
  4. />
  5.  
  6. 正确
  7. <Foo
  8. hidden
  9. />

2.5 key 属性是怎么回事

示例

  1. <ul>
  2. {NavRoutes.map((route,index) => (
  3. <li key={route.id}>
  4. {route.title}
  5. </li>
  6. ))}
  7. </ul>

如果不写呢

看来是绕不过的

总结

react利用key来识别组件,它是一种身份标识标识
同层的唯一就行,不用全局唯一
避免使用数组的index作为 key

2.6 防注入攻击

代码

  1. const jsContent = `
  2. <script type="text/javascript">
  3. alert("JSX 防注入攻击!")
  4. </script>`
  5. const ElementInject = () => <div>{jsContent}</div>

打印

内容在渲染之前都被转换成了字符串,这样可以有效地防止 XSS(跨站脚本) 攻击

2.7 childen 表示子节点对象

这个属性表示,当前组件嵌套的对象集合

  1. render() {
  2. return(
  3. <RadioGroup>
  4. <RadioButton value="first">First</RadioButton>
  5. <RadioButton value="second">Second</RadioButton>
  6. <RadioButton value="third">Third</RadioButton>
  7. </RadioGroup>
  8. )
  9. }
RadioGroupprops.childen 就是这三个 RadioButton

你可能会问这有什么用,我们可以用来加工呀,各种循环、克隆、修改,当然我是不太推荐这样去修改 dom 对象

@H_301_101@codepen

https://codepen.io/ducafecat/...

3. 状态 State

组件内部的数据管理对象,自动状态控制

有的同学可能对 MVVM 比较了解,一定会说 React 怎么没有双向绑定

这可能就是设计思想问题了,不想给工具赋予过多负重,轻巧才灵活,下一章,我会通过一个函数解决双向绑定来处理表单操作,就几行代码

这里我们还是谈谈基础操作,懂得同学可以 PASS

3.1 初始化

  1. class ElementStateStatic extends Component {
  2. constructor(props) {
  3. super(props)
  4. this.state = {date: new Date()}
  5. }
  6. render() {
  7. return <p>初始时间 => {this.state.date.toLocaleString()}</p>
  8. }
  9. }
constructor 是组件构造函数,给 this.state 初始值时 要用 key/val 形式的对象
jsx 中使用时直接 this.state.data 对象调用即可

3.2 更新

  1. class ElementStateUpdate extends Component {
  2. constructor(props) {
  3. super(props)
  4. this.date = props.date
  5. this.state = {date: new Date()}
  6. }
  7. componentDidMount() {
  8. if (this.date !== undefined) {
  9. // 传值方式
  10. this.setState({date: this.date})
  11. // 函数方式
  12. // this.setState((state,props) => {
  13. // return {date: this.date}
  14. // })
  15. }
  16. }
  17. render() {
  18. return <p>更新时间 => {this.state.date.toLocaleString()}</p>
  19. }
  20. }
需要使用 this.setState 函数设置

简单操作,直接 传入 key / value 的对象

  • 如果需要之前的 state 或者组件属性 props ,需要写成
  1. this.setState((state,props) => {
  2. let date = state.data
  3. date = date.addDay(10)
  4. return {date}
  5. })

@H_301_101@codepen

https://codepen.io/ducafecat/...

4. 生命周期 Lifecycle

4.1 组件有三种状态

状态 说明
Mount 已插入真实 DOM
Update 正在被重新渲染
Unmount 已移出真实 DOM

4.2 组件周期函数

状态 说明
componentDidMount 在第一次渲染后调用,只在客户端。
shouldComponentUpdate 返回一个布尔值。在组件接收到新的props或者state时被调用
componentDidUpdate 在组件完成更新后立即调用。在初始化时不会被调用
componentWillUnmount 在组件从 DOM 中移除的时候立刻被调用
getDerivedStateFromProps 组件实例化后和接受新属性时将会调用 @H_301_101@新增
getSnapshotBeforeUpdate 在最新的渲染输出提交给DOM前将会立即调用 @H_301_101@新增

4.3 示例打印周期过程

代码

  1. class ElementLifecycle extends Component {
  2. constructor(props) {
  3. super(props)
  4. this.date = props.date
  5. this.state = {date: this.date}
  6. }
  7. componentDidMount() {
  8. console.log('componentDidMount 在第一次渲染后调用')
  9. if (this.date !== undefined) {
  10. this.setState({date: this.date})
  11. }
  12. }
  13. shouldComponentUpdate(nextProps,nextState) {
  14. console.log(
  15. 'shouldComponentUpdate 在组件接收到新的props或者state时被调用',nextProps,nextState
  16. )
  17. return true // 返回一个布尔值,大家可以试着在这里返回 false
  18. }
  19. componentDidUpdate(prevProps,prevState) {
  20. console.log(
  21. 'componentDidUpdate 在组件完成更新后立即调用',prevProps,prevState
  22. )
  23. }
  24. componentWillUnmount() {
  25. console.log('componentWillUnmount 在组件从 DOM 中移除的时候立刻被调用')
  26. }
  27. render() {
  28. return <p>时间 => {this.state.date.toLocaleString()}</p>
  29. }
  30. }

打印截图

@H_301_101@codepen

https://codepen.io/ducafecat/...

4.4 react 16.x 新版本变动

@H_712_403@4.4.1 计划取消 3 个生命周期函数@H_648_404@

分别是 componentWillMount,componentWillReceiveProps,componentWillUpdate

理由是在新的升级中,存在漏洞(在Facebook上,他们维护了超过50,000个React组件。 )

注意:
弃用警告将在未来的16.x版本中启用,但旧版生命周期将继续运行至17.x版。
即使在17.x版中,仍然可以使用它们,但它们会以『UNSAFE_』为前缀被重命名,以表明它们可能会引起问题。我们还准备了一个自动化的脚本来在现有代码中对它们重新命名。
UNSAFE_componentWillMount() UNSAFE_componentWillReceiveProps() UNSAFE_componentWillUpdate()
@H_712_403@4.4.2 新增 getDerivedStateFromProps @H_648_404@
组件实例化后和接受新属性时将会调用

代码

  1. // 组件
  2. class ElementLifecycleNew extends Component {
  3. constructor(props) {
  4. super(props)
  5. this.state = {}
  6. }
  7. static getDerivedStateFromProps(nextProps,prevState) {
  8. console.log(
  9. 'getDerivedStateFromProps 组件实例化后和接受新属性时将会调用',prevState
  10. )
  11. // return null // 无需改变 返回 null
  12. return {
  13. date: new Date('2011-11-11 11:11:11')
  14. }
  15. }
  16. render() {
  17. return <p>{this.state.date.toLocaleString()}</p>
  18. }
  19. }
  20.  
  21. // 调用
  22. <ElementLifecycleNew date={new Date('2009-09-09 09:09:09')} />

如果你不想改变状态 state,返回 null

@H_712_403@4.4.3 新增 getSnapshotBeforeUpdate + componentDidUpdate @H_648_404@
getSnapshotBeforeUpdate() 在最新的渲染输出提交给DOM前将会立即调用。它让你的组件能在当前的值可能要改变前获得它们。这一生命周期返回的任何值将会 作为参数被传递给 componentDidUpdate()
  1. // 代码
  2. class ElementLifecycleNew2 extends Component {
  3. listRef = React.createRef()
  4. constructor(props) {
  5. super(props)
  6. this.state = {
  7. date: props.date
  8. }
  9. }
  10. componentDidMount() {
  11. console.log('componentDidMount')
  12. this.setState({date: new Date('2011-11-22 22:22:22')})
  13. }
  14. getSnapshotBeforeUpdate(prevProps,prevState) {
  15. console.log('getSnapshotBeforeUpdate',prevState,this.state)
  16. return {
  17. offset: 80
  18. }
  19. }
  20. componentDidUpdate(prevProps,snapshot) {
  21. console.log('componentDidUpdate',snapshot)
  22. this.listRef.current.style.top = `${snapshot.offset}px`
  23. }
  24. render() {
  25. return (
  26. <div
  27. style={{
  28. height: 200,width: 150,backgroundColor: 'blue',position: 'relative',color: '#fff'
  29. }}
  30. >
  31. <p>{this.state.date.toLocaleString()}</p>
  32. <div
  33. ref={this.listRef}
  34. style={{
  35. height: 20,backgroundColor: 'red',top: 0,position: 'absolute'
  36. }}
  37. />
  38. </div>
  39. )
  40. }
  41. }
  42.  
  43. // 调用

这个例子的流程是:

  1. 1. `componentDidMount` 修改 state 触发 `getSnapshotBeforeUpdate`
  2. 2. `getSnapshotBeforeUpdate` 获取修改前的 属性、状态,已修改 状态,然后一个修改 `offset`
  3. 3. `componentDidUpdate` 中的 `snapshot` 获取修改 ,直接 `ref` `dom` 修改 `style`
简单说就是不修改 state 更新机制,来维护 dom,比如改改 宽 高 位置

5. 事件 Event

在 react 里使用事件,写法很多,这里采用官方推荐的方式

5.1 无参数传递

  • 采用非箭头函数定义事件
  1. handleChange(e) {
  2. ...
  3. }
  • 在构造函数时绑定事件
  1. class InputView extends Component {
  2. constructor(props) {
  3. ...
  4. this.handleChange = this.handleChange.bind(this)
  5. }
使用 bind(this) 方式
  • 绑定事件
  1. <input
  2. ...
  3. onChange={this.handleChange}
  4. />

5.2 有参数传递

  • 事件参数要放在最后
  1. handleChangeVal(val,e) {
  2. console.log(val)
  3. this.setState({value: e.target.value})
  4. }
  • 绑定事件
  1. <input
  2. ...
  3. onChange={this.handleChangeVal.bind(this,'123')}
  4. />

5.3 例子

  1. class InputView extends Component {
  2. constructor(props) {
  3. super(props)
  4. this.state = {value: ''}
  5. this.handleChange = this.handleChange.bind(this)
  6. this.handleSubmit = this.handleSubmit.bind(this)
  7. }
  8. handleChange(e) {
  9. this.setState({value: e.target.value})
  10. }
  11. handleChangeVal(val,e) {
  12. console.log(val)
  13. this.setState({value: e.target.value})
  14. }
  15. handleSubmit(e) {
  16. e.preventDefault() // 阻止事件
  17. console.log('handleSubmit')
  18. }
  19. render() {
  20. return (
  21. <form onSubmit={this.handleSubmit} style={{display: 'inline-flex'}}>
  22. <input
  23. type="text"
  24. value={this.state.value}
  25. onChange={this.handleChange}
  26. />
  27. <input
  28. type="text"
  29. value={this.state.value}
  30. onChange={this.handleChangeVal.bind(this,'123')}
  31. />
  32. <input type="submit" value="提交" />
  33. <p>{this.state.value}</p>
  34. </form>
  35. )
  36. }
  37. }

@H_301_101@codepen

https://codepen.io/ducafecat/...

6. 样式 Style

有两种方式维护样式

6.1 import 样式文件

  1. .bg {
  2. background-color: rgb(101,40,241);
  3. color: white;
  4. font-weight: 500;
  5. }
  • .js 中引入
  1. import './base.css'

6.2 直接编写 style

  • 编写 style 对象
  1. const styles = {}
  2.  
  3. styles.fill = {
  4. position: 'relative',height: '200px',width: '500px'
  5. }
  6.  
  7. ...
  • 传值使用
  1. <div style={styles.fill}>...
  • 混合使用
  1. <div
  2. style={{
  3. ...styles.fill,...styles.hsl,background: `hsl(${params.h},${params.s}%,${params.l}%)`
  4. }}
  5. >

可以发现这样的写法,对编程控制样式还是很有用的

本文代码

参考


© 会煮咖啡的猫咪
微信 ducafecat
@L_404_24@ Git

猜你在找的React相关文章