JSX

嵌入表达式

  1. const name = 'Josh Perez';
  2. const element = <h1>Hello, {name}</h1>;

在 JSX 语法中,你可以在大括号内放置任何有效的 JavaScript 表达式。例如,2 + 2,user.firstName 或 formatName(user) 都是有效的 JavaScript 表达式,同时JSX本身也属于表达式,也就是说,你可以在 if 语句和 for 循环的代码块中使用 JSX,将 JSX 赋值给变量,把 JSX 当作参数传入,以及从函数中返回 JSX:

  1. function getGreeting(user) {
  2. if (user) {
  3. return <h1>Hello, {formatName(user)}!</h1>;
  4. }
  5. return <h1>Hello, Stranger.</h1>;
  6. }

JSX特定属性

你可以通过使用引号,来将属性值指定为字符串字面量:

  1. const element = <div tabIndex="0"></div>;

也可以使用大括号,来在属性值中插入一个 JavaScript 表达式:

  1. const element = <img src={user.avatarUrl}></img>;

在属性中嵌入 JavaScript 表达式时,不要在大括号外面加上引号。你应该仅使用引号(对于字符串值)或大括号(对于表达式)中的一个,对于同一属性不能同时使用这两种符号。

防止注入攻击

你可以安全地在 JSX 当中插入用户输入内容:

  1. const title = response.potentiallyMaliciousInput;
  2. // 直接使用是安全的:
  3. const element = <h1>{title}</h1>;

React DOM 在渲染所有输入内容之前,默认会进行转义。它可以确保在你的应用中,永远不会注入那些并非自己明确编写的内容。所有的内容在渲染之前都被转换成了字符串。这样可以有效地防止XSS(cross-site-scripting, 跨站脚本)攻击(使用dangerouslySetInnerHTML展示的html串应该依旧不安全)。

Props

只读性

组件无论是使用函数声明还是通过 class 声明,都决不能修改自身的 props。
所有 React 组件都必须像纯函数一样保护它们的 props 不被更改。

条件渲染

与运算符 &&

在花括号嵌入逻辑与 (&&) 运算符可以很方便地进行元素的条件渲染。

  1. function Mailbox(props) {
  2. const unreadMessages = props.unreadMessages;
  3. return (
  4. <div>
  5. <h1>Hello!</h1>
  6. {unreadMessages.length > 0 &&
  7. <h2>
  8. You have {unreadMessages.length} unread messages.
  9. </h2>
  10. }
  11. </div>
  12. );
  13. }

之所以能这样做,是因为在 JavaScript 中,true && expression 总是会返回 expression, 而 false && expression 总是会返回 false。

因此,如果条件是 true,&& 右侧的元素就会被渲染,如果是 false,React 会忽略并跳过它。

三目运算符

另一种内联条件渲染的方法是使用 JavaScript 中的三目运算符 condition ? true : false。
在下面这个示例中,我们用它来条件渲染一小段文本

  1. render() {
  2. const isLoggedIn = this.state.isLoggedIn;
  3. return (
  4. <div>
  5. The user is <b>{isLoggedIn ? 'currently' : 'not'}</b> logged in.
  6. </div>
  7. );
  8. }

同样的,它也可以用于较为复杂的表达式中,虽然看起来不是很直观:

  1. render() {
  2. const isLoggedIn = this.state.isLoggedIn;
  3. return (
  4. <div>
  5. {isLoggedIn
  6. ? <LogoutButton onClick={this.handleLogoutClick} />
  7. : <LoginButton onClick={this.handleLoginClick} />
  8. }
  9. </div>
  10. );
  11. }

在编写代码时,应该选择可读性更高的代码风格。如果条件变得过于复杂,那你应该考虑如何提取组件。

阻止组件渲染

在极少数情况下,你可能希望能隐藏组件,即使它已经被其他组件渲染。若要完成此操作,你可以让 render 方法直接返回 null,而不进行任何渲染。

下面的示例中, 会根据 prop 中 warn 的值来进行条件渲染。如果 warn 的值是 false,那么组件则不会渲染:

  1. function WarningBanner(props) {
  2. if (!props.warn) {
  3. return null;
  4. }
  5. return (
  6. <div className="warning">
  7. Warning!
  8. </div>
  9. );
  10. }
  11. class Page extends React.Component {
  12. constructor(props) {
  13. super(props);
  14. this.state = {showWarning: true};
  15. this.handleToggleClick = this.handleToggleClick.bind(this);
  16. }
  17. handleToggleClick() {
  18. this.setState(state => ({
  19. showWarning: !state.showWarning
  20. }));
  21. }
  22. render() {
  23. return (
  24. <div>
  25. <WarningBanner warn={this.state.showWarning} />
  26. <button onClick={this.handleToggleClick}>
  27. {this.state.showWarning ? 'Hide' : 'Show'}
  28. </button>
  29. </div>
  30. );
  31. }
  32. }
  33. ReactDOM.render(
  34. <Page />,
  35. document.getElementById('root')
  36. );

在组件的 render 方法中返回 null 并不会影响组件的生命周期。例如,上面这个示例中,componentDidUpdate 依然会被调用。

列表&key

渲染多个组件

你可以通过使用 {} 在 JSX 内构建一个元素集合。

下面,我们使用 Javascript 中的 map() 方法来遍历 numbers 数组。将数组中的每个元素变成

  • 标签,将得到的数组插入到
      元素中,然后渲染进 DOM:

      1. ReactDOM.render(
      2. <ul>{numbers.map((number) =><li>{number}</li>)}</ul>,
      3. document.getElementById('root')
      4. );

      key

      通过map生成的元素集合需要设置一个key,key 帮助 React 识别哪些元素改变了,比如被添加或删除(详细可以了解diff)。因此你应当给数组中的每一个元素赋予一个确定的标识

      1. const numbers = [1, 2, 3, 4, 5];
      2. const listItems = numbers.map((number) =>
      3. <li key={number.toString()}>
      4. {number}
      5. </li>
      6. );

      一个元素的 key 最好是这个元素在列表中拥有的一个独一无二的字符串。通常,我们使用数据中的 id 来作为元素的 key:

      1. const todoItems = todos.map((todo) =>
      2. <li key={todo.id}>
      3. {todo.text}
      4. </li>
      5. );

      如果使用index作为key的话,由于diff算法可能会以key来做对比,所以当列表顺序变化的时候会使得性能变差。

      另外key需要在整个列表唯一,但是不需要在全局是唯一的,因此两个元素组可以出现相同的key。

      表单

      受控组件

      在 HTML 中,表单元素(如、 )之类的表单元素通常自己维护 state,并根据用户输入进行更新。而在 React 中,可变状态(mutable state)通常保存在组件的 state 属性中,并且只能通过使用 setState()来更新。

      我们可以把两者结合起来,使 React 的 state 成为“唯一数据源”。渲染表单的 React 组件还控制着用户输入过程中表单发生的操作。被 React 以这种方式控制取值的表单输入元素就叫做“受控组件”。

      在表单元素上设置了 value 属性,因此显示的值将始终为 this.state.value,这使得 React 的 state 成为唯一数据源。由于 handlechange 在每次按键时都会执行并更新 React 的 state,因此显示的值将随着用户输入而更新。

      select 标签

      在 HTML 中, 创建下拉列表标签。例如,如下 HTML 创建了水果相关的下拉列表:

      1. <select>
      2. <option value="grapefruit">葡萄柚</option>
      3. <option value="lime">酸橙</option>
      4. <option selected value="coconut">椰子</option>
      5. <option value="mango">芒果</option>
      6. </select>

      由于 selected 属性的缘故,椰子选项默认被选中。React 并不会使用 selected 属性,而是在根 select 标签上使用 value 属性。

      处理多个输入

      当需要处理多个 input 元素时,我们可以给每个元素添加 name 属性,并让处理函数根据event.target.name的值选择要执行的操作

      1. class Reservation extends React.Component {
      2. constructor(props) {
      3. super(props);
      4. this.state = {
      5. isGoing: true,
      6. numberOfGuests: 2
      7. };
      8. this.handleInputChange = this.handleInputChange.bind(this);
      9. }
      10. handleInputChange(event) {
      11. const target = event.target;
      12. const value = target.name === 'isGoing' ? target.checked : target.value;
      13. const name = target.name;
      14. this.setState({
      15. [name]: value
      16. });
      17. }
      18. render() {
      19. return (
      20. <form>
      21. <label>
      22. 参与:
      23. <input
      24. name="isGoing"
      25. type="checkbox"
      26. checked={this.state.isGoing}
      27. onChange={this.handleInputChange} />
      28. </label>
      29. <br />
      30. <label>
      31. 来宾人数:
      32. <input
      33. name="numberOfGuests"
      34. type="number"
      35. value={this.state.numberOfGuests}
      36. onChange={this.handleInputChange} />
      37. </label>
      38. </form>
      39. );
      40. }
      41. }

      受控输入空值

      在受控组件上指定 value 的 prop 会阻止用户更改输入。如果你指定了 value,但输入仍可编辑,则可能是你意外地将value 设置为 undefined 或 null。

      1. ReactDOM.render(<input value="hi" />, mountNode);
      2. setTimeout(function() {
      3. ReactDOM.render(<input value={null} />, mountNode);
      4. }, 1000);

      状态提升

      多个组件需要反映相同的变化数据,这时将共享状态提升到最近的共同父组件中去即为状态提升

      基本来说,就是将多个组件需要一起变化的数据通过父组件传入,在子组件中通过this.props来渲染,同时,因为react的props为只读,不能修改,所以要从父组件中传入修改该值的方法,当需要修改的时候子组件调用该方法来修改父组件中的state来实现共享状态,即将子组件的状态提升