亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > JS > 正文

詳解react阻止無效重渲染的多種方式

2024-05-06 16:47:05
字體:
來源:轉載
供稿:網友

在開發React組件的過程中,我們經常會遇到這個問題:什么情況下組件會重新渲染?

當內部data發生改變,state發生改變(通過調用this.setState()) 以及父組件傳過來的props發生改變時,會導致組件重新渲染。

以下幾個問題同樣值得我們思考:

setState()函數在任何情況下都會導致組件重渲染嗎?如果setState中的state沒有發生改變呢?

如果state和從父組件傳過來的props都沒變化,那他就一定不會發生重渲染嗎?

首先,我們來解決這兩個問題

沒有導致state的值發生變化的this.setState()是否會導致重渲染  --- 

import React from 'react'class Test extends React.Component{ constructor(props) {  super(props);  this.state = {   Number:1//設state中Number值為1  } } //這里調用了setState但是并沒有改變setState中的值 handleClick = () => {   const preNumber = this.state.Number   this.setState({    Number:this.state.Number   }) } render(){  //當render函數被調用時,打印當前的Number  console.log(this.state.Number)  return(<h1 onClick = {this.handleClick}>       {this.state.Number}      </h1>) }}

從控制臺的打印結果可以看出:共打印了15次1,但是組件并沒有發生任何變化!?。?/p>

這樣的結果不是我們想要的,如何阻止組件的重渲染呢?這時我們想到了React的一個生命周期鉤子 shouldComponentUpdate

react生命周期中有這樣一個鉤子,叫shouldComponentUpdate函數,是重渲染時render()函數調用前被調用的函數,

兩個參數 nextProps和nextState ,分別表示下一個props和state的值。

當函數返回false時,阻止接下來的render()函數的調用,阻止組件重渲染,返回true時,組件照常渲染

 //加入shouldComponentUpdate鉤子//在render函數調用前判斷:如果前后state中Number不變,通過return false阻止render調用 shouldComponentUpdate(nextProps,nextState){   if(nextState.Number == this.state.Number){    return false   } }

加入上述代碼后,打開控制臺,點擊按鈕,還是白白的,說明無效的重渲染被我們阻止了

第二個問題,組件的state和從父組件傳遞過來的props都沒改變,組件還會重渲染嗎 --- 可能

同樣可以通過shouldComponentUpdate鉤子進行阻止

所以說,前后不改變state的值的setState和無數據交換的父組件的重渲染都會導致組件的重渲染,但我們可以通過shouldComponentUpdate來阻止這兩種情況

shouldComponentUpdate并不是完美的,只能阻止扁平的對象

nextState.Number == this.state.Number

如果調用層次比較深

nextState.NumberObject.number == this.state.NumberObject.number

Number 是一個數字變量

NumberObject是一個對象

數字變量(number類型)和對象(Object)類型的內存存儲機制不同

這時候,因為兩者都指向堆中的同一個對象,所以一直都是true  shouldComponentUpdate失效了

js變量分為基本類型的變量和引用類型的變量

對于number,string,boolean,undefined,null這些基本類型變量,值存在棧中

對于object,Array,function這些引用類型變量,引用存在棧中,而不同的引用卻可以指向堆內存中的同一個對象

那么,問題就來了

怎么樣才能取到不同的NumberObject呢?

四種方法:

1、ES6的擴展語法Object.assign()

2、深拷貝/淺拷貝或利用JSON.parse(JSON.stringify(data))相當于深拷貝,但使用受一定限制

3、引入immutable.js react官方推薦的第三方庫

4、繼承react的PureComponent組件(代替Component)

在js中,引用類型的數據,優點在于頻繁的操作數據都是在原對象的基礎上修改,不會創建新對象,從而可以有效的利用內存,不會浪費內存,這種特性稱為mutable(可變),但恰恰它的優點也是它的缺點,太過于靈活多變在復雜數據的場景下也造成了它的不可控性,假設一個對象在多處用到,在某一處不小心修改了數據,其他地方很難預見到數據是如何改變的,針對這種問題的解決方法,一般就像剛才的例子,會想復制一個新對象,再在新對象上做修改,這無疑會造成更多的性能問題以及內存浪費。

為了解決這種問題,出現了immutable對象,每次修改immutable對象都會創建一個新的不可變對象,而老的對象不會改變。

immutable.js主要有三大特性:

Persistent data structure (持久化數據結構)

structural sharing (結構共享)

support lazy operation (惰性操作)

Immutable Data 就是一旦創建,就不能再被更改的數據。對 Immutable 對象的任何修改或添加刪除操作都會返回一個新的 Immutable 對象。Immutable 實現的原理是 Persistent Data Structure(持久化數據結構),也就是使用舊數據創建新數據時,要保證舊數據同時可用且不變。同時為了避免 deepCopy 把所有節點都復制一遍帶來的性能損耗,Immutable 使用了 Structural Sharing(結構共享),即如果對象樹中一個節點發生變化,只修改這個節點和受它影響的父節點,其它節點則進行共享

三個最重要的數據結構: Map List Set

Map:鍵值對集合,對應于 Object,ES6 也有專門的 Map 對象

List:有序可重復的列表,對應于 Array

Set:無序且不可重復的列表

//Map() 原生object轉Map對象 (只會轉換第一層,注意和fromJS區別)immutable.Map({name:'danny', age:18})//List() 原生array轉List對象 (只會轉換第一層,注意和fromJS區別)immutable.List([1,2,3,4,5])//fromJS()  原生js轉immutable對象 (深度轉換,會將內部嵌套的對象和數組全部轉成immutable)immutable.fromJS([1,2,3,4,5])  //將原生array --> Listimmutable.fromJS({name:'danny', age:18})  //將原生object --> Map//toJS() immutable對象轉原生js (深度轉換,會將內部嵌套的Map和List全部轉換成原生js)immutableData.toJS();//查看List或者map大小 immutableData.size 或者 immutableData.count()// is()  判斷兩個immutable對象是否相等immutable.is(imA, imB);//merge() 對象合并var imA = immutable.fromJS({a:1,b:2});var imA = immutable.fromJS({c:3});var imC = imA.merge(imB);console.log(imC.toJS()) //{a:1,b:2,c:3}

對于兩個一樣的數據,只有通過equals進行比較才是相等的  ==  ===都不行

如果 某個是另一個克隆出來的,那么全部都相等

push添加 unshift在頭部添加 concat組合  返回的是新數據,而不是數據的長度

//增刪改查(所有操作都會返回新的值,不會修改原來值)var immutableData = immutable.fromJS({  a:1,  b:2,  c:{    d:3  }});var data1 = immutableData.get('a') // data1 = 1 var data2 = immutableData.getIn(['c', 'd']) // data2 = 3  getIn用于深層結構訪問var data3 = immutableData.set('a' , 2);  // data3中的 a = 2var data4 = immutableData.setIn(['c', 'd'], 4);  //data4中的 d = 4var data5 = immutableData.update('a',function(x){return x+4}) //data5中的 a = 5var data6 = immutableData.updateIn(['c', 'd'],function(x){return x+4}) //data6中的 d = 7var data7 = immutableData.delete('a')  //data7中的 a 不存在var data8 = immutableData.deleteIn(['c', 'd'])  //data8中的 d 不存在復制代碼

優點:

  • 降低mutable帶來的復雜度
  • 節省內存
  • 歷史追溯性(時間旅行):時間旅行指的是,每時每刻的值都被保留了,想回退到哪一步只要簡單的將數據取出就行,想一下如果現在頁面有個撤銷的操作,撤銷前的數據被保留了,只需要取出就行,這個特性在redux或者flux中特別有用
  • 擁抱函數式編程:immutable本來就是函數式編程的概念,純函數式編程的特點就是,只要輸入一致,輸出必然一致,相比于面向對象,這樣開發組件和調試更方便

缺點:

  • 需要重新學習api
  • 資源包大小增加(源碼5000行左右)
  • 容易與原生對象混淆:由于api與原生不同,混用的話容易出錯。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JavaScript/Ajax教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成年人精品视频| 高清日韩电视剧大全免费播放在线观看| 亚洲欧美日韩在线一区| 精品亚洲一区二区三区在线观看| 亚洲成人黄色在线| 久久五月天色综合| 久久久久久久一| 毛片精品免费在线观看| 国产中文字幕亚洲| 欧美在线亚洲一区| 欧美极品少妇xxxxⅹ裸体艺术| 91超碰中文字幕久久精品| 欧美一区第一页| 欧美性xxxxx极品| 麻豆精品精华液| 欧美电影在线免费观看网站| 精品国产一区二区三区久久| 岛国视频午夜一区免费在线观看| 亚洲视频在线免费看| 国产精品久久久久久久9999| 亚洲精品福利视频| 黑人巨大精品欧美一区二区一视频| 91免费在线视频网站| 久久精品国产91精品亚洲| 中文字幕av一区| 国模私拍一区二区三区| 红桃视频成人在线观看| 日本精品性网站在线观看| 亚洲欧美一区二区三区四区| 午夜精品三级视频福利| 久久久久免费精品国产| 国产精品视频一区国模私拍| 欧美大尺度激情区在线播放| 91精品视频在线免费观看| 亚洲欧美中文在线视频| 狠狠综合久久av一区二区小说| 亚洲欧美日韩天堂| 亚洲最大中文字幕| 欧美人成在线视频| 久久久免费精品视频| 91亚洲精品在线| 日韩福利伦理影院免费| 欧美在线播放视频| 亚洲自拍高清视频网站| 亚洲图片欧洲图片av| 欧美激情视频在线免费观看 欧美视频免费一| 国产综合香蕉五月婷在线| 色综合久久精品亚洲国产| 亚洲精品国产精品自产a区红杏吧| 亚洲欧美综合区自拍另类| 国产成+人+综合+亚洲欧美丁香花| 中文字幕一精品亚洲无线一区| 日韩在线观看网址| 2020欧美日韩在线视频| 亚洲精品动漫久久久久| 91精品久久久久久综合乱菊| 国内精品模特av私拍在线观看| 日韩男女性生活视频| 538国产精品一区二区在线| 亚洲精品日产aⅴ| 亚洲天堂第一页| 91日韩在线视频| 欧美精品久久久久| 最新69国产成人精品视频免费| 亚洲欧美另类国产| 在线视频欧美日韩精品| 亚洲综合大片69999| 欧美肥婆姓交大片| 久久久久www| 日韩**中文字幕毛片| 久久人人爽国产| 一区二区成人av| 国产午夜精品一区二区三区| 精品久久久国产| 国产精品欧美亚洲777777| 欧美成人免费全部观看天天性色| 欧美激情一区二区三区久久久| 国产精品久久久91| 国产偷国产偷亚洲清高网站| 久久久国产精品一区| 另类天堂视频在线观看| 久久久久北条麻妃免费看| 97人人模人人爽人人喊中文字| 亚洲精品色婷婷福利天堂| 高清欧美一区二区三区| 久久久在线视频| 亚洲一区亚洲二区亚洲三区| 91麻豆国产语对白在线观看| 91po在线观看91精品国产性色| 欧美性高跟鞋xxxxhd| 国产精品精品视频一区二区三区| 免费不卡在线观看av| 一区二区欧美在线| 97香蕉超级碰碰久久免费软件| 国内精品久久久久影院优| 欧美激情按摩在线| 久久久精品2019中文字幕神马| 午夜精品福利电影| 亚洲欧美激情一区| 国产精品亚发布| 亚洲男人天堂2019| 久久精品亚洲94久久精品| 日韩在线观看免费高清完整版| 久久久久国产精品免费网站| 亚洲欧美日韩一区在线| 久久久久久91香蕉国产| 国产亚洲精品一区二555| 黑人极品videos精品欧美裸| 欧美性xxxxx极品娇小| 国产精品高潮在线| 国产精品爽爽ⅴa在线观看| 国产精品永久在线| 欧美国产极速在线| 国产成人亚洲综合青青| 91精品国产色综合| 一本色道久久综合狠狠躁篇怎么玩| 91综合免费在线| 欧美日韩成人网| 国产日韩在线精品av| 日韩av理论片| 亚洲欧美国产制服动漫| 亚洲护士老师的毛茸茸最新章节| 成人a级免费视频| 91精品啪aⅴ在线观看国产| 国产第一区电影| 日韩免费在线播放| 在线精品视频视频中文字幕| 日韩欧美在线视频观看| 亚洲成年人在线播放| 亚洲欧美在线播放| 成人免费视频网址| 亚洲成色999久久网站| 夜色77av精品影院| 国产不卡av在线| 国产精品美女久久久免费| 久久在精品线影院精品国产| 国产精品大陆在线观看| 91久热免费在线视频| 有码中文亚洲精品| 日韩美女在线播放| 亚洲日韩第一页| 亚洲免费电影在线观看| 亚洲偷熟乱区亚洲香蕉av| 亚洲图中文字幕| 欧洲成人性视频| 欧美一乱一性一交一视频| 久久久之久亚州精品露出| 中文字幕成人在线| 亚洲色图欧美制服丝袜另类第一页| 久久久精品久久久久| 韩剧1988免费观看全集| 午夜精品一区二区三区在线播放| 久久不射电影网| 91久久精品国产91性色| 亚洲无亚洲人成网站77777| 另类美女黄大片| 亚洲影影院av| 日韩精品在线电影| 91在线观看免费高清| 欧美最猛黑人xxxx黑人猛叫黄| 午夜精品一区二区三区在线视频| 亚洲丁香久久久| 麻豆国产精品va在线观看不卡| www日韩中文字幕在线看|