摘要:作為構(gòu)造函數(shù)使用,綁定到新創(chuàng)建的對象。內(nèi)部實現(xiàn)類和類的繼承構(gòu)造函數(shù)構(gòu)造函數(shù)調(diào)用父類構(gòu)造函數(shù)參考請盡可能詳盡的解釋的工作原理的原理簡單來說通過對象來向服務器發(fā)異步請求,從服務器獲得數(shù)據(jù),然后用來操作而更新頁面。
1 . 請解釋事件代理 (event delegation)
當需要對很多元素添加事件的時,可以通過將事件添加到它們的父節(jié)點通過委托來觸發(fā)處理函數(shù)。其中利用到了瀏覽器的事件冒泡機制。
var delegate = function(client, clientMethod) { return function() { return clientMethod.apply(client, arguments); } } var agentMethod = delegate (client, clientMethod); agentMethod(); // 獲取父節(jié)點,并為它添加一個click事件 document.getElementById("parent-list").addEventListener("click",function(e) { // 檢查事件源e.targe是否為Li if(e.target && e.target.nodeName.toUpperCase == "LI") { // 真正的處理過程在這里 console.log("List item ",e.target.id.replace("post-")," was clicked!"); } });
2 . 談談瀏覽器的事件冒泡機制
對于事件的捕獲和處理,不同的瀏覽器廠商有不同的處理機制,我們以W3C對DOM2.0定義的標準事件為例
DOM2.0模型將事件處理流程分為三個階段:一、事件捕獲階段,二、事件目標階段,三、事件起泡階段。
事件捕獲:當某個元素觸發(fā)某個事件(如onclick),頂層對象document就會發(fā)出一個事件流,隨著DOM樹的節(jié)點向目標元素節(jié)點流去,直到到達事件真正發(fā)生的目標元素。在這個過程中,事件相應的監(jiān)聽函數(shù)是不會被觸發(fā)的。
事件目標:當?shù)竭_目標元素之后,執(zhí)行目標元素該事件相應的處理函數(shù)。如果沒有綁定監(jiān)聽函數(shù),那就不執(zhí)行。
事件起泡:從目標元素開始,往頂層元素傳播。途中如果有節(jié)點綁定了相應的事件處理函數(shù),這些函數(shù)都會被一次觸發(fā)。如果想阻止事件起泡,可以使用e.stopPropagation()(Firefox)或者e.cancelBubble=true(IE)來組織事件的冒泡傳播。
3 . JavaScript 中 this 是如何工作的。
作為函數(shù)調(diào)用,this 綁定全局對象,瀏覽器環(huán)境全局對象為 window 。
內(nèi)部函數(shù)的 this 也綁定全局對象,應該綁定到其外層函數(shù)對應的對象上,這是 JavaScript的缺陷,用that替換。
作為構(gòu)造函數(shù)使用,this 綁定到新創(chuàng)建的對象。
作為對象方法使用,this 綁定到該對象。
使用apply或call調(diào)用 this 將會被顯式設(shè)置為函數(shù)調(diào)用的第一個參數(shù)。
4 . 談談CommonJs 、AMD 和CMD
CommonJS規(guī)范,一個多帶帶的文件就是一個模塊。每一個模塊都是一個多帶帶的作用域 CommonJS的使用代表:NodeJS
AMD 即Asynchronous Module Definition 異步模塊定義 它是一個在瀏覽器端模塊化開發(fā)的規(guī)范 AMD 是 RequireJS 在推廣過程中對模塊定義的規(guī)范化的產(chǎn)出
CMD 即Common Module Definition 通用模塊定義 其代表為SeaJS
requireJS主要解決兩個問題
多個js文件可能有依賴關(guān)系,被依賴的文件需要早于依賴它的文件加載到瀏覽器
js加載的時候瀏覽器會停止頁面渲染,加載文件越多,頁面失去響應時間越長
CMD和AMD的區(qū)別
AMD推崇依賴前置,在定義模塊的時候就要聲明其依賴的模塊
CMD推崇就近依賴,只有在用到某個模塊的時候再去require
5 . 談談對IIFE的理解
IIFE即Immediately-Invoked Function Expression 立即執(zhí)行函數(shù)表達式
不推薦
(function(){})();
推薦
(function(){}());
在javascript里,括號內(nèi)部不能包含語句,當解析器對代碼進行解釋的時候,先碰到了(),然后碰到function關(guān)鍵字就會自動將()里面的代碼識別為函數(shù)表達式而不是函數(shù)聲明。
知識拓展:
function(){ /* code */ }(); 解釋下該代碼能正確執(zhí)行嗎?
不行,在javascript代碼解釋時,當遇到function關(guān)鍵字時,會默認把它當做是一個函數(shù)聲明,而不是函數(shù)表達式,如果沒有把它顯視地表達成函數(shù)表達式,就報錯了,因為函數(shù)聲明需要一個函數(shù)名,而上面的代碼中函數(shù)沒有函數(shù)名。(以上代碼,也正是在執(zhí)行到第一個左括號(時報錯,因為(前理論上是應該有個函數(shù)名的。)
function foo(){ /* code */ }(); 解釋下該代碼能正確執(zhí)行嗎?
在一個表達式后面加上括號,表示該表達式立即執(zhí)行;而如果是在一個語句后面加上括號,該括號完全和之前的語句無法匹配,而只是一個分組操作符,用來控制運算中的優(yōu)先級(小括號里的先運算)相當于先聲明了一個叫foo的函數(shù),之后進行()內(nèi)的表達式運算,但是()(分組操作符)內(nèi)的表達式不能為空,所以報錯。(以上代碼,也就是執(zhí)行到右括號時,發(fā)現(xiàn)表達式為空,所以報錯)。
6 . .call 和 .apply 的區(qū)別是什么?
foo.call(this, arg1,arg2,arg3) == foo.apply(this, arguments)==this.foo(arg1, arg2, arg3)
call, apply方法區(qū)別是,從第二個參數(shù)起, call方法參數(shù)將依次傳遞給借用的方法作參數(shù), 而apply直接將這些參數(shù)放到一個數(shù)組中再傳遞, 最后借用方法的參數(shù)列表是一樣的.
7 . 請解釋 Function.prototype.bind?
bind() 方法的主要作用就是將函數(shù)綁定至某個對象,bind() 方法會創(chuàng)建一個函數(shù),函數(shù)體內(nèi)this對象的值會被綁定到傳入bind() 函數(shù)的值。
原理
Function.prototype.bind = function(context) { var self = this; // 保存原函數(shù) return function() { // 返回一個新函數(shù) return self.apply(context, arguments); // 執(zhí)行新函數(shù)時,將傳入的上下文context作為新函數(shù)的this } }
用法:
var paint = { color: "red", count: 0, updateCount: function() { this.count++; console.log(this.count); } }; // 事件處理函數(shù)綁定的錯誤方法: document.querySelector("button") .addEventListener("click", paint.updateCount); // paint.updateCount函數(shù)的this指向變成了該DOM對象 // 事件處理函數(shù)綁定的正確方法: document.querySelector("button") .addEventListener("click", paint.updateCount.bind(paint)); // paint.updateCount函數(shù)的this指向變成了paint
8 . 請解釋原型繼承 (prototypal inheritance) 的原理。
當查找一個對象的屬性時,JavaScript 會向上遍歷原型鏈,直到找到給定名稱的屬性為止?!鲎訨avaScript秘密花園
JavaScript中的每個對象,都有一個內(nèi)置的 proto 屬性。這個屬性是編程不可見的(雖然ES6標準中開放了這個屬性,然而瀏覽器對這個屬性的可見性的支持不同),它實際上是對另一個對象或者 null 的引用。
當一個對象需要引用一個屬性時,JavaScript引擎首先會從這個對象自身的屬性表中尋找這個屬性標識,如果找到則進行相應讀寫操作,若沒有在自身的屬性表中找到,則在 proto 屬性引用的對象的屬性表中查找,如此往復,直到找到這個屬性或者 proto 屬性指向 null 為止。
以下代碼展示了JS引擎如何查找屬性:
//__proto__ 是一個不應在你代碼中出現(xiàn)的非正規(guī)的用法,這里僅僅用它來解釋JavaScript原型繼承的工作原理。 function getProperty(obj, prop) { if (obj.hasOwnProperty(prop)) return obj[prop] else if (obj.__proto__ !== null) return getProperty(obj.__proto__, prop) else return undefined }
JS的ECMA規(guī)范只允許我們采用 new 運算符來進行原型繼承
原型繼承
function Point(x, y) { this.x = x; this.y = y; } Point.prototype = { print: function () { console.log(this.x, this.y); } }; var p = new Point(10, 20); p.print(); // 10 20
順便闡述下new 運算符是如何工作的?
創(chuàng)建類的實例。這步是把一個空的對象的 proto 屬性設(shè)置為 F.prototype 。
初始化實例。函數(shù) F 被傳入?yún)?shù)并調(diào)用,關(guān)鍵字 this 被設(shè)定為該實例。
返回實例。
function New (f) { var n = { "__proto__": f.prototype }; /*第一步*/ return function () { f.apply(n, arguments); /*第二步*/ return n; /*第三步*/ }; }
JavaScript中真正的原型繼承
Object.create = function (parent) { function F() {} F.prototype = parent; return new F(); };
使用真正的原型繼承(如 Object.create 以及 __proto__)還是存在以下缺點:
標準性差:__proto__ 不是一個標準用法,甚至是一個不贊成使用的用法。同時原生態(tài)的 Object.create 和道爺寫的原版也不盡相同。
優(yōu)化性差: 不論是原生的還是自定義的 Object.create ,其性能都遠沒有 new 的優(yōu)化程度高,前者要比后者慢高達10倍。
ES6 內(nèi)部實現(xiàn)類和類的繼承
class Parent { constructor(name) { //構(gòu)造函數(shù) this.name = name; } say() { console.log("Hello, " + this.name + "!"); } } class Children extends Parent { constructor(name) { //構(gòu)造函數(shù) super(name); //調(diào)用父類構(gòu)造函數(shù) // ... } say() { console.log("Hello, " + this.name + "! hoo~~"); } }
參考:
http://blog.csdn.net/xujie_03...
http://blog.vjeux.com/2011/ja...
9 . 請盡可能詳盡的解釋 AJAX 的工作原理
Ajax 的原理簡單來說通過 XmlHttpRequest 對象來向服務器發(fā)異步請求,從服務器獲得數(shù)據(jù),然后用 JavaScript來操作 DOM 而更新頁面。 這其中最關(guān)鍵的一步就是從服務器獲得請求數(shù)據(jù)。
不使用ajax工作原理
使用ajax工作原理
10 . javascript中"attribute" 和 "property" 的區(qū)別是什么?
property 和 attribute非常容易混淆,兩個單詞的中文翻譯也都非常相近(property:屬性,attribute:特性),但實際上,二者是不同的東西,屬于不同的范疇。每一個DOM對象都會有它默認的基本屬性,而在創(chuàng)建的時候,它只會創(chuàng)建這些基本屬性,我們在TAG標簽中自定義的屬性是不會直接放到DOM中的。
property是DOM中的屬性,是JavaScript里的對象;
attribute是HTML標簽上的特性,它的值只能夠是字符串;
DOM有其默認的基本屬性,而這些屬性就是所謂的“property”,無論如何,它們都會在初始化的時候再DOM對象上創(chuàng)建。
如果在TAG對這些屬性進行賦值,那么這些值就會作為初始值賦給DOM的同名property。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/83356.html
摘要:獲取的對象范圍方法獲取的是最終應用在元素上的所有屬性對象即使沒有代碼,也會把默認的祖宗八代都顯示出來而只能獲取元素屬性中的樣式。因此對于一個光禿禿的元素,方法返回對象中屬性值如果有就是據(jù)我測試不同環(huán)境結(jié)果可能有差異而就是。 花了很長時間整理的前端面試資源,喜歡請大家不要吝嗇star~ 別只收藏,點個贊,點個star再走哈~ 持續(xù)更新中……,可以關(guān)注下github 項目地址 https:...
摘要:系列種優(yōu)化頁面加載速度的方法隨筆分類中個最重要的技術(shù)點常用整理網(wǎng)頁性能管理詳解離線緩存簡介系列編寫高性能有趣的原生數(shù)組函數(shù)數(shù)據(jù)訪問性能優(yōu)化方案實現(xiàn)的大排序算法一怪對象常用方法函數(shù)收集數(shù)組的操作面向?qū)ο蠛驮屠^承中關(guān)鍵詞的優(yōu)雅解釋淺談系列 H5系列 10種優(yōu)化頁面加載速度的方法 隨筆分類 - HTML5 HTML5中40個最重要的技術(shù)點 常用meta整理 網(wǎng)頁性能管理詳解 HTML5 ...
摘要:系列種優(yōu)化頁面加載速度的方法隨筆分類中個最重要的技術(shù)點常用整理網(wǎng)頁性能管理詳解離線緩存簡介系列編寫高性能有趣的原生數(shù)組函數(shù)數(shù)據(jù)訪問性能優(yōu)化方案實現(xiàn)的大排序算法一怪對象常用方法函數(shù)收集數(shù)組的操作面向?qū)ο蠛驮屠^承中關(guān)鍵詞的優(yōu)雅解釋淺談系列 H5系列 10種優(yōu)化頁面加載速度的方法 隨筆分類 - HTML5 HTML5中40個最重要的技術(shù)點 常用meta整理 網(wǎng)頁性能管理詳解 HTML5 ...
摘要:中使用操作符具體做了哪些事情創(chuàng)建了一個空對象空對象的屬性指向構(gòu)造函數(shù)的屬性執(zhí)行構(gòu)造函數(shù),將的指向前端面試題及答案瀏覽器篇前端面試題及答案篇前端面試題及答案篇前端面試題及答案性能優(yōu)化篇 這篇文章并不是最全的前端面試題(沒有最全,只有更全),只是針對自己面試過程中遇到的一些難題、容易忽略的題做一個簡單的筆記,方便后面有面試需要的小伙伴們借鑒,后續(xù)內(nèi)容會不定時更新,有錯誤之處希望大家不吝指出...
閱讀 3737·2021-11-24 10:23
閱讀 2780·2021-09-06 15:02
閱讀 1284·2021-08-23 09:43
閱讀 2361·2019-08-30 15:44
閱讀 3058·2019-08-30 13:18
閱讀 795·2019-08-23 16:56
閱讀 1753·2019-08-23 16:10
閱讀 551·2019-08-23 15:08