国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

讀 zepto 源碼之工具函數

姘擱『 / 2371人閱讀

摘要:在遍歷的時候,還對回調函數的返回值進行判斷,如果回調函數返回,立即中斷遍歷。可以遍歷數組類數組或對象中的元素,根據回調函數的返回值,將返回值組成一個新的數組,并將該數組扁平化后返回,會將及排除。

Zepto 提供了豐富的工具函數,下面來一一解讀。

源碼版本

本文閱讀的源碼為 zepto1.2.0

$.extend

$.extend 方法可以用來擴展目標對象的屬性。目標對象的同名屬性會被源對象的屬性覆蓋。

$.extend 其實調用的是內部方法 extend, 所以我們先看看內部方法 extend 的具體實現。

function extend(target, source, deep) {
        for (key in source)  // 遍歷源對象的屬性值
            if (deep && (isPlainObject(source[key]) || isArray(source[key]))) { // 如果為深度復制,并且源對象的屬性值為純粹對象或者數組
                if (isPlainObject(source[key]) && !isPlainObject(target[key])) // 如果為純粹對象
                    target[key] = {}  // 如果源對象的屬性值為純粹對象,并且目標對象對應的屬性值不為純粹對象,則將目標對象對應的屬性值置為空對象
                if (isArray(source[key]) && !isArray(target[key])) // 如果源對象的屬性值為數組,并且目標對象對應的屬性值不為數組,則將目標對象對應的屬性值置為空數組
                    target[key] = []
                extend(target[key], source[key], deep) // 遞歸調用extend函數
            } else if (source[key] !== undefined) target[key] = source[key]  // 不對undefined值進行復制
    }

extend 的第一個參數 taget 為目標對象, source 為源對象, deep 表示是否為深度復制。當 deeptrue 時為深度復制, false 時為淺復制。

extend 函數用 for···insource 的屬性進行遍歷

如果 deepfalse 時,只進行淺復制,將 source 中不為 undefined 的值賦值到 target 對應的屬性中(注意,這里用的是 !==,不是 != ,所以只排除嚴格為 undefined 的值,不包含 null )。如果 source 對應的屬性值為對象或者數組,會保持該對象或數組的引用。

如果 deeptrue ,并且 source 的屬性值為純粹對象或者數組時

3.1. 如果 source 的屬性為純粹對象,并且 target 對應的屬性不為純粹對象時,將 target 的對應屬性設置為空對象

3.2. 如果 source 的屬性為數組,并且 target 對應屬性不為數組時,將 target 的對應屬性設置為空數組

3.3. 將 sourcetarget 對應的屬性及 deep 作為參數,遞歸調用 extend 函數,以實現深度復制。

現在,再看看 $.extend 的具體實現

$.extend = function(target) {
        var deep, args = slice.call(arguments, 1)
        if (typeof target == "boolean") {
            deep = target
            target = args.shift()
        }
        args.forEach(function(arg) { extend(target, arg, deep) })
        return target
    }

在說原理之前,先來看看 $.extend 的調用方式,調用方式如下:

$.extend(target, [source, [source2, ...]])
                  或
$.extend(true, target, [source, ...])

$.extend 中,如果不需要深度復制,第一個參數可以是目標對象 target, 后面可以有多個 source 源對象。如果需要深度復制,第一個參數為 deep ,第二個參數為 target ,為目標對象,后面可以有多個 source 源對象。

$.extend 函數的參數設計得很優雅,不需要深度復制時,可以不用顯式地將 deep 置為 false。這是如何做到的呢?

$.extend 函數中,定義了一個數組 args,用來接受除第一個參數外的所有參數。

然后判斷第一個參數 target 是否為布爾值,如果為布爾值,表示第一個參數為 deep ,那么第二個才為目標對象,因此需要重新為 target 賦值為 args.shift()

最后就比較簡單了,循環源對象數組 args, 分別調用 extend 方法,實現對目標對象的擴展。

$.each

$.each 用來遍歷數組或者對象,源碼如下:

$.each = function(elements, callback) {
        var i, key
        if (likeArray(elements)) {  // 類數組
            for (i = 0; i < elements.length; i++)
                if (callback.call(elements[i], i, elements[i]) === false) return elements
        } else { // 對象
            for (key in elements)
                if (callback.call(elements[key], key, elements[key]) === false) return elements
        }

        return elements
    }

先來看看調用方式:$.each(collection, function(index, item){ ... })

$.each 接收兩個參數,第一個參數 elements 為需要遍歷的數組或者對象,第二個 callback 為回調函數。

如果 elements 為數組,用 for 循環,調用 callback ,并且將數組索引 index 和元素值 item 傳給回調函數作為參數;如果為對象,用 for···in 遍歷屬性值,并且將屬性 key 及屬性值傳給回調函數作為參數。

注意回調函數調用了 call 方法,call 的第一個參數為當前元素值或當前屬性值,所以回調函數的上下文變成了當前元素值或屬性值,也就是說回調函數中的 this 指向的是 item 。這在dom集合的遍歷中相當有用。

在遍歷的時候,還對回調函數的返回值進行判斷,如果回調函數返回 falseif (callback.call(elements[i], i, elements[i]) === false) ) ,立即中斷遍歷。

$.each 調用結束后,會將遍歷的數組或對象( elements )返回。

$.map

可以遍歷數組(類數組)或對象中的元素,根據回調函數的返回值,將返回值組成一個新的數組,并將該數組扁平化后返回,會將 nullundefined 排除。

$.map = function(elements, callback) {
        var value, values = [],
            i, key
        if (likeArray(elements))
            for (i = 0; i < elements.length; i++) {
                value = callback(elements[i], i)
                if (value != null) values.push(value)
            }
        else
            for (key in elements) {
                value = callback(elements[key], key)
                if (value != null) values.push(value)
            }
        return flatten(values)
    }

先來看看調用方式: $.map(collection, function(item, index){ ... })

elements 為類數組或者對象。callback 為回調函數。當為類數組時,用 for 循環,當為對象時,用 for···in 循環。并且將對應的元素(屬性值)及索引(屬性名)傳遞給回調函數,如果回調函數的返回值不為 null 或者 undefined ,則將返回值存入新數組中,最后將新數組扁平化后返回。

$.camelCase

該方法是將字符串轉換成駝峰式的字符串

$.camelCase = camelize

$.camelCase 調用的是內部方法 camelize ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述,本篇文章就不再展開。

$.contains

用來檢查給定的父節點中是否包含有給定的子節點,源碼如下:

$.contains = document.documentElement.contains ?
        function(parent, node) {
            return parent !== node && parent.contains(node)
        } :
        function(parent, node) {
            while (node && (node = node.parentNode))
                if (node === parent) return true
            return false
        }

先來看看調用:$.contains(parent, node)

參數 parent 為父子點,node 為子節點。

$.contains 的主體是一個三元表達式,返回的是一個匿名函數。三元表達式的條件是 document.documentElement.contains, 用來檢測瀏覽器是否支持 contains 方法,如果支持,則直接調用 contains 方法,并且將 parentnode 為同一個元素的情況排除。

否則,返回另一外匿名函數。該函數會一直向上尋找 node 元素的父元素,如果能找到跟 parent 相等的父元素,則返回 true, 否則返回 false

$.grep

該函數其實就是數組的 filter 函數

  $.grep = function(elements, callback) {
       return filter.call(elements, callback)
   }

從源碼中也可以看出,$.grep 調用的就是數組方法 filter

$.inArray

返回指定元素在數組中的索引值

 $.inArray = function(elem, array, i) {
        return emptyArray.indexOf.call(array, elem, i)
    }

先來看看調用 $.inArray(element, array, [fromIndex])

第一個參數 element 為指定的元素,第二個參數為 array 為數組, 第三個參數 fromIndex 為可選參數,表示從哪個索引值開始向后查找。

$.inArray 其實調用的是數組的 indexOf 方法,所以傳遞的參數跟 indexOf 方法一致。

$.isArray

判斷是否為數組

$.isArray = isArray

$.isArray 調用的是內部方法 isArray ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述。

$.isFunction

判讀是否為函數

$.isFunction = isFunction

$.isFunction 調用的是內部方法 isFunction ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述。

$.isNumeric

是否為數值

$.isNumeric = function(val) {
        var num = Number(val), // 將參數轉換為Number類型
            type = typeof val
        return val != null && 
          type != "boolean" &&
            (type != "string" || val.length) &&
          !isNaN(num) &&
          isFinite(num) 
          || false
    }

判斷是否為數值,需要滿足以下條件

不為 null

不為布爾值

不為NaN(當傳進來的參數不為數值或如"123"這樣形式的字符串時,都會轉換成NaN)

為有限數值

當傳進來的參數為字符串的形式,如"123" 時,會用到下面這個條件來確保字符串為數字的形式,而不是如 123abc 這樣的形式。(type != "string" || val.length) && !isNaN(num) 。這個條件的包含邏輯如下:如果為字符串類型,并且為字符串的長度大于零,并且轉換成數組后的結果不為NaN,則斷定為數值。(因為 Number("") 的值為 0

$.isPlainObject

是否為純粹對象,即以 {} 常量或 new Object() 創建的對象

$.isPlainObject = isPlainObject

$.isPlainObject 調用的是內部方法 isPlainObject ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述。

$.isWindow

是否為瀏覽器的 window 對象

$.isWindow = isWindow

$.isWindow 調用的是內部方法 isWindow ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述。

$.noop

空函數

$.noop = function() {}

這個在需要傳遞回調函數作為參數,但是又不想在回調函數中做任何事情的時候會非常有用,這時,只需要傳遞一個空函數即可。

$.parseJSON

將標準JSON格式的字符串解釋成JSON

if (window.JSON) $.parseJSON = JSON.parse

其實就是調用原生的 JSON.parse, 并且在瀏覽器不支持的情況下,zepto 還不提供這個方法。

$.trim

刪除字符串頭尾的空格

$.trim = function(str) {
  return str == null ? "" : String.prototype.trim.call(str)
}

如果參數為 null 或者 undefined ,則直接返回空字符串,否則調用字符串原生的 trim 方法去除頭尾的空格。

$.type

類型檢測

$.type = type

$.type 調用的是內部方法 type ,該方法在前一篇文章《讀Zepto源碼之內部方法》中已有闡述。

能檢測的類型有 "Boolean Number String Function Array Date RegExp Object Error"

系列文章

讀Zepto源碼之代碼結構

讀 Zepto 源碼之內部方法

參考

Zepto中文文檔

Node.contains()

Array.prototype.indexOf()

String.prototype.trim()

最后,所有文章都會同步發送到微信公眾號上,歡迎關注,歡迎提意見:

作者:對角另一面

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/82770.html

相關文章

  • Zepto源碼Form模塊

    摘要:模塊處理的是表單提交。表單提交包含兩部分,一部分是格式化表單數據,另一部分是觸發事件,提交表單。最終返回的結果是一個數組,每個數組項為包含和屬性的對象。否則手動綁定事件,如果沒有阻止瀏覽器的默認事件,則在第一個表單上觸發,提交表單。 Form 模塊處理的是表單提交。表單提交包含兩部分,一部分是格式化表單數據,另一部分是觸發 submit 事件,提交表單。 讀 Zepto 源碼系列文章已...

    陳江龍 評論0 收藏0
  • Zepto源碼集合操作

    摘要:調用來獲取符合條件的集合元素,這在上篇文章讀源碼之神奇的已經有詳細的論述。然后調用方法來合并兩個集合,用內部方法來過濾掉重復的項,方法在讀源碼之內部方法已經有論述。最后也是返回一個集合。 接下來幾個篇章,都會解讀 zepto 中的跟 dom 相關的方法,也即源碼 $.fn 對象中的方法。 讀Zepto源碼系列文章已經放到了github上,歡迎star: reading-zepto 源碼...

    pepperwang 評論0 收藏0
  • Zepto源碼IOS3模塊

    摘要:用法與參數要理解這段代碼,先來看一下的用法和參數用法參數回調函數,有如下參數上一個回調函數返回的值或者是初始值當前值當前值在數組中的索引調用的數組初始值,如果沒有提供,則為數組的第一項。接下來,檢測回調函數是否為,如果不是,拋出類型錯誤。 IOS3 模塊是針對 IOS 的兼容模塊,實現了兩個常用方法的兼容,這兩個方法分別是 trim 和 reduce 。 讀 Zepto 源碼系列文章...

    lavnFan 評論0 收藏0
  • Zepto源碼Stack模塊

    摘要:讀源碼系列文章已經放到了上,歡迎源碼版本本文閱讀的源碼為改寫原有的方法模塊改寫了以上這些方法,這些方法在調用的時候,會為返回的結果添加的屬性,用來保存原來的集合。方法的分析可以看讀源碼之模塊。 Stack 模塊為 Zepto 添加了 addSelf 和 end 方法。 讀 Zepto 源碼系列文章已經放到了github上,歡迎star: reading-zepto 源碼版本 本文閱讀的...

    crossea 評論0 收藏0
  • Zepto源碼Selector模塊

    摘要:如果偽類的參數不可以用轉換,則參數為字符串,用正則將字符串前后的或去掉,再賦值給最后執行回調,將解釋出來的參數傳入回調函數中,將執行結果返回。重寫的方法,改過的調用的是方法,在回調函數中處理大部分邏輯。 Selector 模塊是對 Zepto 選擇器的擴展,使得 Zepto 選擇器也可以支持部分 CSS3 選擇器和 eq 等 Zepto 定義的選擇器。 在閱讀本篇文章之前,最好先閱讀《...

    Jioby 評論0 收藏0

發表評論

0條評論

姘擱『

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<