摘要:事件模型事件捕獲階段。事件到達目標元素觸發(fā)目標元素的監(jiān)聽函數(shù)。的狀態(tài)值與狀態(tài)碼的狀態(tài)值未初始化還沒有調用方法。載入完成已經執(zhí)行完成,已經接收到全部的響應內容。
前言
總括: 包含這三個月來碰到的一些覺得比較好的面試題,三個月沒怎么寫博客著實有些手癢,哈哈哈。7000余字,不成敬意2333
原文地址:我的前端進階之路
知乎專欄&&簡書專題:前端進擊者(知乎)&&前端進擊者(簡書)
博主博客地址:Damonare的個人博客
烈火試真金,逆境試強者
正文 React和Vue對比相同點:
數(shù)據(jù)驅動視圖,提供響應式的視圖組件
都有Virtual DOM,組件化開發(fā),通過props參數(shù)進行父子組件數(shù)據(jù)的傳遞,都實現(xiàn)webComponents規(guī)范
數(shù)據(jù)流動單向
都支持服務端渲染
都有支持native的方案,React的React native,Vue的weex
不同點:
社區(qū):React社區(qū)還是要比vue大很多;
開發(fā)模式:React在view層侵入性還是要比Vue大很多的,React嚴格上只針對MVC的view層,Vue則是MVVM模式的一種實現(xiàn);
數(shù)據(jù)綁定:Vue有實現(xiàn)了雙向數(shù)據(jù)綁定,React數(shù)據(jù)流動是單向的
數(shù)據(jù)渲染:對于大規(guī)模數(shù)據(jù)渲染,React要比Vue更快,渲染機制啟動時候要做的工作比較多;
數(shù)據(jù)更新方面:Vue 由于采用依賴追蹤,默認就是優(yōu)化狀態(tài):你動了多少數(shù)據(jù),就觸發(fā)多少更新,不多也不少。React在復雜的應用里有兩個選擇:
(1). 手動添加 shouldComponentUpdate 來避免不需要的 vdom re-render。 (2).Components 盡可能都用 pureRenderMixin,然后采用 redux 結構 + Immutable.js;
開發(fā)風格的偏好:React 推薦的做法是 JSX + inline style,也就是把 HTML 和 CSS 全都寫進 JavaScript 了,即"all in js";Vue進階之后推薦的是使用 webpack + vue-loader 的單文件組件格式,即html,css,js寫在同一個文件;
使用場景:React配合Redux架構適合超大規(guī)模多人協(xié)作的復雜項目;Vue則適合小快靈的項目。對于需要對 DOM 進行很多自定義操作的項目,Vue 的靈活性優(yōu)于 React;
Vue要比React更好上手,具體可能體現(xiàn)在很多人不熟悉React的JSX語法和函數(shù)式編程的思想,以及想要發(fā)揮出React的最大威力需要學習它一系列生態(tài)的緣故;
Vue著重提高開發(fā)效率,讓前端程序員更快速方便的開發(fā)應用。React著重于變革開發(fā)思想,提升前端程序員編程的深度與創(chuàng)造力,讓前端工程師成為真正的程序員而不是UI的構建者;
gulp和webpack區(qū)別gulp是一種工具,我們可以用它來優(yōu)化前端的工作流程,比如自動刷新頁面、combo、壓縮css、js、編譯less等等。具體體現(xiàn)為:在gulp的配置文件中書寫一個個的task,webpack則是一種打包工具,或者說是一種模塊化解決方案,實際上很大一部分人剛開始使用webpack的方式就是通過gulp-webpack這個插件,寫好task來使用webpack對前端的一些文件進行打包;
gulp的處理任務需要自己去寫,webpack則有現(xiàn)成的解決方案,只需要在webpack.config.js配置好即可;
防止重復發(fā)送Ajax請求用戶點擊之后按鈕disabled;
函數(shù)節(jié)流
abort掉上一個請求。
事件模型事件捕獲階段(capturing phase)。事件從document一直向下傳播到目標元素, 依次檢查經過的節(jié)點是否綁定了事件監(jiān)聽函數(shù),如果有則執(zhí)行。
事件處理階段(target phase)。事件到達目標元素, 觸發(fā)目標元素的監(jiān)聽函數(shù)。
事件冒泡階段(bubbling phase)。事件從目標元素冒泡到document, 依次檢查經過的節(jié)點是否綁定了事件監(jiān)聽函數(shù),如果有則執(zhí)行。
瀏覽器緩存機制Expires策略
Expires是Web服務器響應消息頭字段,在響應http請求時告訴瀏覽器在過期時間前瀏覽器可以直接從瀏覽器緩存取數(shù)據(jù),而無需再次請求。Expires 是HTTP 1.0的東西,現(xiàn)在默認瀏覽器均默認使用HTTP 1.1,所以它的作用基本忽略。
Cache-Control策略
Cache-Control與Expires的作用一致,都是指明當前資源的有效期,控制瀏覽器是否直接從瀏覽器緩讀取數(shù)據(jù)還是重新發(fā)請求到服務器取數(shù)據(jù)。只不過Cache-Control的選擇更多,設置更細致,如果同時設置的話,其優(yōu)先級高于Expires。
以上是設置緩存時間的兩種方法。那么當緩存時間過了咋整呢?有人肯定說了,那就再次發(fā)起請求啊,這是對的。問題是如果服務器資源并沒有更新呢?比如說我有一個jQuery.js文件已經緩存了,當它的緩存時間到了之后服務器的jQuery.js文件也沒有更新,那實際上我們直接使用本地緩存的文件就可以??!沒必要浪費帶寬和時間去重新請求一個新的文件??!這時候我們就需要再進一步看一下HTTP協(xié)議里這幾個參數(shù)的作用了。
Last-Modified/If-Modified-Since
首先Last-Modified/If-Modified-Since要配合Cache-Control使用。
Last-Modified:標示這個響應資源的最后修改時間。web服務器在響應請求時,告訴瀏覽器資源的最后修改時間(這個參數(shù)是和Cache-Control一起過來的)。
If-Modified-Since:當資源過期時(使用Cache-Control標識的max-age),發(fā)現(xiàn)資源具有Last-Modified聲明,則再次向web服務器請求時帶上頭 If-Modified-Since,表示請求時間。web服務器收到請求后發(fā)現(xiàn)有頭If-Modified-Since ,則與被請求資源的最后修改時間進行比對。若最后修改時間較新,說明資源又被改動過,則響應整片資源內容(寫在響應消息包體內),HTTP 200;若最后修改時間較舊,說明資源無新修改,則響應HTTP 304 (無需包體,節(jié)省瀏覽),告知瀏覽器繼續(xù)使用所保存的cache。
ETag/If-None-Match
Etag/If-None-Match也要配合Cache-Control使用。
Etag:web服務器響應請求時,告訴瀏覽器當前資源在服務器的唯一標識(生成規(guī)則由服務器覺得)。Apache中,ETag的值,默認是對文件的索引節(jié)(INode),大?。⊿ize)和最后修改時間(MTime)進行Hash后得到的。
If-None-Match:當資源過期時(使用Cache-Control標識的max-age),發(fā)現(xiàn)資源具有Etage聲明,則再次向web服務器請求時帶上頭If-None-Match(Etag的值)。web服務器收到請求后發(fā)現(xiàn)有頭If-None-Match 則與被請求資源的相應校驗串進行比對,決定返回200或304。
ETag和Last-Modified
HTTP1.1中Etag的出現(xiàn)主要是為了解決幾個Last-Modified比較難解決的問題:
Last-Modified標注的最后修改只能精確到秒級,如果某些文件在1秒鐘以內,被修改多次的話,它將不能準確標注文件的修改時間
如果某些文件會被定期生成,當有時內容并沒有任何變化,但Last-Modified卻改變了,導致文件沒法使用緩存
有可能存在服務器沒有準確獲取文件修改時間,或者與代理服務器時間不一致等情形
Etag是服務器自動生成或者由開發(fā)者生成的對應資源在服務器端的唯一標識符,能夠更加準確的控制緩存。Last-Modified與ETag是可以一起使用的,服務器會優(yōu)先驗證ETag,一致的情況下,才會繼續(xù)比對Last-Modified,最后才決定是否返回304。
Ajax的狀態(tài)值與HTTP狀態(tài)碼Ajax的狀態(tài)值
0: (未初始化)還沒有調用send()方法。
1: (載入)已經調用send()方法,正在派發(fā)請求。
2: (載入完成)send()已經執(zhí)行完成,已經接收到全部的響應內容。
3: (交互)正在解析響應內容。
4: (完成)響應內容已經解析完成,用戶可以調用。
HTTP狀態(tài)碼
200 & OK: 請求成功;
204 & No Content: 請求處理成功,但沒有資源可以返回;
206 & Partial Content: 對資源某一部分進行請求(比如對于只加載了一般的圖片剩余部分的請求);
301 & Move Permanently: 永久性重定向;
302 & Found: 臨時性重定向;
303 & See Other: 請求資源存在另一個URI,應使用get方法請求;
304 & Not Modified: 服務器判斷本地緩存未更新,可以直接使用本地的緩存;
307 & Temporary Redirect: 臨時重定向;
400 & Bad Request: 請求報文存在語法錯誤;
401 & Unauthorized: 請求需要通過HTTP認證;
403 & Forbidden: 請求資源被服務器拒絕,訪問權限的問題;
404 & Not Found: 服務器上沒有請求的資源;
500 & Internal Server Error: 服務器執(zhí)行請求時出現(xiàn)錯誤;
502 & Bad Gateway: 錯誤的網關;
503 & Service Unavailable: 服務器超載或正在維護,無法處理請求;
504 & Gateway timeout: 網關超時;
React-router原理1.History
老瀏覽器的history: 主要通過hash來實現(xiàn),對應createHashHistory
高版本瀏覽器: 通過html5里面的history,對應createBrowserHistory
node環(huán)境下: 主要存儲在memeory里面,對應createMemoryHistory
內部createHistory實現(xiàn):
// 內部的抽象實現(xiàn) function createHistory(options={}) { ... return { listenBefore, // 內部的hook機制,可以在location發(fā)生變化前執(zhí)行某些行為,AOP的實現(xiàn) listen, // location發(fā)生改變時觸發(fā)回調 transitionTo, // 執(zhí)行l(wèi)ocation的改變 push, // 改變location replace, go, goBack, goForward, createKey, // 創(chuàng)建location的key,用于唯一標示該location,是隨機生成的 createPath, createHref, createLocation, // 創(chuàng)建location } }
createLocation方法:
function createLocation() { return { pathname, // url的基本路徑 search, // 查詢字段 hash, // url中的hash值 state, // url對應的state字段 action, // 分為push、replace、pop三種 key // 生成方法為: Math.random().toString(36).substr(2, length) } }
三種方法各自執(zhí)行URL前進的方式:
createBrowserHistory: pushState、replaceState
createHashHistory: location.hash=*** location.replace()
createMemoryHistory: 在內存中進行歷史記錄的存儲
偽代碼實現(xiàn):
// createBrowserHistory(HTML5)中的前進實現(xiàn) function finishTransition(location) { ... const historyState = { key }; ... if (location.action === "PUSH") ) { window.history.pushState(historyState, null, path); } else { window.history.replaceState(historyState, null, path) } } // createHashHistory的內部實現(xiàn) function finishTransition(location) { ... if (location.action === "PUSH") ) { window.location.hash = path; } else { window.location.replace( window.location.pathname + window.location.search + "#" + path ); } } // createMemoryHistory的內部實現(xiàn) entries = []; function finishTransition(location) { ... switch (location.action) { case "PUSH": entries.push(location); break; case "REPLACE": entries[current] = location; break; } }
React-router的基本原理
URL對應Location對象,而UI是由react的 components來決定的,這樣就轉變成location與components之間的同步問題。
什么是原型鏈每一個對象都會在內部鏈接到另一個對象(該對象的原型對象),該對象有一個原型prototype,當訪問對象的屬性或是方法的時候,不僅僅會在原對象上查找,還會順著原型鏈在原型對象的原型鏈上查找,直到查到null(所有原型鏈的頂層)為止。原型是JavaScript實現(xiàn)繼承的基礎,new關鍵字做的主要的事情就是將實例對象的__proto__屬性指向原型對象的prototype。
什么是閉包閉包是javascript支持頭等函數(shù)的一種方式,它是一個能夠引用其內部作用域變量(在本作用域第一次聲明的變量)的表達式,這個表達式可以賦值給某個變量,可以作為參數(shù)傳遞給函數(shù),也可以作為一個函數(shù)返回值返回。
閉包是函數(shù)開始執(zhí)行的時候被分配的一個棧幀,在函數(shù)執(zhí)行結束返回后仍不會被釋放(就好像一個棧幀被分配在堆里而不是棧里!)
閉包的應用:
比如寫柯里化函數(shù)的時候利用閉包,保存參數(shù)在內存中;
var currying = function(fun) { //格式化arguments var args = Array.prototype.slice.call(arguments, 1); return function() { //收集所有的參數(shù)在同一個數(shù)組中,進行計算 var _args = args.concat(Array.prototype.slice.call(arguments)); return fun.apply(null, _args); }; }
?
模擬私有變量或是私有方法;
people = (num) => {
var num = num;
return {
increase: () => {
num++;
},
get: () => { return num; } }
}
man = people(4);
man.increase();
man.get();
避免引用錯誤
(var i = 0; i < 4; i++) {
(function(_i) {
setTimeout(function() { console.log(_i)
}, 1000)
})(i)
}
?
圖片懶加載與預加載圖片懶加載的原理就是暫時不設置圖片的src屬性,而是將圖片的url隱藏起來,比如先寫在data-src里面,等某些事件觸發(fā)的時候(比如滾動到底部,點擊加載圖片)再將圖片真實的url放進src屬性里面,從而實現(xiàn)圖片的延遲加載
圖片預加載,是指在一些需要展示大量圖片的網站,實現(xiàn)圖片的提前加載。從而提升用戶體驗。常用的方式有兩種,一種是隱藏在css的background的url屬性里面,一種是通過javascript的Image對象設置實例對象的src屬性實現(xiàn)圖片的預加載。相關代碼如下:
CSS預加載圖片方式:
#preload-01 { background: url(http://domain.tld/image-01.png) no-repeat -9999px -9999px; } #preload-02 { background: url(http://domain.tld/image-02.png) no-repeat -9999px -9999px; } #preload-03 { background: url(http://domain.tld/image-03.png) no-repeat -9999px -9999px; }
Javascript預加載圖片的方式:
function preloadImg(url) { var img = new Image(); img.src = url; if(img.complete) { //接下來可以使用圖片了 //do something here } else { img.onload = function() { //接下來可以使用圖片了 //do something here }; } }跨域
跨域的方式有很多種,最常用的是jsonp主要利用了script的開放策略:通過script標簽引入一個js或者是一個其他后綴形式(如php,jsp等)的文件,此文件返回一個js函數(shù)的調用。缺點在于只支持get請求而且存在安全問題。
CORS跨域,關鍵在于服務器,如果服務器實現(xiàn)了CORS跨域的接口,那么就可以使用ajax(請求路徑為絕對路徑)進行跨域請求。CORS請求分為兩種,一種是簡單請求,一種是非簡單請求。簡單請求是指請求方法在HEAD,GET,POST三者之間并且請求頭信息局限在
Accept
Accept-Language
Content-Language
Content-Type:只限于三個值application/x-www-form-urlencoded、multipart/form-data、text/plain
非簡單請求請求頭:
(1)Access-Control-Request-Method
該字段是必須的,用來列出瀏覽器的CORS請求會用到哪些HTTP方法
(2)Access-Control-Request-Headers
該字段是一個逗號分隔的字符串,指定瀏覽器CORS請求會額外發(fā)送的頭信息字段
執(zhí)行簡單請求的時候,瀏覽器會在請求頭信息增加origin字段,服務器據(jù)此來判斷請求域名是否在許可范圍之內,來決定是否返回Access-Control-Allow-Origin字段。響應頭有以下幾種:
(1)Access-Control-Allow-Origin
該字段是必須的。它的值要么是請求時Origin字段的值,要么是一個*,表示接受任意域名的請求。
(2)Access-Control-Allow-Credentials
該字段可選。它的值是一個布爾值,表示是否允許發(fā)送Cookie。默認情況下,Cookie不包括在CORS請求之中。設為true,即表示服務器明確許可,Cookie可以包含在請求中,一起發(fā)給服務器。這個值也只能設為true,如果服務器不要瀏覽器發(fā)送Cookie,刪除該字段即可。
(3)Access-Control-Expose-Headers
該字段可選。CORS請求時,XMLHttpRequest對象的getResponseHeader()方法只能拿到6個基本字段:Cache-Control、Content-Language、Content-Type、Expires、Last-Modified、Pragma。如果想拿到其他字段,就必須在Access-Control-Expose-Headers里面指定。
(4)Access-Control-Max-Age
Access-Control-Max-Age 首部字段指明了預檢請求的響應的有效時間。
(5)Access-Control-Allow-Methods
Access-Control-Allow-Methods 首部字段用于預檢請求的響應。其指明了實際請求所允許使用的 HTTP 方法。
(6)Access-Control-Allow-Headers
Access-Control-Allow-Headers首部字段用于預檢請求的響應。其指明了實際請求中允許攜帶的首部字段。
其他方法:document.domin,html5的postMessage,window.name等
函數(shù)節(jié)流和函數(shù)防抖函數(shù)節(jié)流讓指函數(shù)有規(guī)律的進行調用,應用場景:window.resize,游戲中子彈發(fā)射(1s只能發(fā)射一顆子彈)等;
函數(shù)防抖讓函數(shù)在"調用""之后的一段時間后生效,應用場景:輸入框(例:在用戶停止輸入的500ms后再處理用戶數(shù)據(jù))。
//函數(shù)節(jié)流 /* * @params {Function} fun 調用函數(shù) * @params {delay} number 延遲時間 */ const throttle = (fun, delay, ...rest) => { let last = null; return () => { const now = + new Date(); if (now - last > delay) { fun(rest); last = now; } } } //實例 const throttleExample = throttle(() => console.log(1), 1000); //調用 throttleExample(); throttleExample(); throttleExample(); //函數(shù)防抖 const debouce = (fun, delay, ...rest) => { let timer = null; return () => { clearTimeout(timer); timer = setTimeout(() => { fun(rest); }, delay); } } //實例 const debouceExample = debouce(() => console.log(1), 1000); //調用 debouceExample(); debouceExample(); debouceExample();快速排序
從數(shù)列中挑出一個元素,稱為"基準"(pivot),
重新排序數(shù)列,所有比基準值小的元素擺放在基準前面,所有比基準值大的元素擺在基準后面(相同的數(shù)可以到任一邊)。在這個分區(qū)結束之后,該基準就處于數(shù)列的中間位置。這個稱為分區(qū)(partition)操作。
遞歸地(recursively)把小于基準值元素的子數(shù)列和大于基準值元素的子數(shù)列排序。
時間復雜度平均情況:O(nlog n) 最快:O(n^{2}) 空間復雜度: O(log n)
var quickSort = function(arr) { console.time("2.快速排序耗時"); if (arr.length <= 1) { return arr; } var pivot = arr.splice(0, 1)[0]; var left = []; var right = []; for (var i = 0; i < arr.length; i++){ if (arr[i] < pivot) { left.push(arr[i]); } else { right.push(arr[i]); } } console.timeEnd("2.快速排序耗時"); return quickSort(left).concat([pivot], quickSort(right)); }; var arr=[3,44,38,5,47,15,36,26,27,2,46,4,19,50,48]; console.log(quickSort(arr));//[2, 3, 4, 5, 15, 19, 26, 27, 36, 38, 44, 46, 47, 48, 50]AMD和CMD的區(qū)別
AMD 是 RequireJS 在推廣過程中對模塊定義的規(guī)范化產出。
CMD 是 SeaJS 在推廣過程中對模塊定義的規(guī)范化產出。
對于依賴的模塊,AMD 是提前執(zhí)行,CMD 是延遲執(zhí)行。不過 RequireJS 從 2.0 開始,也改成可以延遲執(zhí)行(根據(jù)寫法不同,處理方式不同)。CMD 推崇 as lazy as possible.
CMD 推崇依賴就近,AMD 推崇依賴前置。
AMD 的 API 默認是一個當多個用,CMD 的 API 嚴格區(qū)分,推崇職責單一。比如 AMD 里,require 分全局 require 和局部 require,都叫 require。CMD 里,沒有全局 require,而是根據(jù)模塊系統(tǒng)的完備性,提供 seajs.use 來實現(xiàn)模塊系統(tǒng)的加載啟動。CMD 里,每個 API 都簡單純粹。
JavaScript內存泄露的原因以及如何去手動釋放內存易出現(xiàn)泄露的場景
XMLHttpRequest 泄漏發(fā)生在IE7-8,釋放方法,將XMLHttpRequest實例對象設置為Null;
DOM&BOM等COM對象循環(huán)綁定 泄漏發(fā)生在IE6-8,釋放方法,切斷循環(huán)引用,將對對象的應用設置為Null;
定時器(嚴格上說不能算是泄露,是被閉包持有了,是正常的表現(xiàn)),對于閉包中無用的變量可以使用delete操作符進行釋放;
JavaScript垃圾回收機制
引用計數(shù)
此算法把“對象是否不再需要”簡化定義為“對象有沒有其他對象引用到它”。如果沒有引用指向該對象(零引用),對象將被垃圾回收機制回收。
限制:無法處理循環(huán)引用。在下面的例子中,兩個對象被創(chuàng)建,并互相引用,形成了一個循環(huán)。它們被調用之后不會離開函數(shù)作用域,所以它們已經沒有用了,可以被回收了。然而,引用計數(shù)算法考慮到它們互相都有至少一次引用,所以它們不會被回收。
標記清除
當變量進入環(huán)境時,例如,在函數(shù)中聲明一個變量,就將這個變量標記為“進入環(huán)境”。從邏輯上講,永遠不能釋放進入環(huán)境的變量所占用的內存,因為只要執(zhí)行流進入相應的環(huán)境,就可能會用到它們。而當變量離開環(huán)境時,則將其標記為“離開環(huán)境”。
垃圾回收器在運行的時候會給存儲在內存中的所有變量都加上標記(當然,可以使用任何標記方式)。然后,它會去掉環(huán)境中的變量以及被環(huán)境中的變量引用的變量的標記(閉包)。而在此之后再被加上標記的變量將被視為準備刪除的變量,原因是環(huán)境中的變量已經無法訪問到這些變量了。最后,垃圾回收器完成內存清除工作,銷毀那些帶標記的值并回收它們所占用的內存空間。
柯里化函數(shù)所謂的柯里化函數(shù)簡單的說就是將本來接受多個參數(shù)的函數(shù)變?yōu)橹唤邮芤粋€參數(shù)的函數(shù)??吕锘瘮?shù)的模板和實例如下:
var currying = function(fun) { //格式化arguments var args = Array.prototype.slice.call(arguments, 1); return function() { //收集所有的參數(shù)在同一個數(shù)組中,進行計算 var _args = args.concat(Array.prototype.slice.call(arguments)); return fun.apply(null, _args); }; } var add = currying(function() { var args = Array.prototype.slice.call(arguments); return args.reduce(function(a, b) { return a + b; }); }) add(1, 2, 4) /* * 經典面試題 * 函數(shù)參數(shù)不定回調函數(shù)數(shù)目不定 * 編寫函數(shù)實現(xiàn): * add(1,2,3,4,5)==15 * add(1,2)(3,4)(5)==15 */ function add() { // 第一次執(zhí)行時,定義一個數(shù)組專門用來存儲所有的參數(shù) var _args = [].slice.call(arguments); // 在內部聲明一個函數(shù),利用閉包的特性保存_args并收集所有的參數(shù)值 var adder = function () { var _adder = function() { [].push.apply(_args, [].slice.call(arguments)); return _adder; }; // 利用隱式轉換的特性,當最后執(zhí)行時隱式轉換,并計算最終的值返回 _adder.toString = function () { return _args.reduce(function (a, b) { return a + b; }); } return _adder; } return adder.apply(null, _args); } // 輸出結果,可自由組合的參數(shù) console.log(add(1, 2, 3, 4, 5)); // 15 console.log(add(1, 2, 3, 4)(5)); // 15 console.log(add(1)(2)(3)(4)(5)); // 15Less常用特性
變量(@color = #fff)
混合(Mixin)
內置函數(shù)(顏色,字符串,類型判斷,數(shù)學)
循環(huán)
嵌套
運算
導入(@import)
ES6常用特性變量定義(let和const,可變與不可變,const定義對象的特殊情況)
解構賦值
模板字符串
數(shù)組新API(例:Array.from(),entries(),values(),keys())
箭頭函數(shù)(rest參數(shù),擴展運算符,::綁定this)
Set和Map數(shù)據(jù)結構(set實例成員值唯一存儲key值,map實例存儲鍵值對(key-value))
Promise對象(前端異步解決方案進化史,generator函數(shù),async函數(shù))
Class語法糖(super關鍵字)
react中setState的原理題目:
import React from "react" class App extends React.Component { constructor() { super(); this.state = { value: 0 } } componentDidMount() { this.setState({value: this.state.value + 1}); console.log(this.state.value); this.setState({value: this.state.value + 1}); console.log(this.state.value); this.setState({value: this.state.value + 1}); console.log(this.state.value); setTimeout(() => { this.setState({value: this.state.value + 1}); console.log(this.state.value); this.setState({value: this.state.value + 1}); console.log(this.state.value); }, 0) } }
答案: 0、0、0、2、3;
分析:
當setState方法調用的時候React就會重新調用render方法來重新渲染組件;setState通過一個隊列來更新state,當調用setState方法的時候會將需要更新的state放入這個狀態(tài)隊列中,這個隊列會高效的批量更新state;
源碼地址:enqueueUpdate
function enqueueUpdate(component) { ensureInjected(); //判斷是否處于批量更新模式 if (!batchingStrategy.isBatchingUpdates) { //關鍵!下面的代碼片段是這個方法的源碼 batchingStrategy.batchedUpdates(enqueueUpdate, component); return; } //如果處于批量更新模式,則將這個組件保存在dirtyComponents dirtyComponents.push(component); }
源碼地址:ReactDefaultBatchingStrategy
//batchingStrategy對象 var ReactDefaultBatchingStrategy = { //注意默認為false isBatchingUpdates: false, batchedUpdates: function(callback, a, b, c, d, e) { var alreadyBatchingUpdates = ReactDefaultBatchingStrategy.isBatchingUpdates; ReactDefaultBatchingStrategy.isBatchingUpdates = true; if (alreadyBatchingUpdates) { callback(a, b, c, d, e); } else { //關鍵!??!事務的理解 transaction.perform(callback, null, a, b, c, d, e); } }, };
源碼地址:Transaction
如圖:事務會將所需要執(zhí)行的方法(圖中的anyMethod)使用wrapper封裝起來,再通過perform方法執(zhí)行該方法,但在perform執(zhí)行之前會先執(zhí)行所有wrapper中的initialize方法,perform方法執(zhí)行結束后,再執(zhí)行所有的close方法;
var Transaction = require("./Transaction"); // 我們自己定義的 var MyTransaction = function() { //do something }; Object.assign(MyTransaction.prototype, Transaction.Mixin, { //需要自定義一個getTransactionWrappers對象,獲取所有需要封裝的initialize方法和close方法 getTransactionWrappers: function() { return [{ initialize: function() { console.log("before method perform"); }, close: function() { console.log("after method perform"); } }]; }; }); //實例化一個transaction var transaction = new MyTransaction(); //需要調用的方法 var testMethod = function() { console.log("test"); } transaction.perform(testMethod); //before method perform //test //after method perform
理解題目的關鍵是,整個組件渲染到DOM中的過程就已經處于一次大的事務中了,因此在componentDidMount方法中調用setState的時候 ReactDefaultBatchingStrategy.isBatchingUpdates = true;這句代碼已經執(zhí)行過了,所以setState的結果并沒有立即生效,而是扔進了dirtyComponent;因此執(zhí)行三次setState的結果this.state.value的值依然是0,而setTimeout中的兩次setState由于沒有調用過batchedUpdates方法(isBatchingUpdates默認為false),所以setState方法立即生效,第二次setSState同理
XSS與CSRF介紹XSS是一種跨站腳本攻擊,是屬于代碼注入的一種,攻擊者通過將代碼注入網頁中,其他用戶看到會受到影響(代碼內容有請求外部服務器);
CSRF是一種跨站請求偽造,冒充用戶發(fā)起請求,完成一些違背用戶請求的行為(刪帖,改密碼,發(fā)郵件,發(fā)帖等)
防御方法舉例:
對一些關鍵字和特殊字符進行過濾(<>,?,script等),或對用戶輸入內容進行URL編碼(encodeURIComponent);
Cookie不要存放用戶名和密碼,對cookie信息進行MD5等算法散列存放,必要時可以將IP和cookie綁定;
后記時隔三個月,終于迎來了博文的更新,有看到博友在評論留言:
心里很溫暖,這篇不算博文的博文就當是回歸之作吧,接下來的時間會盡量保持在一周一更,實習結束有的是時間了,哈哈哈。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/83052.html
摘要:推薦高性能網站建設指南高性能網站建設進階指南理由在讀完前幾本書之后我們對前端的性能和自己的代碼的效率已經達到相當?shù)母叨攘耍缓笪覀冊诮佑|一些前端工程師的一些精髓。 WEB前端研發(fā)工程師,在國內算是一個朝陽職業(yè),這個領域沒有學校的正規(guī)教育,大多數(shù)人都是靠自己自學成才。本文主要介紹自己從事web開發(fā)以來(從大二至今)看過的書籍和自己的成長過程,目的是給想了解JavaScript或者是剛...
摘要:推薦高性能網站建設指南高性能網站建設進階指南理由在讀完前幾本書之后我們對前端的性能和自己的代碼的效率已經達到相當?shù)母叨攘?,然后我們在接觸一些前端工程師的一些精髓。 WEB前端研發(fā)工程師,在國內算是一個朝陽職業(yè),這個領域沒有學校的正規(guī)教育,大多數(shù)人都是靠自己自學成才。本文主要介紹自己從事web開發(fā)以來(從大二至今)看過的書籍和自己的成長過程,目的是給想了解JavaScript或者是剛...
摘要:推薦高性能網站建設指南高性能網站建設進階指南理由在讀完前幾本書之后我們對前端的性能和自己的代碼的效率已經達到相當?shù)母叨攘?,然后我們在接觸一些前端工程師的一些精髓。 WEB前端研發(fā)工程師,在國內算是一個朝陽職業(yè),這個領域沒有學校的正規(guī)教育,大多數(shù)人都是靠自己自學成才。本文主要介紹自己從事web開發(fā)以來(從大二至今)看過的書籍和自己的成長過程,目的是給想了解JavaScript或者是剛...
摘要:業(yè)務和架構不分家,架構是建立在對業(yè)務的理解之上的。主鍵最好保持順序遞增,隨機主鍵會導致聚簇索引樹頻繁分裂,隨機增多,數(shù)據(jù)離散,性能下降。沒有索引的更新,可能會導致全表數(shù)據(jù)都被鎖住。 本博客并非全部原創(chuàng),其實是一個知識的歸納和匯總,里面我引用了很多網上、書上的內容。也給出了相關的鏈接。 本文涉及的知識點比較多,大家可以根據(jù)關鍵字去搜索相關的內容和購買相應的書籍進行系統(tǒng)的學習。不對的地方...
摘要:去年年底因為使用了云存儲和其他方面的原因,計劃的將服務器縮減一個機柜出來。云服務的回源服務器的配置中間漏了一臺,后期給補上了。監(jiān)控遷移完畢之后,除了常規(guī)的業(yè)務代碼,還需要注意圖片資源的回源是否正常服務器壓力是否正常檢查日志是否出現(xiàn)錯誤。 去年年底因為使用了云存儲和其他方面的原因,計劃的將服務器縮減一個機柜出來。這樣今年每月機房的費用可以減少1萬左右。前前后后抽空在弄這個任務,現(xiàn)做個筆記...
閱讀 4627·2021-09-26 09:55
閱讀 1369·2019-12-27 12:16
閱讀 890·2019-08-30 15:56
閱讀 1908·2019-08-30 14:05
閱讀 995·2019-08-30 13:05
閱讀 1271·2019-08-30 10:59
閱讀 1447·2019-08-26 16:19
閱讀 1889·2019-08-26 13:47