摘要:調(diào)用函數(shù),判斷返回的類型字符串,就知道是什么數(shù)據(jù)類型了判斷是否為瀏覽器的對象要為對象首先要滿足的條件是不能為或者,并且為自身的引用。必須存在中必須存在屬性返回的值調(diào)用函數(shù),返回的數(shù)據(jù)類型。
數(shù)組方法 定義
var emptyArray = [] concat = emptyArray.concat filter = emptyArray.filter slice = emptyArray.slice
zepto 一開始就定義了一個空數(shù)組 emptyArray,定義這個空數(shù)組是為了取得數(shù)組的 concat、filter、slice 方法
compactfunction compact(array) { return filter.call(array, function(item) { return item != null }) }
刪除數(shù)組中的 null 和 undefined
這里用的是數(shù)組的 filter 方法,過濾出 item != null 的元素,組成新的數(shù)組。這里刪除掉 null 很容易理解,為什么還可以刪除 undefined 呢?這是因為這里用了 != ,而不是用 !== ,用 != 時, null 各 undefined 都會先轉(zhuǎn)換成 false 再進行比較。
關(guān)于 null 和 undefined 推薦看看這篇文章: undefined與null的區(qū)別
flattenfunction flatten(array) { return array.length > 0 ? $.fn.concat.apply([], array) : array }
將數(shù)組扁平化,例如將數(shù)組 [1,[2,3],[4,5],6,[7,[89]] 變成 [1,2,3,4,5,6,7,[8,9]] ,這個方法只能展開一層,多層嵌套也只能展開一層。
這里,我們先把 $.fn.concat 等價于數(shù)組的原生方法 concat,后面的章節(jié)也會分析 $.fn.concat 的。
這里比較巧妙的是利用了 apply ,apply 會將 array 中的 item 當(dāng)成參數(shù),concat.apply([], [1,2,3,[4,5]]) 相當(dāng)于 [].concat(1,2,3,[4,5]),這樣數(shù)組就扁平化了。
uniquniq = function(array) { return filter.call(array, function(item, idx) { return array.indexOf(item) == idx }) }
數(shù)組去重。
數(shù)組去重的原理是檢測 item 在數(shù)組中第一次出現(xiàn)的位置是否和 item 所處的位置相等,如果不相等,則證明不是第一次出現(xiàn),將其過濾掉。
字符串方法 camelizecamelize = function(str) { return str.replace(/-+(.)?/g, function(match, chr) { return chr ? chr.toUpperCase() : "" }) }
將 word-word 的形式的字符串轉(zhuǎn)換成 wordWord 的形式, - 可以為一個或多個。
正則表達式匹配了一個或多個 - ,捕獲組是捕獲 - 號后的第一個字母,并將字母變成大寫。
dasherizefunction dasherize(str) { return str.replace(/::/g, "/") .replace(/([A-Z]+)([A-Z][a-z])/g, "$1_$2") .replace(/([a-zd])([A-Z])/g, "$1_$2") .replace(/_/g, "-") .toLowerCase() }
將駝峰式的寫法轉(zhuǎn)換成連字符 - 的寫法。
例如 a = A6DExample::Before
第一個正則表達式是將字符串中的 :: 替換成 / 。a 變成 A6DExample/Before
第二個正則是在出現(xiàn)一次或多次大寫字母和出現(xiàn)一次大寫字母和連續(xù)一次或多次小寫字母之間加入 _。a 變成 A6D_Example/Before
第三個正則是將出現(xiàn)一次小寫字母或數(shù)字和出現(xiàn)一次大寫字母之間加上 _。a 變成A6_D_Example/Before
第四個正則表達式是將 _ 替換成 -。a 變成A6-D-Example/Before
最后是將所有的大寫字母轉(zhuǎn)換成小寫字母。a 變成 a6-d-example/before
我對正則不太熟悉,正則解釋部分參考自:zepto源碼--compact、flatten、camelize、dasherize、uniq--學(xué)習(xí)筆記
數(shù)據(jù)類型檢測 定義class2type = {}, toString = class2type.toString, // Populate the class2type map $.each("Boolean Number String Function Array Date RegExp Object Error".split(" "), function(i, name) { class2type["[object " + name + "]"] = name.toLowerCase() })
$.each 函數(shù)后面的文章會講到,這段代碼是將基本類型掛到 class2type 對象上。class2type 將會是如下的形式:
class2type = { "[object Boolean]": "boolean", "[object Number]": "number" ... }type
function type(obj) { return obj == null ? String(obj) : class2type[toString.call(obj)] || "object" }
type 函數(shù)返回的是數(shù)據(jù)的類型。
如果 obj == null ,也就是 null 和 undefined,返回的是字符串 null 或 undefined
否則調(diào)用 Object.prototype.toString (toString = class2type.toString)方法,將返回的結(jié)果作為 class2type 的 key 取值。Object.prototype.toString 對不同的數(shù)據(jù)類型會返回形如 [object Boolean] 的結(jié)果。
如果都不是以上情況,默認(rèn)返回 object 類型。
isFunction & isObjectfunction isFunction(value) { return type(value) === "function" } function isObject(obj) { return type(obj) == "object" }
調(diào)用 type 函數(shù),判斷返回的類型字符串,就知道是什么數(shù)據(jù)類型了
isWindowfunction isWindow(obj) { return obj != null && obj == obj.window }
判斷是否為瀏覽器的 window 對象
要為 window 對象首先要滿足的條件是不能為 null 或者 undefined, 并且 obj.window 為自身的引用。
isDocumentfunction isDocument(obj) { return obj != null && obj.nodeType == obj.DOCUMENT_NODE }
判斷是否為 document 對象
節(jié)點上有 nodeType 屬性,每個屬性值都有對應(yīng)的常量。document 的 nodeType 值為 9 ,常量為 DOCUMENT_NODE。
具體見:MDN文檔:Node.nodeType
isPlainObjectfunction isPlainObject(obj) { return isObject(obj) && !isWindow(obj) && Object.getPrototypeof(obj) == Object.prototype }
判斷是否為純粹的對象
純粹對象首先必須是對象 isObject(obj)
并且不是 window 對象 !isWindow(obj)
并且原型要和 Object 的原型相等
isArrayisArray = Array.isArray || function(object) { return object instanceof Array}
這個方法來用判斷是否為數(shù)組類型。
如果瀏覽器支持?jǐn)?shù)組的 isArray 原生方法,就采用原生方法,否則檢測數(shù)據(jù)是否為 Array 的實例。
我們都知道,instanceof 的檢測的原理是查找實例的 prototype 是否在構(gòu)造函數(shù)的原型鏈上,如果在,則返回 true。 所以用 instanceof 可能會得到不太準(zhǔn)確的結(jié)果。例如:
index.html
Document
frame.html
Document frame page
由于 iframe 是在獨立的環(huán)境中運行的,所以 fdata instanceof Array 返回的 false 。
在 MDN 上看到,可以用這樣的 ployfill 來使用 isArray
if (!Array.isArray) { Array.isArray = function(arg) { return Object.prototype.toString.call(arg) === "[object Array]" } }
也就是說,isArray 可以修改成這樣:
isArray = Array.isArray || function(object) { return Object.prototype.toString.call(object) === "[object Array]"}
為什么 zepto 不這樣寫呢?知道的可以留言告知下。
likeArrayfunction likeArray(obj) { var length = !!obj && // obj必須存在 "length" in obj && // obj 中必須存在 length 屬性 obj.length, // 返回 length的值 type = $.type(obj) // 調(diào)用 type 函數(shù),返回 obj 的數(shù)據(jù)類型。這里我有點不太明白,為什么要覆蓋掉上面定義的 type 函數(shù)呢?再定義多一個變量,直接調(diào)用 type 函數(shù)不好嗎? return "function" != type && // 不為function類型 !isWindow(obj) && // 并且不為window類型 ( "array" == type || length === 0 || // 如果為 array 類型或者length 的值為 0,返回true (typeof length == "number" && length > 0 && (length - 1) in obj) // 或者 length 為數(shù)字,并且 length的值大于零,并且 length - 1 為 obj 的 key ) }
判斷是否為數(shù)據(jù)是否為類數(shù)組。
類數(shù)組的形式如下:
likeArrayData = { "0": 0, "1": 1, "2": 2 length: 3 }
可以看到,類數(shù)組都有 length 屬性,并且 key 為按0,1,2,3 順序的數(shù)字。
代碼已經(jīng)有注釋了,這里再簡單總結(jié)下
首先將 function類型和 window 對象排除
再將 type 為 array 和 length === 0 的認(rèn)為是類數(shù)組。type 為 array 比較容易理解,length === 0 其實就是將其看作為空數(shù)組。
最后一種情況必須要滿足三個條件:
length 必須為數(shù)字
length 必須大于 0 ,表示有元素存在于類數(shù)組中
key length - 1 必須存在于 obj 中。我們都知道,數(shù)組最后的 index 值為 length -1 ,這里也是檢查最后一個 key 是否存在。
系列文章讀Zepto源碼之代碼結(jié)構(gòu)
參考MDN文檔:Array.isArray()
MDN文檔:Function.prototype.apply()
MDN文檔:Node.nodeType
undefined與null的區(qū)別
zepto源碼--compact、flatten、camelize、dasherize、uniq--學(xué)習(xí)筆記
最后,所有文章都會同步發(fā)送到微信公眾號上,歡迎關(guān)注,歡迎提意見:
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/82432.html
摘要:所以模塊依賴于模塊,在引入前必須引入模塊。原有的方法分析見讀源碼之樣式操作方法首先調(diào)用原有的方法,將元素顯示出來,這是實現(xiàn)動畫的基本條件。如果沒有傳遞,或者為值,則表示不需要動畫,調(diào)用原有的方法即可。 fx 模塊提供了 animate 動畫方法,fx_methods 利用 animate 方法,提供一些常用的動畫方法。所以 fx_methods 模塊依賴于 fx 模塊,在引入 fx_m...
摘要:調(diào)用來獲取符合條件的集合元素,這在上篇文章讀源碼之神奇的已經(jīng)有詳細(xì)的論述。然后調(diào)用方法來合并兩個集合,用內(nèi)部方法來過濾掉重復(fù)的項,方法在讀源碼之內(nèi)部方法已經(jīng)有論述。最后也是返回一個集合。 接下來幾個篇章,都會解讀 zepto 中的跟 dom 相關(guān)的方法,也即源碼 $.fn 對象中的方法。 讀Zepto源碼系列文章已經(jīng)放到了github上,歡迎star: reading-zepto 源碼...
摘要:模塊處理的是表單提交。表單提交包含兩部分,一部分是格式化表單數(shù)據(jù),另一部分是觸發(fā)事件,提交表單。最終返回的結(jié)果是一個數(shù)組,每個數(shù)組項為包含和屬性的對象。否則手動綁定事件,如果沒有阻止瀏覽器的默認(rèn)事件,則在第一個表單上觸發(fā),提交表單。 Form 模塊處理的是表單提交。表單提交包含兩部分,一部分是格式化表單數(shù)據(jù),另一部分是觸發(fā) submit 事件,提交表單。 讀 Zepto 源碼系列文章已...
摘要:讀源碼系列文章已經(jīng)放到了上,歡迎源碼版本本文閱讀的源碼為改寫原有的方法模塊改寫了以上這些方法,這些方法在調(diào)用的時候,會為返回的結(jié)果添加的屬性,用來保存原來的集合。方法的分析可以看讀源碼之模塊。 Stack 模塊為 Zepto 添加了 addSelf 和 end 方法。 讀 Zepto 源碼系列文章已經(jīng)放到了github上,歡迎star: reading-zepto 源碼版本 本文閱讀的...
摘要:的模塊用來獲取節(jié)點中的屬性的數(shù)據(jù),和儲存跟相關(guān)的數(shù)據(jù)。獲取節(jié)點指定的緩存值。如果存在,則刪除指定的數(shù)據(jù),否則將緩存的數(shù)據(jù)全部刪除。為所有下級節(jié)點,如果為方法,則節(jié)點自身也是要被移除的,所以需要將自身也加入到節(jié)點中。 Zepto 的 Data 模塊用來獲取 DOM 節(jié)點中的 data-* 屬性的數(shù)據(jù),和儲存跟 DOM 相關(guān)的數(shù)據(jù)。 讀 Zepto 源碼系列文章已經(jīng)放到了github上,歡...
摘要:在遍歷的時候,還對回調(diào)函數(shù)的返回值進行判斷,如果回調(diào)函數(shù)返回,立即中斷遍歷??梢员闅v數(shù)組類數(shù)組或?qū)ο笾械脑?,根?jù)回調(diào)函數(shù)的返回值,將返回值組成一個新的數(shù)組,并將該數(shù)組扁平化后返回,會將及排除。 Zepto 提供了豐富的工具函數(shù),下面來一一解讀。 源碼版本 本文閱讀的源碼為 zepto1.2.0 $.extend $.extend 方法可以用來擴展目標(biāo)對象的屬性。目標(biāo)對象的同名屬性會被...
閱讀 25649·2021-09-29 09:41
閱讀 4814·2021-09-10 11:20
閱讀 1932·2021-09-09 09:32
閱讀 1897·2019-08-30 15:44
閱讀 3205·2019-08-29 17:13
閱讀 2816·2019-08-29 14:14
閱讀 2071·2019-08-29 14:11
閱讀 3235·2019-08-29 12:36