React

1、什么时候使用状态管理器?
2、render函数中return如果没有使用()会有什么问题?

答案借鉴地址

我们在使用JSX语法书写react代码时,babel会将JSX语法编译成js,同时会在每行自动添加分号(;),如果return后换行了,那么就会变成 return; 一般情况下会报错:Nothing was returned from render. This usually means a return statement is missing. Or, to render nothing, return null.译文 : 渲染没有返回任何内容。这通常意味着缺少return语句。或者,为了不渲染,返回null

举两个正确的书写例子:

const Nav = () => {
  return (
    <nav className="c_navbar">
      { some jsx magic here }
    </nav>
  )
}

const Nav = () => {
 return <nav className="c_navbar">
    { some jsx magic here }
  </nav>
}

错误的写法:

const Nav = () => {
  return
    <nav className="c_navbar">
      { some jsx magic here }
    </nav>
}
3、componentWillUpdate可以直接修改state的值吗?

答案借鉴地址

首先你能这么做,但是不建议,因为会造成无限渲染

并不是说任何情况下不能在此生命周期方法中使用this.setState(),你可以设置合理的条件,保证它不会在每次渲染的时候执行this.setState(),而只是在需要的情况下执行【这很重要】

不止componentWillUpdate方法,比如componentWillReceiveProps也可以这样设置条件

4、说说你对React的渲染原理的理解

答案借鉴地址

从数据方面 : 单向数据流。React是一个MVVM框架,简单来说是在MVC的模式下在前端部分拆分出数据层和视图层。单向数据流指的是只能由数据层的变化去影响视图层的变化,而不能反过来(除非双向绑定);数据驱动视图。我们无需关注页面的DOM,只需要关注数据即可.

从渲染方面 : 首先编写的jsx会被转换执行返回node对象,node对象就可以理解为是虚拟dom,渲染时都会将老的虚拟dom和新的虚拟dom来进行比对,比对的过程中就涉及到了diff算法,只比较同一层的节点,节点tag不同,就不再比较,以新的节点为准,节点相同就比较key,key不同也会以新节点为准;之后就将最终的虚拟dom,再渲染在浏览器中;

通过props state与一些高阶组件执行更新

5、什么渲染劫持?

答案借鉴地址

组合渲染和条件渲染都是渲染劫持的一种,通过反向继承,不仅可以实现以上两点,还可以增强由原组件render函数产生的React元素。

实际的操作中 通过 操作 state、props 都可以实现渲染劫持

条件渲染

const HOC = (WrappedComponent) =>
  class extends WrappedComponent {
    render() {
      if (this.props.isRender) {
        return super.render();
      } else {
        return <div>暂无数据</div>;
      }
    }
  }

组合渲染

// 例子来源于《深入React技术栈》
function HigherOrderComponent(WrappedComponent) {
  return class extends WrappedComponent {
    render() {
      const tree = super.render();
      const newProps = {};
      if (tree && tree.type === 'input') {
        newProps.value = 'something here';
      }
      const props = {
        ...tree.props,
        ...newProps,
      };
      const newTree = React.cloneElement(tree, props, tree.props.children);
      return newTree;
    }
  };
}

6、React Intl是什么原理?

实现原理和react-redux的实现原理类似,最外层包一个Provider,利用getChildContext,将intlConfigPropTypes存起来,在FormattedMessage、FormattedNumber等组件或者调用injectIntl生成的高阶组件中使用,来完成国际化的。

使用过程也非常简单,和react-redux的使用方法非常类似。

7、说说Context有哪些属性?

答案借鉴地址

context属于一种解决组件间层级过多传递数据的问题,避免了层层嵌套的通过props传递的形式,同时对于不需要使用到redux时,是一种解决方案,关于组件的复用性变差的问题,我觉得是可以通过高阶组件和context配合来解决的,因为react-redux使用的就是这样的形式;

主要的形式:createContext(value):创建一个context实例;其中的参数为当前数据的默认值,只有没在Provider中指定value时,才会生效; Context.Provider:生产者,数据提供方;通过value属性来定义需要被传递的数据 Context.Consumer:消费者,数据获取方;根据是函数组件还是class组件,有不同的使用形式;class组件可以指定contextType来确定要使用哪一个context对象的值,函数组件需要使用回调函数的形式来获取context的值;需要显示的指定context对象;

8、为什么React并不推荐我们优先考虑使用Context?

答案借鉴地址

1、Context目前还处于实验阶段,可能会在后面的发行版本中有很大的变化,事实上这种情况已经发生了,所以为了避免给今后升级带来大的影响和麻烦,不建议在app中使用context。

2、尽管不建议在app中使用context,但是独有组件而言,由于影响范围小于app,如果可以做到高内聚,不破坏组件树之间的依赖关系,可以考虑使用context 3、对于组件之间的数据通信或者状态管理,有效使用props或者state解决,然后再考虑使用第三方的成熟库进行解决,以上的方法都不是最佳的方案的时候,在考虑context。 4、context的更新需要通过setState()触发,但是这并不是很可靠的,Context支持跨组件的访问,但是如果中间的子组件通过一些方法不影响更新,比如 shouldComponentUpdate() 返回false 那么不能保证Context的更新一定可以使用Context的子组件,因此,Context的可靠性需要关注。

9、childContextTypes是什么?它有什么用?

childContextTypes用来定义context数据类型,该api从16.3开始已被废弃

class MessageList extends React.Component {
  getChildContext() {
    return {color: "purple"};
  }

  render() {
    return <div>MessageList</div>;
  }
}

MessageList.childContextTypes = {
  color: PropTypes.string
};

hooks有此用法

import PropTypes from 'prop-types';
const MessageList = ({
    color
}) => {
    MessageList.propTypes={
         color: PropTypes.string
    }
}
10、contextType是什么?它有什么用?

类似于hooks中 useContext用法

11、Consumer向上找不到Provider的时候怎么办?

The defaultValue argument is only used when a component does not have a matching Provider above it in the tree. This can be helpful for testing components in isolation without wrapping them. Note: passing undefined as a Provider value does not cause consuming components to use defaultValue.

官方解释 : 只有在树中组件上方没有匹配的Provider时,才使用defaultValue参数。 这对于隔离测试组件而不进行包装很有帮助。 注意:将undefined传递为Provider值不会导致使用组件使用defaultValue。

12、说说你对windowing的了解

Windowing 是一种技术,它在任何给定时间只呈现一小部分行,并且可以显著减少重新呈现组件所需的时间以及创建的 DOM 节点的数量。如果应用程序呈现长的数据列表,则建议使用此技术。react-window 和 react-virtualized 都是常用的 windowing 库,它提供了几个可重用的组件,用于显示列表、网格和表格数据。

13、React的严格模式有什么用处?

识别不安全的生命周期

关于使用过时字符串 ref API 的警告

关于使用废弃的 findDOMNode 方法的警告

检测意外的副作用

检测过时的 context API

14、React组件的构造函数有什么作用?

1、指定this –> super(props)

2、设置初始化的状态 –> this.setState({});

3、为组件上的构造函数绑定this

15、React中在哪捕获错误

​ 定义错误边界当做常规组件包裹ui组件

官网例子 :

class ErrorBoundary extends React.Component {
  constructor(props) {
    super(props);
    this.state = { hasError: false };
  }

  static getDerivedStateFromError(error) {
    // 更新 state 使下一次渲染能够显示降级后的 UI
    return { hasError: true };
  }

  componentDidCatch(error, errorInfo) {
    // 你同样可以将错误日志上报给服务器
    logErrorToMyService(error, errorInfo);
  }

  render() {
    if (this.state.hasError) {
      // 你可以自定义降级后的 UI 并渲染
      return <h1>Something went wrong.</h1>;
    }

    return this.props.children; 
  }
}

使用

<ErrorBoundary>
  <MyWidget />
</ErrorBoundary>

但是错误边界不会捕获 :

try{}catch(err){}
///异步代码(例如 setTimeout 或 requestAnimationFrame 回调函数)
///服务端渲染
///它自身抛出来的错误(并非它的子组件)
16、React怎样引入svg的文件?

代码式 :

const HouseTabsMenuKeyGreySvg = () => (
    <svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 14 14">
        <g transform="translate(-218 -71)">
            <circle cx="7" cy="7" r="7" transform="translate(218 71)" fill='#f2f2f2'/>
            <path fill='#ddd'
                  d="M231.89,95.73l.111-.11a.319.319,0,0,0,0-.452l-.816-.814a.322.322,0,0,0-.454,0l-1.471,1.468a.889.889,0,0,0,.011,1.26l.01.009.009.011a.893.893,0,0,0,1.259.011l.168-.168.289.288a.322.322,0,1,0,.455-.454l-.289-.289.283-.282.289.288a.322.322,0,1,0,.455-.454Z"
                  transform="translate(-7.501 -15.86)"/>
            <path fill='#ddd'
                  d="M239.288,82.761A2.6,2.6,0,0,0,237.473,82a2.5,2.5,0,0,0-2.536,2.53,2.608,2.608,0,0,0,2.58,2.573,2.5,2.5,0,0,0,2.536-2.53,2.585,2.585,0,0,0-.764-1.811Z"
                  transform="translate(-11.551 -7.5)"/>
            <path fill='#ccc'
                  d="M241.782,87.954a.326.326,0,0,1-.46,0,.3.3,0,1,0-.186.505.331.331,0,0,1,.349.3.318.318,0,0,1-.3.343.964.964,0,0,1-.737-.281.939.939,0,1,1,1.329-1.326.324.324,0,0,1,0,.459Z"
                  transform="translate(-15.114 -11.055)"/>
        </g>
    </svg>
);

文件式 :

<img src='path/logo.svg' className="App-logo" alt="logo" />
17、react中的const

讲讲 es6中的const

const framework = "weex";
framework = "react native";
//TypeError: Assignment to constant variable.

const weex = {"weexpack": "good", "weex-toolkit": "sucks"};
 weex["weex-toolkit"] = "good";
//允许


在 ES6 中,const 只限制对应的变量不能够被再次赋值

但是变量的属性可以被再次赋值

react中声明函数,useState都用到const

18、为什么说React中的props是只读的?

React 组件都必须像纯函数一样保护它们的 props 不被更改。

将react组件理解成纯函数,数据流驱动,参数传入不允许做更改

扩展 :

state内容可以更改,但是不允许直接赋值,需要借助setState

props用于定义外部接口,state用于记录内部状态

props的赋值在于外部世界使用组件,state的赋值在于组件内部

组件不应该改变props的值,而state存在的目的就是让组件来修改的

state 只能在constructor中设置默认值

setState修改state的值是异步的

19、你有用过哪些React的表单库吗?说说它们的优缺点

redux-form

26.4k

不支持第三方表单验证,需手动验证

性能优异,虽然大家都不喜欢它

react-json-schema-form

复杂的表单交互不易用,比如我们这边有些 array 结构的 form 表单,有大量的数据条数(在50条左右),react-json-schema-form 交互使用了每个字段占用一行的设计,也就是说如果有6个字段,共50条数据,需要占用 300行空间;

不支持通过 js 表达式 ;

uiSchema 和 json-schema 分开,导致在遇到深层级节点时不好配,还有节点有改动,需要同步 uiSchema。

formik

15k

支持,可以使用 Yup schema 第三方表单验证

Field 组件在大型的 form 表单中会出现性能问题,可以使用 FastField 组件改善

unform

react-hook-form

8.7k

支持多种第三方验证,像 Yup, Joi, Superstruct

在非受控组件中表现最好但是不会触发 re-render, 但是如果触发了 re-render 性能比不上 redux-formformik

20、如果组件的属性没有传值,那么它的默认值是什么?

true

官方文档写了,别满脑子是undefined

21、可以使用TypeScript写React应用吗?怎么操作?

答案借鉴地址

使用ts启动新的 create react app项目 yarn create react-app my-app --typescript 将ts添加到已经创建好的create react app项目中 yarn add typescript @types/node @types/react @types/react-dom @types/jest PS:适用于react-scripts@2.1.0 及更高版本。

22、super()super(props)有什么区别?

答案借鉴地址

react 中的class 是基于es6的规范实现的, 继承是使用extends关键字实现继承的,子类必须在constructor()中调用super() 方法否则新建实例 就会报错,报错的原因是 子类是没有自己的this对象的,它只能继承父类的this对象,然后对其进行加工,而super()就是将父类中的this对象继承给子类的,没有super() 子类就得不到this对象。

如果你使用了constructor就必须写super() 这个是用来初始化this的,可以绑定事件到this上 如果你想要在constructor中使用this.props,就必须给super添加参数 super(props) 注意,无论有没有 constructor,在render中的this.props都是可以使用的,这是react自动附带的 如果没有用到constructor 是可以不写的,react会默认添加一个空的constroctor.

重点 : 是在constructor中使用this.props

22、你有使用过loadable组件吗?它帮我们解决了什么问题?

react-loadableloadable component,用于代码分割,解决打包体积过大的问题

23、你有使用过suspense组件吗?它帮我们解决了什么问题?

答案借鉴地址

实验加载地址

让你可以“等待”目标代码加载,并且可以直接指定一个加载的界面(像是个 spinner)

const ProfilePage = React.lazy(() => import('./ProfilePage')); // 懒加载

// 在 ProfilePage 组件处于加载阶段时显示一个 spinner
<Suspense fallback={<Spinner />}>
  <ProfilePage />
</Suspense>

以下注意 :

想要用Suspense 成为组件读取异步数据的主要方式有以下问题:

它不是数据获取的一种实现。它并不假定你使用 GraphQL,REST,或者任何其他特定的数据格式、库、数据传输方式、协议。

它不是一个可以直接用于数据获取的客户端。你不能用 Suspense 来“替代” fetch 或者 Relay。不过你可以使用集成 Suspense 的库(比如说,新的 Relay API)。

它不使数据获取与视图层代码耦合。它协助编排加载状态在 UI 中的显示,但它并不将你的网络逻辑捆绑到 React 组件。

Suspense 到底有什么用呢

它能让数据获取库与 React 紧密整合。如果一个数据请求库实现了对 Suspense 的支持,那么,在 React 中使用 Suspense 将会是自然不过的事。

它能让你有针对性地安排加载状态的展示。虽然它不干涉数据怎样获取,但它可以让你对应用的视图加载顺序有更大的控制权。

它能够消除 race conditions。即便是用上 await,异步代码还是很容易出错。相比之下,Suspense 更给人同步读取数据的感觉 —— 假定数据已经加载完毕。

扩展 : react-cache与Suspense使用

24、怎样动态导入组件?

import

require

react-loadable

25、如何给非控组件设置默认的值?

答案借鉴地址

在React中,所谓受控组件和非受控组件,是针对表单而言的。

  • 表单元素依赖于状态(state),表单元素需要默认值实时映射到状态的时候,就是受控组件
<input name="username" type="text" value={this.state.username} onChange={this.handleChange} />
  • 不通过state控制表单元素,而是通过ref来控制的表单元素就是非受控组件
<input name="username" type="text" ref={username=>this.username=username}/>
26、怎样使用Hooks获取服务端数据?

需要 useEffectuseState配合

要求函数式写法,不能是class写法

///案例一
const {data,setData} = useState({});//渲染值
///页面初始化
useEffect(()=>{
    axios.get().then(res => {
        setData(res);
    })
},[]);
///案例二
const {data,setData} = useState({});//渲染值
///条件调用
const getPostData = ({data}) =>{
     axios.get(data).then(res => {
        setData(res);
    })
}
27、使用Hooks要遵守哪些原则?

官网说的

只在最顶层使用 Hook

不要在循环,条件或嵌套函数中调用 Hook, 确保总是在你的 React 函数的最顶层以及任何 return 之前调用他们。遵守这条规则,你就能确保 Hook 在每一次渲染中都按照同样的顺序被调用。这让 React 能够在多次的 useStateuseEffect 调用之间保持 hook 状态的正确。(如果你对此感到好奇,我们在下面会有更深入的解释。)

只在 React 函数中调用 Hook

不要在普通的 JavaScript 函数中调用 Hook。你可以:

  • ✅ 在 React 的函数组件中调用 Hook
  • ✅ 在自定义 Hook 中调用其他 Hook (我们将会在下一页 中学习这个。)

遵循此规则,确保组件的状态逻辑在代码中清晰可见。

28、render方法的原理你有了解吗?它返回的数据类型是什么?