在React中,可以通过数组方法返回一组 组件元素,并将该数组作为render()的js插值。
1 function NumberList(props) { 2 const numbers = props.numbers; 3 const listItems = numbers.map((number) => 4 <li>{number}</li> 5 ); 6 return ( 7 <ul>{listItems}</ul> 8 ); 9 } 10 11 const numbers = [1, 2, 3, 4, 5]; 12 ReactDOM.render( 13 <NumberList numbers={numbers} />, 14 document.getElementById('root') 15 );
当我们创建列表时,要记得在列表元素中(不单指li,一组div或其他以数组形式渲染的标签也一样)添加key特性。key的值是一个字符串,它表示在列表中该元素的标志(或者说id)。key的作用主要是让DOM在重新渲染前能比较出列表中的哪些元素不必重新渲染,降低了开销。“When children have keys, React uses the key to match children in the original tree with children in the subsequent tree.”-- in-depth explanation about why keys are necessary
1 const listItems = numbers.map((number) => 2 <li key={number.toString()}> 3 {number} 4 </li> 5 );
生成组件数组时,如果元素本身有id,我们将id作为数组元素的key即可;如果没有的话,我们往往会使用index索引替代。但文档不推荐,因为当生成的列表在以后的行为中发生动态增删/重新排序的话,索引就没意义了,此时重新渲染的效率依然不高。
为什么呢?因为React使用key,是为了标识组件。key相同的时候,React只会更新组件的已变化属性,key不同时,才会销毁重新创建。
举例,用数组['Alan','Mar','Kat']生成一组li,以索引为key,得到:
(key0)Alan,input1
(key1)Mar ,input2
(key2)Kat ,input3
对数组进行排序后,['Kat','Alan','Mar'],此时再以索引为key生成列表,得到:
(key0)Kat ,input1
(key1)Alan ,input2
(key2)Mar ,input3
发现名字后面对应的原先的input并未发现更新。因为对React来说,key相同的组件被React认定是同一个组件,哪怕名字在排序后改变,并在重新渲染后体现,但原先的key0和现在的key0仍被看作同一个组件,其底下的input不会发生更改。
*key应该放在数组所在上下文的(自定义)元素的特性上。
1 const listItems = numbers.map((number) => 2 // Correct! Key should be specified inside the array. 3 <ListItem key={number.toString()} 4 value={number} /> 5 ); 6 return ( 7 <ul> 8 {listItems} 9 </ul> 10 );
由于往往用数组方法如map生成组件数组,因此在map方法中给元素加key就好了。
在render函数return的JSX中:
1 {numbers.map((number) => 2 <ListItem key={number.toString()} 3 value={number} /> 4 )} 5 6 7 8 等价于 9 const listItems = numbers.map((number) => 10 <ListItem key={number.toString()} 11 value={number} /> 12 ); 13 //... 14 15 <ul> 16 {listItems} 17 </ul>
刚看到一篇关于react的所谓学习路线:react入门、react router、redux 然后是 react native
这让我想起vue、vue router、vuex.......
真的吼相似.......