react面试题总结(二)

1、如何在 ReactJS 的 Props上应用验证?

React组件的props可以通过多种方式进行验证。以下是几种常用的验证方法:

  1. PropTypes:
    React内置了PropTypes库,可以用来验证props。通过设置propTypes属性,可以指定props的数据类型和必需性。例如:
import PropTypes from 'prop-types';

class MyComponent extends React.Component {
  render() {
    return <div>{this.props.name}</div>;
  }
}

MyComponent.propTypes = {
  name: PropTypes.string.isRequired
};

上面的代码指定了name props为必须的字符串类型。

  1. 使用第三方库:
    除了PropTypes,还有一些第三方库可以用来验证props,例如Joi和Yup。这些库提供了一些更高级的验证功能,例如对复杂数据结构的验证和自定义验证规则。

  2. 手动验证:
    在组件内部手动验证props也是一种有效的方法。可以在组件的constructor方法中使用console.warn()输出警告信息,或者抛出一个错误来提示开发者。例如:

class MyComponent extends React.Component {
  constructor(props) {
    super(props);
  
    if (typeof props.name !== 'string') {
      console.error('Invalid prop `name` supplied to `MyComponent`. Expected `string`');
    }
  }
  
  render() {
    return <div>{this.props.name}</div>;
  }
}

上面的代码手动验证了name props是否为字符串类型。如果props的类型不正确,就会在控制台中输出错误信息。

2、当调用setState时,React render 是如何工作的?

  1. 虚拟 DOM 渲染:当render方法被调用时,它返回一个新的组件的虚拟 DOM 结构。当调用setState()时,render会被再次调用,因为默认情况下shouldComponentUpdate总是返回true,所以默认情况下 React 是没有优化的。
  2. 原生 DOM 渲染:React 只会在虚拟DOM中修改真实DOM节点,而且修改的次数非常少——这是很棒的React特性,它优化了真实DOM的变化,使React更新变得更快。

3、React中setState的第二个参数作用是什么?

在 React 中,setState() 方法的第二个参数是一个可选的回调函数,其作用是在 setState 方法完成并且组件重新渲染后调用该函数。

具体来说,当调用 setState() 更新组件状态后,React 会重新渲染组件,并在此过程中更新状态。当状态更新完成并且重渲染后,React 会调用该回调函数。因此,当需要在状态更新后执行一些逻辑时,可以使用该回调函数。

4、解释 React 中 render() 的目的。

在 React 中,render() 方法是类组件中唯一必须实现的方法。它的主要目的是根据组件的状态和属性来构建虚拟 DOM。虚拟 DOM 是一个 React 抽象层中的 JavaScript 对象树,用于描述 UI 的结构和状态。

当组件的状态或属性发生变化时,React 会自动调用 render() 方法来更新虚拟 DOM,并比较新旧虚拟 DOM 的差异,从而只对需要更新的部分进行重新渲染,从而提高了性能。

render() 方法必须返回一个 React 元素,这可以是原生的 HTML 元素、React 组件实例或其他自定义的对象。React 会将这些元素转换成真实的 DOM 节点,并将它们注入到真实的 DOM 树中。

所以,render() 方法的主要目的是基于组件的当前状态和属性来构建虚拟 DOM,并返回一个描述组件在特定时刻应该呈现什么样子的 React 元素。

5、React的状态提升是什么?

React的状态提升就是用户对子组件操作,子组件不改变自己的状态,通过自己的props把这个操作改变的数据传递给父组件,改变父组件的状态,从而改变受父组件控制的所有子组件的状态,这也是React单项数据流的特性决定的。

6、react 强制刷新

在 React 中,我们可以使用 forceUpdate() 方法来强制组件重新渲染。它的作用是在不更改组件状态或 props 的情况下刷新组件,使其重新执行 render() 方法。

一般来说,强制更新不是 React 的首选方法,因为它会跳过 shouldComponentUpdate 生命周期方法,从而可能导致性能问题。所以我们只应在特殊情况下使用它,例如当我们需要手动更新组件内部的某些信息时,或者组件的状态和 props 没有变化,但是需要强制重新渲染时。

在组件中调用 forceUpdate() 方法比较简单,只需要通过 this.forceUpdate() 来调用即可。但是,它不接受任何参数,因此我们无法直接将新的状态或 props 传递给它。如果需要刷新的数据是从外部传入的,我们还是需要将其存储在组件的状态中,并在 render() 方法中使用它来更新视图。

7、react中的事件this怎么修改

  1. bind()函数
  2. 创建箭头函数:在创建函数的时候创建箭头函数
  3. 在constructor中提前对事件进行绑定
  4. 将事件调用的写法改为箭头函数的形式

8、路由模式是什么?什么是重定向路由?

React 路由模式是指通过 React Router 包提供的组件和方法来管理前端路由,以便实现单页应用(SPA)的页面切换效果。React Router 包含了多种路由模式,包括 BrowerRouter、HashRouter、MemoryRouter 等,常用的是 BrowserRouter 和 HashRouter。

BrowserRouter 使用 HTML5 提供的 History API 来实现路由切换效果,而 HashRouter 使用 URL 中的 hash 值来实现路由切换效果。

重定向路由是指将一个路由重定向到另一个路由上,常用于实现页面访问权限控制或用户访问不存在的页面时的跳转。在 React Router 中,可以使用 <Redirect> 组件实现路由重定向,也可以在路由配置项中使用 Redirect 属性来进行重定向。

9、怎么实现路由懒加载

React 路由懒加载可以通过使用 React.lazy() 和 Suspense 组件来实现。React.lazy() 函数可以让你定义一个动态加载的组件,即当组件被渲染到页面上时才会被加载,而不是在页面一开始就加载所有组件。这样可以减少应用程序的初始加载时间,提高应用程序的性能。

以下是实现路由懒加载的代码示例:

import React, { lazy, Suspense } from 'react';
import { BrowserRouter as Router, Switch, Route, Redirect } from 'react-router-dom';

const Home = lazy(() => import('./Home'));
const About = lazy(() => import('./About'));
const NotFound = lazy(() => import('./NotFound'));

function App() {
  return (
    <Router>
      <Suspense fallback={<div>Loading...</div>}>
        <Switch>
          <Route exact path="/">
            <Home />
          </Route>
          <Route path="/about">
            <About />
          </Route>
          <Route path="/404">
            <NotFound />
          </Route>
          <Redirect to="/404"> {/* 重定向到 404 页面 */}
        </Switch>
      </Suspense>
    </Router>
  );
}

在这个例子中,我们使用了 React.lazy() 函数加载了 Home、About 和 NotFound 组件。Suspense 组件作为包裹路由组件的容器,当需要加载的组件还未加载完成时,会显示 fallback 属性指定的加载提示信息。

需要注意的是,React.lazy() 函数只支持默认导出的组件,因此需要在被导入的组件模块中使用 export default 关键字导出组件。

10、正向传值方式

React 中的正向传值是指从父组件向子组件传递数据或函数。实现正向传值有以下几种方式:

  1. props 属性传递:通过在父组件中给子组件定义属性并传递值的方式,让子组件可以访问传递的值。
// ParentComponent.jsx
import React from 'react';
import ChildComponent from './ChildComponent';

function ParentComponent() {
  const message = 'Hello, world!';
  return (
    <div>
      <ChildComponent message={message} />
    </div>
  );
}

// ChildComponent.jsx
import React from 'react';

function ChildComponent(props) {
  return (
    <div>
      <h1>{props.message}</h1>
    </div>
  );
}
  1. 使用 context:context 提供了一种在组件之间共享值的方式,可以避免通过一层层传递 props 的方式来传递数据。
// MyContext.jsx
import React from 'react';

const MyContext = React.createContext();

export default MyContext;

// ParentComponent.jsx
import React from 'react';
import ChildComponent from './ChildComponent';
import MyContext from './MyContext';

function ParentComponent() {
  const message = 'Hello, world!';
  return (
    <div>
      <MyContext.Provider value={message}>
        <ChildComponent />
      </MyContext.Provider>
    </div>
  );
}

// ChildComponent.jsx
import React from 'react';
import MyContext from './MyContext';

function ChildComponent() {
  return (
    <div>
      <MyContext.Consumer>
        {(message) => <h1>{message}</h1>}
      </MyContext.Consumer>
    </div>
  );
}
  1. 使用回调函数:可以在父组件中定义一个函数,然后将该函数作为 props 属性传递给子组件,在子组件中调用该函数来更新父组件中的状态。
// ParentComponent.jsx
import React, { useState } from 'react';
import ChildComponent from './ChildComponent';

function ParentComponent() {
  const [message, setMessage] = useState('Hello, world!');
  const updateMessage = (newMessage) => {
    setMessage(newMessage);
  };
  return (
    <div>
      <ChildComponent updateMessage={updateMessage} />
      <p>{message}</p>
    </div>
  );
}

// ChildComponent.jsx
import React from 'react';

function ChildComponent(props) {
  const handleClick = () => {
    props.updateMessage('Goodbye, world!');
  };
  return (
    <div>
      <button onClick={handleClick}>Update message</button>
    </div>
  );
}

以上是 React 中实现正向传值的几种方式,根据实际情况选择最适合的方式进行传值即可。

11、逆向传值方式

React 中的逆向传值是指从子组件向父组件传递数据或函数。实现逆向传值有以下两种方式:

  1. 通过 props 传递回调函数:在子组件中调用父组件中传递过来的回调函数,并且将需要传递的值作为参数传递给该函数。
// ParentComponent.jsx
import React, { useState } from 'react';
import ChildComponent from './ChildComponent';

function ParentComponent() {
  const [message, setMessage] = useState('Hello, world!');
  const handleMessageChange = (newMessage) => {
    setMessage(newMessage);
  };
  return (
    <div>
      <ChildComponent handleMessageChange={handleMessageChange} />
      <p>{message}</p>
    </div>
  );
}

// ChildComponent.jsx
import React from 'react';

function ChildComponent(props) {
  const handleChange = (event) => {
    props.handleMessageChange(event.target.value);
  };
  return (
    <div>
      <input type="text" onChange={handleChange} />
    </div>
  );
}
  1. 使用 context:类似于实现正向传值时使用 context 的方式,可以在子组件中使用 context 来向所有祖先组件传递数据或函数。
// MyContext.jsx
import React from 'react';

const MyContext = React.createContext();

export default MyContext;

// ParentComponent.jsx
import React, { useState } from 'react';
import ChildComponent from './ChildComponent';
import MyContext from './MyContext';

function ParentComponent() {
  const [message, setMessage] = useState('Hello, world!');
  const handleMessageChange = (newMessage) => {
    setMessage(newMessage);
  };
  return (
    <div>
      <MyContext.Provider value={handleMessageChange}>
        <ChildComponent />
      </MyContext.Provider>
      <p>{message}</p>
    </div>
  );
}

// ChildComponent.jsx
import React, { useContext } from 'react';
import MyContext from './MyContext';

function ChildComponent() {
  const handleMessageChange = useContext(MyContext);
  const handleChange = (event) => {
    handleMessageChange(event.target.value);
  };
  return (
    <div>
      <input type="text" onChange={handleChange} />
    </div>
  );
}

以上是 React 中实现逆向传值的两种常用方式,可以根据实际应用场景选择最适合的方式来传递数据或函数。React 中的逆向传值是指从子组件向父组件传递数据或函数。实现逆向传值有以下两种方式:

  1. 通过 props 传递回调函数:在子组件中调用父组件中传递过来的回调函数,并且将需要传递的值作为参数传递给该函数。
// ParentComponent.jsx
import React, { useState } from 'react';
import ChildComponent from './ChildComponent';

function ParentComponent() {
  const [message, setMessage] = useState('Hello, world!');
  const handleMessageChange = (newMessage) => {
    setMessage(newMessage);
  };
  return (
    <div>
      <ChildComponent handleMessageChange={handleMessageChange} />
      <p>{message}</p>
    </div>
  );
}

// ChildComponent.jsx
import React from 'react';

function ChildComponent(props) {
  const handleChange = (event) => {
    props.handleMessageChange(event.target.value);
  };
  return (
    <div>
      <input type="text" onChange={handleChange} />
    </div>
  );
}
  1. 使用 context:类似于实现正向传值时使用 context 的方式,可以在子组件中使用 context 来向所有祖先组件传递数据或函数。
// MyContext.jsx
import React from 'react';

const MyContext = React.createContext();

export default MyContext;

// ParentComponent.jsx
import React, { useState } from 'react';
import ChildComponent from './ChildComponent';
import MyContext from './MyContext';

function ParentComponent() {
  const [message, setMessage] = useState('Hello, world!');
  const handleMessageChange = (newMessage) => {
    setMessage(newMessage);
  };
  return (
    <div>
      <MyContext.Provider value={handleMessageChange}>
        <ChildComponent />
      </MyContext.Provider>
      <p>{message}</p>
    </div>
  );
}

// ChildComponent.jsx
import React, { useContext } from 'react';
import MyContext from './MyContext';

function ChildComponent() {
  const handleMessageChange = useContext(MyContext);
  const handleChange = (event) => {
    handleMessageChange(event.target.value);
  };
  return (
    <div>
      <input type="text" onChange={handleChange} />
    </div>
  );
}

以上是 React 中实现逆向传值的两种常用方式,可以根据实际应用场景选择最适合的方式来传递数据或函数。

12、同胞传值方式

React 同胞传值指的是在同一级别的组件之间传递数据,常见的方式有以下三种:

  1. 通过 props 传递:如果同胞组件之间有共同的父组件,可以通过在父组件中将数据传递给两个同胞组件的方式实现数据的传递。

  2. 使用 Context:Context 是 React 提供的一种组件间通信的方式,可以通过在父组件中创建 Context,然后将数据传递给同胞组件。

  3. 使用事件总线:通过在同胞组件中监听事件,然后在发生事件时传递需要传递的数据,从而实现同胞组件之间的数据传递,这种方式需要借助第三方库如 EventBus 或者自己实现事件中心机制。

总之,React 同胞组件之间传递数据的方式比较灵活,可以根据实际需求选择合适的方式。

13、跨组件传值方式

React 跨组件传值方式常见的有以下几种:

  1. props:通过在父组件中向子组件传递 props,实现数据的跨组件传递。这种方式的优点是简单易用,适用于数据传递量小的情况。

  2. context:React 中提供了 context 对象,可以在组件树中实现数据的跨组件传递。这种方式的优点是可以实现跨层级的数据传递,但是需要注意该方式可能会降低应用程序的可维护性。

  3. Redux:Redux 是一种状态管理库,可以实现全局状态的管理和跨组件数据的传递。通过将数据存储在 Redux 中,可以方便地共享和更新数据,同时也提高了应用程序的可维护性。但是,使用 Redux 需要引入额外的依赖和代码量,对于小型应用程序可能会显得过于繁琐。

  4. 其他状态管理库:除了 Redux,还有其他一些状态管理库,例如 Mobx、VueX 等,都可以实现类似的功能。

总之,React 跨组件传值方式有多种,可以根据具体的业务需求和应用场景选择合适的方式。

14、你对“单一数据源”有什么理解?

“单一数据源”是指在应用程序中,所有的数据都来自于一个唯一的数据源,并且该数据源是应用程序的“唯一可信来源”。这个数据源可以是一个对象、一个数组、一个类似 Redux 的全局状态管理器、一个数据库,或者一个 API。

在前端开发中,使用“单一数据源”的好处主要体现在以下几个方面:

  1. 统一的数据管理:由于整个应用程序的数据都来自于同一个数据源,因此可以更方便地对数据进行统一的管理和维护,避免不同部分之间的数据传递和变更带来的复杂性和不一致性。

  2. 更容易实现状态共享:数据源中的数据可以被多个组件共享,因此可以更容易地实现状态共享,避免了多个组件之间的数据传递和同步问题。

  3. 便于测试和调试:由于数据源是应用程序的唯一可信来源,因此可以更方便地进行测试和调试,避免不同部分之间的数据传递和变更带来的复杂性和不一致性。

总之,“单一数据源”是一种管理和维护应用程序数据的有效方式,能够提高应用程序的可维护性、可扩展性和可复用性,也符合现代前端开发中的“数据驱动”的思想。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值