搜尋

首頁  >  問答  >  主體

javascript - js中call方法的疑问。

var pObj = document.getElementsByTagName('p');
console.log( [].slice.call(pObj) );                // 返回数组形式[p,p,p]
console.log( pObj.slice() );                       // Error

我的疑问:
1.两个slice方法中的this是不是都是指向的pObj?
2.正因为typeof pObjobject类型,所以不具备slice方法,正如第二个中的error。但是它又像数组所以可以采用第一种方法,理解的对吗?
3.如果上述两个我理解都对的话,其实slice方法中的this处理的是类数组,现在理解的对吗?
4.请大神一一指正。

PHP中文网PHP中文网2845 天前704

全部回覆(3)我來回復

  • ringa_lee

    ringa_lee2017-04-10 14:45:16

    1.pObj 是一个 HTML 集合,也是一个类数组对象。

    • Array.prototype.slice.call()
    • [].slice.call()

    slice() 可以用来将类数组对象或者集合(比如上面的pObj)转换为数组。只需要把 slice() 方法绑定到要转换的类数组对象或者集合上即可。

    [].slice.call(pObj) 这就是一个转换数组的过程。

    • 参考 - Array.prototype.slice()

    2.pObj.slice() 报错。

    看下面这个图,这个类数组对象(集合)没有 slice() 方法,当然报错。

    3.上述问题可忽略你对 this 的想法。建议重新理解一次 this 关键字的应用。

    回覆
    0
  • ringa_lee

    ringa_lee2017-04-10 14:45:16

    回答问题的时候刚好要出去,所以就截了个图示意了一下,没想到你没懂,哎。

    1. 是。
    2. pObj是object没错,不过准确的说是HTMLCollection对象

      HTMLCollection对象中没有没有slice方法(就是你说的不具备)。我上回就跟你说了,此slice非彼slice。你完全可以自己创建一个slice方法啊。

    3.错误,[1,2,3].slice(0,1)中处理的就是数组,要看你具体处理的对象是什么。[].slice.call(pObj)处理的才是伪数组。

    回覆
    0
  • 高洛峰

    高洛峰2017-04-10 14:45:16

    首先document.getElementsByTagName返回的结果并非array,而是HTMLCollection,可以通过document.getElementsByTagName('p').__proto__查验

    因为HTMLCollection的原型里没有slice方法,所以pObj.slice是undefined。如果给它赋值一个function,那么通过pObj.slice()调用时,this就也会指向pObj (问1)

    var pObj = document.getElementsByTagName('p');
    pObj.slice = function() {console.log(this, this===pObj); return 123;};
    pObj.slice();
    

    至于23,差不多对,本质是浏览器实现的Array.prototype.slice,包括数组的其他很多方法,是不局限于接受原生数组的,比如这里的HTMLCollection,比如document.querySelectorAllNodeList,还有函数里面的arguments

    实际上,chrome里面的Array.prototype.slice其实只看length属性和0,1,..这些属性

    [].slice.call({0:3, get 1() {console.log('get index 1'); return 'asdf';},get length() {console.log('get length'); return 3;}});
    //get length 
    //get index 1 
    //[3, "asdf", undefined × 1]
    

    回覆
    0
  • 取消回覆