最后
全网独播-价值千万金融项目前端架构实战
从两道网易面试题-分析JavaScript底层机制
RESTful架构在Nodejs下的最佳实践
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
一线互联网企业如何初始化项目-做一个自己的vue-cli
思维无价,看我用Nodejs实现MVC
代码优雅的秘诀-用观察者模式深度解耦模块
前端高级实战,如何封装属于自己的JS库
VUE组件库级组件封装-高复用弹窗组件
import React from ‘react’
export default function Plus() {
const myRef = React.useRef()
function show() {
alert(myRef.current.value)
}
return (
<>
点我展示输入内容
</>
)
}
-
上面这些钩子,都是引入某种特定的副效应,而
useEffect()
是通用的副效应钩子 。找不到对应的钩子时,就可以用它。其实,从名字也可以看出来,它跟副效应(side effect)直接相关。 -
它可以用于模拟类组件中的
生命周期钩子
- 语法
React.useEffect(() => {
1.相当于 conponentDidMount()、conponentDidUpdate()
return ()=>{ 2.相当于conponentWillUnmount() }
},[3.监测哪个状态])
- 案例
-
组件挂载,当前求和之后的数字以每秒加1的速度累加
-
当点击卸载组件按钮时,useEffect会在卸载组件之前清除定时器。
import React from ‘react’
import ReactDOM from “react-dom”
export default function Plus() {
const [count, setCount] = React.useState(0)
React.useEffect(() => {
let timer = setInterval(() => {
setCount(count => count+1)
},1000)
return () => {
clearInterval(timer)
}
},[])
//卸载组件
function unMount() {
ReactDOM.unmountComponentAtNode(document.getElementById(“root”))
}
return (
<>
当前求和为{count}
卸载组件
</>
)
}
- 总结
-
useEffect传入的第一个值:这个函数,有可能相当于
componentDidMount
、componentDidUpdate
、componentWillUnmount
(甚至有可能同时相当于componentDidMount和componentDidUpdate!
) -
useEffect传入的第二个值:这个[ ],方括号里面是啥(state),它就监测啥(state)
-
useEffect里的参数:
①同时相当于componentDidMount/componentDidUpdate
:useEffect只传一个参数useEffect( ()=>{} ) ----------> 它等于谁都监测,组件一挂载和任何状态变动,都会回调那个作为参数的函数
②相当于componentDidMount
:useEffect( ()=>{} , [] ) ----------> 第二个参数,[ ]里什么都不写 --------> 相当于谁都不监测
③相当于componentWillUnmout
: useEffect( ()=>{ return ()=>{} } ) --------> 第一个参数的返回值的这个函数相当于componentWillUnmout
==========================================================================================
用的时候才加载,一般是路由组件进行懒加载
如果不用路由懒加载,页面在第一次进入的时候,就请求了所有组件的数据,如果组件过多,过多的请求这就没有必要了,应该是用户按哪个链接再请求哪个组件
-
通过React的lazy函数配合import()函数动态加载路由组件【路由组件代码会被分开打包】
-
通过指定在加载得到路由打包文件前显示一个自定义loading界面
import React, { Component, lazy, Suspense} from ‘react’
import {NavLink,Route} from ‘react-router-dom’
// import Home from ‘./Home’
// import About from ‘./About’
import Loading from ‘./Loading’
const Home = lazy(()=> import(‘./Home’) )
const About = lazy(()=> import(‘./About’))
export default class Demo extends Component {
render() {
return (
React Router Demo
{/* 在React中靠路由链接实现切换组件–编写路由链接 */}
About
Home
<Suspense fallback={}>
{/* 注册路由 */}
)
}
}
=============================================================================
文档碎片
可以不用必须有一个真实的DOM根标签了
import React, { Component,Fragment } from ‘react’
export default class Demo extends Component {
render() {
return (
)
}
}
============================================================================
上下文
一种组件间通信方式, 常用于【祖组件】与【后代组件】间通信
- 创建Context容器对象
const XxxContext = React.createContext()
- 渲染子组时,外面包裹xxxContext.Provider, 通过value属性给后代组件传递数据:
<xxxContext.Provider value={数据}>
子组件
</xxxContext.Provider>
- 后代组件读取数据:
第一种方式:仅适用于类组件
static contextType = xxxContext // 声明接收context
this.context // 读取context中的value数据
第二种方式: 函数组件与类组件都可以
<xxxContext.Consumer>
{
value => ( // value就是context中的value数据
要显示的内容
)
}
</xxxContext.Consumer>
在应用开发中一般不用context, 一般都用它来封装react插件
import React, { Component } from ‘react’
import ‘./index.css’
//创建Context对象
const MyContext = React.createContext()
const { Provider, Consumer } = MyContext
export default class A extends Component {
state = {username:‘tom’,age:18}
render() {
const {username,age} = this.state
return (
我是A组件
我的用户名是:{username}
)
}
}
class B extends Component {
render() {
return (
我是B组件
)
}
}
/* class C extends Component {
//声明接收context
static contextType = MyContext
render() {
const {username,age} = this.context
return (
我是C组件
我从A组件接收到的用户名:{username},年龄是{age}
)
}
} */
function C(){
return (
我是C组件
我从A组件接收到的用户名:
{value => ${value.username},年龄是${value.age}
}
)
}
=========================================================================
-
只要执行setState(),即使不改变状态数据, 组件也会重新render() ==> 效率低
-
只当前组件重新render(), 就会自动重新render子组件,纵使子组件没有用到父组件的任何数据 ==> 效率低
只有当组件的state或props数据发生改变时才重新render()
Component中的shouldComponentUpdate()总是返回true(阀门总是打开的)
4.1 办法一:重写shouldComponentUpdate()方法
比较新旧state或props数据, 如果有变化才返回true, 如果没有返回false
shouldComponentUpdate(nextProps,nextState){
// console.log(this.props,this.state); //目前的props和state
// console.log(nextProps,nextState); //接下要变化的目标props,目标state
return !this.state.carName === nextState.carName
}
4.2 办法二: 使用PureComponent
PureComponent重写了shouldComponentUpdate(), 只有state或props数据有变化才返回true
注意
-
只是进行state和props数据的
浅比较
, 如果只是数据对象内部数据变了, 返回false -
不要直接修改state数据, 而是要产生新数据
项目中一般使用PureComponent
来优化
import React, { PureComponent } from ‘react’
import ‘./index.css’
export default class Parent extends PureComponent {
state = {carName:“奔驰c36”,stus:[‘小张’,‘小李’,‘小王’]}
addStu = ()=>{
const {stus} = this.state
this.setState({stus:[‘小刘’,…stus]})
}
changeCar = ()=>{
this.setState({carName:‘迈巴赫’})
render() {
console.log(‘Parent—render’);
const {carName} = this.state
return (
我是Parent组件
{this.state.stus}
我的车名字是:{carName}
点我换车
算法刷题
大厂面试还是很注重算法题的,尤其是字节跳动,算法是问的比较多的,关于算法,推荐《LeetCode》和《算法的乐趣》,这两本我也有电子版,字节跳动、阿里、美团等大厂面试题(含答案+解析)、学习笔记、Xmind思维导图均可以分享给大家学习。
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
写在最后
最后,对所以做Java的朋友提几点建议,也是我的个人心得:
-
疯狂编程
-
学习效果可视化
-
写博客
-
阅读优秀代码
-
心态调整