Rumah >hujung hadapan web >tutorial js >Perbincangan ringkas tentang teknik panggilan, gunakan dan bind_javascript dalam javascript

Perbincangan ringkas tentang teknik panggilan, gunakan dan bind_javascript dalam javascript

WBOY
WBOYasal
2016-05-16 15:11:451374semak imbas

Dalam JavaScript, panggil, gunakan dan ikat adalah tiga kaedah yang disertakan dengan objek Fungsi Fungsi utama ketiga-tiga kaedah ini adalah untuk menukar penunjuk ini dalam fungsi, untuk mencapai kesan `mengambil alih bunga. dan mencabut pokok'. Artikel ini akan menerangkan ketiga-tiga kaedah ini secara terperinci dan menyenaraikan beberapa senario aplikasi klasik.

panggil(iniArgs [,args...])

Kaedah ini boleh menghantar parameter thisArgs dan senarai parameter thisArgs menentukan pemanggil fungsi pada masa jalan, iaitu objek ini dalam fungsi dan senarai parameter akan dihantar ke dalam fungsi panggilan. Nilai thisArgs mempunyai empat situasi berikut:

(1) Jangan lulus, atau lulus null, undefined, ini dalam fungsi menghala ke objek tetingkap

(2) Lulus nama fungsi fungsi lain, dan ini dalam fungsi menunjukkan rujukan kepada fungsi ini

(3) Lulus jenis asas seperti rentetan, nombor atau jenis Boolean Ini dalam fungsi menghala ke objek pembungkusan yang sepadan, seperti String, Number, Boolean

(4) Lulus objek, dan ini dalam fungsi menghala ke objek ini

function a(){
 console.log(this); //输出函数a中的this对象
}
function b(){} //定义函数b

var obj = {name:'onepixel'}; //定义对象obj

a.call(); //window
a.call(null); //window
a.call(undefined);//window
a.call(1); //Number
a.call(''); //String
a.call(true); //Boolean
a.call(b);// function b(){}
a.call(obj); //Object

Ini adalah fungsi teras panggilan Ia membolehkan anda memanggil kaedah yang tidak ditentukan pada objek, dan kaedah ini boleh mengakses sifat dalam objek Bagi faedah melakukan ini, saya akan membincangkannya kemudian lihat contoh mudah dahulu:

var a = {

 name:'onepixel', //定义a的属性

 say:function(){ //定义a的方法
  console.log("Hi,I'm function a!");
 }
};

function b(name){
 console.log("Post params: "+ name);
 console.log("I'm "+ this.name);
 this.say();
}

b.call(a,'test');
>>
Post params: test
I'm onepixel
I'm function a!

Apabila b.call dilaksanakan, rentetan `test` dihantar ke fungsi b sebagai parameter Disebabkan oleh fungsi panggilan, ini dalam fungsi b menghala ke objek a, jadi ia bersamaan dengan memanggil fungsi b pada objek. a. Sebenarnya, b tidak ditakrifkan dalam a.

mohon(thisArgs[,args[]])

Satu-satunya perbezaan antara memohon dan panggilan ialah cara parameter kedua dilalui Parameter kedua permohonan mestilah tatasusunan, manakala panggilan membenarkan senarai parameter diluluskan. Perlu diberi perhatian anda bahawa walaupun aplikasi menerima tatasusunan parameter, apabila ia dihantar ke fungsi panggilan, ia dihantar dalam bentuk senarai parameter Mari lihat contoh mudah:

function b(x,y,z){
 console.log(x,y,z);
}

b.apply(null,[1,2,3]); // 1 2 3

Ciri permohonan ini sangat penting, kami akan menyebut ciri ini dalam senario aplikasi berikut.

ikat(thisArgs [,args...])

bind ialah kaedah baharu dalam ES5 Parameternya serupa dengan panggilan, tetapi ia berbeza dengan ketara daripada panggilan/pakai Iaitu, memanggil atau memohon akan secara automatik melaksanakan fungsi yang sepadan, manakala bind tidak akan melaksanakan fungsi yang sepadan. . Fungsi hanya mengembalikan rujukan kepada fungsi. Sekilas pandang, bind nampaknya ketinggalan di belakang panggilan/apply, jadi mengapa ES5 memperkenalkan bind?

Malah, tujuan sebenar memperkenalkan bind dalam ES5 adalah untuk mengimbangi kekurangan panggilan/apply Memandangkan panggilan/apply akan secara automatik melaksanakan fungsi sasaran, ia tidak boleh digunakan dalam fungsi pengikatan acara kerana pengikatan acara fungsi tidak perlu Kami melaksanakannya secara manual, ia dilaksanakan secara automatik oleh JS secara dalaman apabila peristiwa dicetuskan. Bind tidak akan melaksanakan fungsi sasaran secara automatik semasa menukar fungsi ini, jadi ia boleh menyelesaikan masalah di atas dengan sempurna Anda boleh memahaminya dengan melihat contoh:

var obj = {name:'onepixel'};

/**
 * 给document添加click事件监听,并绑定onClick函数
 * 通过bind方法设置onClick的this为obj,并传递参数p1,p2
 */
document.addEventListener('click',onClick.bind(obj,'p1','p2'),false);

//当点击网页时触发并执行
function onClick(a,b){
 console.log(
   this.name, //onepixel
   a, //p1
   b //p2
 )
}

Apabila halaman web diklik, onClick dicetuskan dan dilaksanakan, dan onepixel p1 p2 dikeluarkan, menunjukkan bahawa ini dalam onClick telah ditukar dengan bind menjadi objek obj Untuk mempunyai pemahaman yang mendalam tentang bind. mari kita lihat pelaksanaan polyfill bind:

if (!Function.prototype.bind) {
 Function.prototype.bind = function (oThis) {
  var aArgs = Array.prototype.slice.call(arguments, 1),
   fToBind = this, //this在这里指向的是目标函数
   fBound = function () {
    return fToBind.apply(
     //如果外部执行var obj = new fBound(),则将obj作为最终的this,放弃使用oThis
     this instanceof fToBind
       ? this //此时的this就是new出的obj
       : oThis || this, //如果传递的oThis无效,就将fBound的调用者作为this

     //将通过bind传递的参数和调用时传递的参数进行合并,并作为最终的参数传递
     aArgs.concat(Array.prototype.slice.call(arguments)));
   };

  //将目标函数的原型对象拷贝到新函数中,因为目标函数有可能被当作构造函数使用
  fBound.prototype = this.prototype;

  //返回fBond的引用,由外部按需调用
  return fBound;
 };
}

Senario permohonan satu: Warisan

Seperti yang kita semua tahu, JavaScript tidak mempunyai kata kunci lanjutan dalam bahasa peringkat tinggi seperti Java dan C#, jadi tiada konsep pewarisan dalam JS Jika pewarisan diperlukan, panggil dan gunakan boleh mencapai fungsi ini :

function Animal(name,weight){
 this.name = name;
 this.weight = weight;
}

function Cat(){
 Animal.call(this,'cat','50');
 //Animal.apply(this,['cat','50']);

 this.say = function(){
  console.log("I am " + this.name+",my weight is " + this.weight);
 }
}

var cat = new Cat();
cat.say();//I am cat,my weight is 50

当通过new运算符产生了cat时,Cat中的this就指向了cat对象,而继承的关键是在于Cat中执行了Animal.call(this,'cat','50') 这句话,在call中将this作为thisArgs参数传递,于是Animal方法中的this就指向了Cat中的this,而cat中的this指向的是cat对象,所以Animal中的this指向的就是cat对象,在Animal中定义了name和weight属性,就相当于在cat中定义了这些属性,因此cat对象便拥有了Animal中定义的属性,从而达到了继承的目的。 

应用场景二:移花接木

在讲下面的内容之前,我们首先来认识一下JavaScript中的一个非标准专业术语:ArrayLike(类数组/伪数组)

ArrayLike 对象即拥有数组的一部分行为,在DOM中早已表现出来,而jQuery的崛起让ArrayLike在JavaScript中大放异彩。ArrayLike对象的精妙在于它和JS原生的Array类似,但是它是自由构建的,它来自开发者对JavaScript对象的扩展,也就是说:对于它的原型(prototype)我们可以自由定义,而不会污染到JS原生的Array。

ArrayLike对象在JS中被广泛使用,比如DOM中的NodeList, 函数中的arguments都是类数组对象,这些对象像数组一样存储着每一个元素,但它没有操作数组的方法,而我们可以通过call将数组的某些方法`移接`到ArrayLike对象,从而达到操作其元素的目的。比如我们可以这样遍历函数中的arguments:

function test(){
 //检测arguments是否为Array的实例
 console.log(
   arguments instanceof Array, //false
   Array.isArray(arguments) //false
 );
 //判断arguments是否有forEach方法
 console.log(arguments.forEach); //undefined

 // 将数组中的forEach应用到arguments上
 Array.prototype.forEach.call(arguments,function(item){
  console.log(item); // 1 2 3 4
 });

}
test(1,2,3,4);

除此之外,对于apply而言,我们上面提到了它独有的一个特性,即apply接收的是数组,在传递给调用函数的时候是以参数列表传递的。 这个特性让apply看起来比call 略胜一筹,比如有这样一个场景:给定一个数组[1,3,4,7],然后求数组中的最大元素,而你知道,数组中并没有获取最大值的方法,一般情况下,你需要通过编写代码来实现。而我们知道,Math对象中有一个获取最大值的方法,即Math.max(), max方法需要传递一个参数列表,然后返回这些参数中的最大值。而apply不仅可以将Math对象的max方法应用到其他对象上,还可以将一个数组转化为参数列表传递给max,看代码就能一目了然:

var arr = [2,3,1,5,4];

Math.max.apply(null,arr); // 5

以上便是call和apply比较经典的几个应用场景,熟练掌握这些技巧,并把这些特性应用到你的实际项目中,会使你的代码看起来更加耐人寻味!

Kenyataan:
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn