本文详细介绍了React的多个核心概念和高级特性,包括基础知识、组件通信、状态管理以及性能优化等。此外,文章还提供了针对React大厂面试真题的解析和实战演练,帮助读者全面准备面试。通过阅读本文,你可以深入了解React的特性和最佳实践,提升自己的技术能力。
React 基础知识回顾 React 基本概念React 是一个用于构建用户界面的开源库,最初由Facebook开发,并于2011年开源。React的核心特性是通过JSX语法来描述UI,使用虚拟DOM来提升性能,并依赖组件化思想来构建可重用的UI部件。React让开发者可以高效地构建大型的Web应用程序,同时保持代码的可维护性。
JSX 语法
JSX 是一个语法扩展,它允许在JavaScript代码中编写类似HTML的标记。JSX代码会被编译成React.createElement()调用。例如,下面的JSX代码:
const element = <h1>Hello, world!</h1>;
会被编译成:
const element = React.createElement('h1', null, 'Hello, world!');
React 组件
在React中,组件是构建界面的基本单元。组件可以看作是一个函数,它接收输入(属性props)并返回一个描述界面的JSX结构。组件可以是类组件(基于React.Component的类)或函数组件(纯JavaScript函数)。类组件和函数组件的主要区别在于,类组件可以利用生命周期方法,而函数组件则不能。
JSX 的理解和使用JSX使得React代码更加接近于HTML,这使得开发者可以更快速地理解和编写React代码。JSX允许在JSX代码中嵌入JavaScript表达式,这使得编写动态的UI成为可能。
例如,下面的代码:
const name = 'World';
const element = <h1>Hello, {name}</h1>;
会被编译成:
const element = React.createElement(
'h1',
null,
'Hello, ',
name
);
注意,JSX代码中的表达式需要用大括号 {} 包裹。
示例
下面是一个简单的React组件例子,它使用了JSX来渲染一个带有文本的<h1>
标签:
function Welcome(props) {
return <h1>Hello, {props.name}</h1>;
}
// 使用组件
const element = <Welcome name="World" />;
React 组件的生命周期
React组件的生命周期可以分为三个阶段:挂载(Mounting)、更新(Updating)、卸载(Unmounting)。每个阶段都有对应的生命周期方法。
-
挂载阶段:组件从创建到首次渲染到DOM中的过程。
constructor(props)
:在组件实例化时调用,通常在这个生命周期方法中初始化状态。static getDerivedStateFromProps(props, state)
:在组件挂载或更新时调用,接收props和state作为参数。render()
:返回描述组件的JSX结构。
.componentDidMount()
:在组件挂载完成后调用,可以在这里执行AJAX请求或者订阅事件。
-
更新阶段:组件接收到新的props或状态更新时调用。
static getDerivedStateFromProps(props, state)
:在组件更新时调用。shouldComponentUpdate(nextProps, nextState)
:决定组件是否应该更新,这个方法返回一个布尔值。render()
:在shouldComponentUpdate返回true时调用。getSnapshotBeforeUpdate(prevProps, prevState)
:在DOM更新之前调用,返回值可以作为componentDidUpdate
的参数。componentDidUpdate(prevProps, prevState, snapshot)
:在组件更新完成后调用。
- 卸载阶段:从DOM中移除组件。
componentWillUnmount()
:在组件卸载前调用。
示例
下面是一个简单的类组件,它监听了窗口大小的变化,并在组件卸载时移除监听。
class MyComponent extends React.Component {
componentDidMount() {
window.addEventListener('resize', this.handleResize);
}
componentWillUnmount() {
window.removeEventListener('resize', this.handleResize);
}
handleResize = () => {
console.log('Window resized');
}
render() {
return <div>Resize the window</div>;
}
}
面向对象的高级特性
React 高阶组件
高阶组件(Higher-Order Component, HOC)是一种高级技巧,用于复用组件逻辑。高阶组件通常是一个函数,它接收一个组件作为参数,并返回一个新的增强组件。这种模式本质上是函数式编程的一种应用,它允许组件之间进行解耦。
示例
下面是一个高阶组件的例子,它添加了一个loading状态到组件中。
function withLoading(WrappedComponent) {
return class extends React.Component {
state = { loading: true };
componentDidMount() {
// 假设这里有异步操作
setTimeout(() => {
this.setState({ loading: false });
}, 2000);
}
render() {
if (this.state.loading) {
return <div>Loading...</div>;
}
return <WrappedComponent {...this.props} />;
}
};
}
const EnhancedComponent = withLoading(MyComponent);
React Hooks 的基本使用
React Hooks 是React 16.8版本引入的新特性,它允许在函数组件中使用状态或其他React特性,而不需要编写类组件。常用的Hooks包括useState、useEffect、useContext、useReducer等。
useState
useState
是一个Hook,它允许你在函数组件中添加状态。useState
返回一个状态数组,数组的第一个元素是当前状态,第二个元素是一个更新状态的函数。
import React, { useState } from 'react';
function Counter() {
const [count, setCount] = useState(0);
return (
<div>
<p>Count: {count}</p>
<button onClick={() => setCount(count + 1)}>
Increment count
</button>
</div>
);
}
useEffect
useEffect
用于在函数组件中执行副作用操作,例如数据获取、订阅、手动修改DOM等。useEffect
接收一个回调函数作为参数,这个回调函数会在组件更新后执行。
import React, { useState, useEffect } from 'react';
function Example() {
const [count, setCount] = useState(0);
useEffect(() => {
document.title = `You clicked ${count} times`;
}, [count]);
return (
<div>
<p>You clicked {count} times</p>
<button onClick={() => setCount(count + 1)}>
Click me
</button>
</div>
);
}
React Context 的应用
React Context 提供了一种在组件树中传递数据的方式,无需手动在每一层组件中传递props。Context 在创建时会指定一个默认值,当组件树中的组件需要访问上下文时,可以通过Context.Consumer
或Context.Provider
来访问和使用上下文。
示例
下面是一个更复杂的React Context应用示例,它展示了如何通过Context来管理主题切换。
import React, { Component, createContext } from 'react';
const ThemeContext = createContext();
class App extends React.Component {
render() {
return (
<ThemeContext.Provider value="dark">
<Toolbar />
</ThemeContext.Provider>
);
}
}
class Toolbar extends React.Component {
render() {
return (
<div>
<ThemedButton />
</div>
);
}
}
class ThemedButton extends React.Component {
static contextType = ThemeContext;
render() {
return <button style={{ background: this.context }}>Click me!</button>;
}
}
组件通信与状态管理
父子组件间的通信
父子组件通信是通过props实现的。父组件将数据通过props传递给子组件,子组件通过回调函数将数据传递回父组件。
示例
下面是一个简单的父子组件通信的例子。父组件传递一个回调函数给子组件,子组件在回调函数中调用父组件的方法来更新状态。
class ParentComponent extends React.Component {
constructor(props) {
super(props);
this.state = { message: 'Hello from parent' };
this.updateMessage = this.updateMessage.bind(this);
}
updateMessage(newMessage) {
this.setState({ message: newMessage });
}
render() {
return (
<div>
{this.state.message}
<ChildComponent updateMessage={this.updateMessage} />
</div>
);
}
}
class ChildComponent extends React.Component {
constructor(props) {
super(props);
this.handleClick = this.handleClick.bind(this);
}
handleClick() {
this.props.updateMessage('Hello from child');
}
render() {
return <button onClick={this.handleClick}>Update message</button>;
}
}
非父子组件间的通信
非父子组件间的通信可以通过事件代理、回调函数、上下文、全局状态管理库(如Redux或MobX)等方式实现。
示例
下面是一个使用事件代理实现非父子组件之间通信的例子。通过一个全局事件对象,子组件可以在回调函数中触发事件,父组件监听该事件并执行相应操作。
const globalEventEmitter = require('global-event-emitter');
class ParentComponent extends React.Component {
componentDidMount() {
globalEventEmitter.on('updateMessage', this.updateMessage);
}
componentWillUnmount() {
globalEventEmitter.removeListener('updateMessage', this.updateMessage);
}
updateMessage(data) {
this.setState({ message: data.message });
}
render() {
return (
<div>
{this.state.message}
<ChildComponent />
</div>
);
}
}
class ChildComponent extends React.Component {
handleClick = () => {
globalEventEmitter.emit('updateMessage', { message: 'Hello from child' });
}
render() {
return <button onClick={this.handleClick}>Update message</button>;
}
}
Redux 或 MobX 的使用
Redux 和MobX 是两种用于状态管理的库。Redux通过单一状态树来管理整个应用的状态,而MobX则通过可观察对象和反应式编程来简化状态管理。
Redux 示例
下面是一个简单的Redux示例,它定义了一个Counter Redux Store。
import { createStore, applyMiddleware } from 'redux';
import thunk from 'redux-thunk';
const initialState = { count: 0 };
const reducer = (state = initialState, action) => {
switch (action.type) {
case 'INCREMENT':
return { count: state.count + 1 };
case 'DECREMENT':
return { count: state.count - 1 };
default:
return state;
}
};
const store = createStore(reducer, applyMiddleware(thunk));
store.subscribe(() =>
console.log(store.getState())
);
store.dispatch({ type: 'INCREMENT' });
store.dispatch({ type: 'DECREMENT' });
MobX 示例
下面是一个简单的MobX示例,它定义了一个可观察的Counter类。
import { observable, action } from 'mobx';
class Counter {
@observable count = 0;
@action increment() {
this.count++;
}
@action decrement() {
this.count--;
}
}
const counter = new Counter();
counter.increment();
counter.decrement();
组件性能优化
函数组件与类组件的性能差异
函数组件由于没有生命周期方法,所以编译后的代码更简洁,通常比类组件执行得更快。但是,类组件由于支持生命周期方法,因此在某些场景下可能更高效,因为它允许开发者在合适的时间点执行特定的操作,如在组件更新后执行副作用操作。
shouldComponentUpdate 和 React.memoshouldComponentUpdate
是一个生命周期方法,用于决定组件是否应该重新渲染。默认情况下,React在每次状态更新时都会重新渲染组件,这可能会导致不必要的渲染。通过复写 shouldComponentUpdate
方法,可以更精细地控制组件的更新行为。
React.memo
是一个函数组件的高阶组件,它实现了同样的优化,但它只能用于纯函数组件。
示例
下面是一个使用shouldComponentUpdate
的类组件的例子。
class Counter extends React.Component {
shouldComponentUpdate(nextProps, nextState) {
return nextProps.count !== this.props.count;
}
render() {
return <div>Count: {this.props.count}</div>;
}
}
下面是一个使用React.memo
的函数组件的例子。
const Counter = React.memo(({ count }) => (
<div>Count: {count}</div>
));
使用 React.memo 和 useMemo 进行性能优化
React.memo
和useMemo
用于避免不必要的组件渲染或计算,它们可以帮助优化性能。
React.memo
React.memo
是一个高阶组件,它包裹了一个函数组件,并在组件的输入属性没有变化时避免重新渲染该组件。它类似于类组件中的shouldComponentUpdate
方法。
const Counter = React.memo(({ count }) => (
<div>Count: {count}</div>
));
useMemo
useMemo
是一个Hooks,它返回一个经过缓存的值。useMemo
只有在依赖项改变时才会重新计算值,这可以避免不必要的计算。
import React, { useState, useMemo } from 'react';
const Counter = ({ count }) => {
const expensiveCalculation = useMemo(() => {
return count * count * count;
}, [count]);
return <div>Count cubed: {expensiveCalculation}</div>;
};
实战演练:模拟面试题
常见面试题解析
面试题1:解释React的生命周期方法
React组件的生命周期可以分为几个阶段:挂载、更新、卸载。生命周期方法允许开发者在组件的不同生命周期阶段执行特定的操作。例如,componentDidMount
在组件挂载完成后被调用,通常在这里执行副作用操作,如数据获取或订阅。shouldComponentUpdate
决定组件是否应该重新渲染,这可以用来优化性能。
面试题2:解释高阶组件的用途和实现
高阶组件是一种高级技巧,用于复用组件逻辑。高阶组件通常是一个函数,它接收一个组件作为参数,并返回一个新的增强组件。这种模式可以用来实现代码复用、封装状态、抽象组件逻辑等。
面试题3:解释React Hooks和它们的用途
React Hooks 是React 16.8版本引入的新特性,它允许在函数组件中使用状态或其他React特性,而不需要编写类组件。常用的Hooks包括useState、useEffect、useContext、useReducer等。useState
用于添加状态,useEffect
用于副作用操作,useContext
用于获取上下文,useReducer
用于更复杂的状态逻辑。
情景1:解释useEffect
的用途和实现
useEffect
是一个Hook,用于在函数组件中执行副作用操作,例如数据获取、订阅、手动修改DOM等。它接收一个回调函数作为参数,这个回调函数会在组件挂载或更新后执行。如果回调函数返回一个函数,这个返回的函数会在组件卸载前执行,通常用来清理副作用。
情景2:解释useContext
的用途和实现
useContext
是一个Hook,它允许在函数组件中访问上下文值。它接收一个Context对象作为参数,并返回当前上下文值。通过useContext
,可以在组件树的任意深度访问上下文值,而不需要手动传递props。
- 了解React核心概念:熟悉React的生命周期、组件通信、状态管理等概念。
- 掌握React Hooks:理解Hooks的基本使用和实现,例如
useState
、useEffect
、useContext
等。 - 实战经验:准备一些实际项目经验,包括遇到的问题和解决方案。
- 代码审查:能够审查和理解其他人的代码,包括常见的性能优化技术。
- 沟通能力:能够清晰地表达自己的想法,解决技术问题时逻辑清晰。
- 持续学习:保持对新技术的关注,了解最新的React特性和最佳实践。
- React基础知识:包括React组件、JSX、生命周期等。
- React高级特性:包括高阶组件、Hooks、Context等。
- 状态管理:包括Redux、MobX等。
- 性能优化:包括函数组件、类组件性能差异、
shouldComponentUpdate
、React.memo
、useMemo
等。
面试时,不仅要展示自己的技术能力,还要展示自己的解决问题的能力。在回答技术问题时,不仅要给出正确的答案,还要解释为什么这样做,以及没有其他更好的解决方案。同时,要注意表达清晰、逻辑严谨,避免使用行业黑话或过于复杂的术语。
学习资源推荐