Rumah  >  Artikel  >  hujung hadapan web  >  Penjelasan terperinci prototaip dan warisan dalam JavaScript (gambar dan teks)_kemahiran javascript

Penjelasan terperinci prototaip dan warisan dalam JavaScript (gambar dan teks)_kemahiran javascript

WBOY
WBOYasal
2016-05-16 16:41:461019semak imbas

Sila lupakan buat sementara waktu semua yang anda telah pelajari tentang orientasi objek di sini. Hanya situasi perlumbaan yang perlu dipertimbangkan di sini. Ya, ia perlumbaan.

Baru-baru ini saya telah menonton 24 Hours of Le Mans , iaitu acara popular di Perancis. Kereta terpantas dikenali sebagai prototaip Le Mans. Walaupun kereta ini dibuat oleh pengeluar seperti "Audi" atau "Peugeot", ia bukanlah jenis kereta yang anda lihat di jalan atau di lebuh raya. Ia dibina khusus untuk bersaing dalam acara ketahanan berkelajuan tinggi.

Pengilang melaburkan sejumlah besar wang dalam penyelidikan, pembangunan, reka bentuk dan pembuatan kenderaan prototaip ini, dan jurutera sentiasa bekerja keras untuk menyempurnakan projek ini. Mereka menjalankan pelbagai eksperimen ke atas aloi, biofuel, teknologi brek, komposisi kompaun tayar dan ciri keselamatan. Dari masa ke masa, beberapa teknologi daripada eksperimen ini telah diperhalusi dan memasuki laluan kenderaan arus perdana. Ada kemungkinan beberapa teknologi dalam kereta yang anda pandu mula-mula muncul pada prototaip perlumbaan.

Anda juga boleh mengatakan bahawa kenderaan arus perdana ini mewarisi teknologi daripada prototaip perlumbaan .

Kini kami mempunyai asas untuk membincangkan isu prototaip dan warisan dalam JavaScript. Ia tidak seperti corak warisan klasik yang anda ketahui dalam C, Java atau C#, tetapi ia sama berkuasa dan berpotensi lebih fleksibel.

Mengenai objek dan kelas

JavaScript adalah mengenai objek, yang merujuk kepada objek dalam erti kata tradisional, iaitu, "satu entiti yang mengandungi keadaan dan tingkah laku." Sebagai contoh, tatasusunan dalam JavaScript ialah objek yang mengandungi beberapa nilai dan mengandungi kaedah tolak, songsang dan pop.

var myArray = [1, 2];
myArray.push(3);
myArray.reverse();
myArray.pop();
var length = myArray.length;

Sekarang persoalannya, dari mana datangnya kaedah tolak? Bahasa statik yang kami nyatakan sebelum ini menggunakan "sintaks kelas" untuk mentakrifkan struktur objek, tetapi JavaScript ialah bahasa tanpa "sintaks kelas" dan tidak boleh menggunakan sintaks "kelas" Array untuk mentakrifkan setiap objek tatasusunan. Dan kerana JavaScript ialah bahasa dinamik, kita boleh meletakkan kaedah pada objek sewenang-wenangnya apabila diperlukan. Sebagai contoh, kod berikut mentakrifkan objek titik yang digunakan untuk mewakili titik dalam ruang dua dimensi, dan juga mentakrifkan kaedah tambah.

var point = {
  x : 10,
  y : 5,
  add: function(otherPoint) {
    this.x += otherPoint.x;
    this.y += otherPoint.y;
  }
};

Tetapi skalabiliti pendekatan di atas tidak bagus. Kami perlu memastikan bahawa setiap objek titik mengandungi kaedah tambah, dan kami juga berharap semua objek titik berkongsi pelaksanaan kaedah tambah yang sama, bukannya menambah kaedah ini secara manual pada setiap objek titik. Di sinilah prototaip dimainkan.

Mengenai prototaip

Dalam JavaScript, setiap objek mengekalkan sekeping keadaan tersembunyi — rujukan kepada objek lain, juga dipanggil prototaip. Tatasusunan yang kami buat sebelum ini merujuk kepada objek prototaip, begitu juga dengan objek titik yang kami cipta sendiri. Dikatakan di atas bahawa rujukan prototaip disembunyikan, tetapi terdapat juga pelaksanaan ECMAScript (nama rasmi JavaScript) yang boleh mengakses rujukan prototaip ini melalui atribut __proto__ objek (seperti Google Chrome). Dari segi konsep, kita boleh memikirkan objek sebagai serupa dengan hubungan objek-prototaip yang diwakili dalam Rajah 1.

Gambar 1

Melihat ke masa hadapan, pembangun akan dapat menggunakan fungsi Object.getPrototypeOf dan bukannya atribut __proto__ untuk mendapatkan rujukan kepada prototaip objek. Pada masa menulis artikel ini, fungsi Object.getPrototypeOf sudah tersedia dalam penyemak imbas Google Chrome, FIrefox dan IE9. Lebih banyak penyemak imbas akan melaksanakan ciri ini pada masa hadapan, kerana ia sudah menjadi sebahagian daripada standard ECMAScript. Kita boleh menggunakan kod berikut untuk membuktikan bahawa objek myArray dan point yang kami cipta merujuk kepada dua objek prototaip yang berbeza.

  1. Object.getPrototypeOf(point) != Object.getPrototypeOf(myArray);

Untuk seluruh artikel ini, saya akan menggunakan fungsi __proto__ dan Object.getPrototypeOf secara bergantian, terutamanya kerana __proto__ lebih mudah dikenali dalam rajah dan ayat. Apa yang perlu diingat ialah ia (__proto__) bukan standard, dan fungsi Object.getPrototypeOf ialah cara yang disyorkan untuk melihat prototaip objek.

Apakah yang membuatkan prototaip itu begitu istimewa?

Kami belum menjawab soalan ini lagi: Dari manakah kaedah tolak dalam tatasusunan berasal? Jawapannya ialah: ia datang dari objek prototaip myArray. Rajah 2 ialah tangkapan skrin penyahpepijat skrip dalam Chrome. Kami telah memanggil kaedah Object.getPrototypeOf untuk melihat objek prototaip myArray.

Gambar 2

  注意 myArray 的原型对象中有许多方法,包括那些在代码示例中调用的 push、pop 和 reverse 方法。因此,原型对象中的确包括 push 方法,但是 myArray 方法如何引用到呢?

myArray.push(3);

  了解其工作原理的第一步,是要认识到原型并不是特别的。原型只是普通的对象。可以给原型添加方法,属性,并把他们当作其他 JavaScript 对象一样看待。然而,套用乔治·奥威尔的小说《动物农场》中“猪”的说法 —— 所有的对象应当是平等的,但有些对象(遵守规则的)比其他人更加平等。

  JavaScript 中的原型对象的确是特殊的,因为他们遵从以下规则。当我们告诉 JavaScript 我们要调用一个对象的 push 方法,或读取对象的 x 属性时,运行时会首先查找对象本身。如果运行时找不到想要的东西,它就会循着 __proto__ 引用和对象原型寻找该成员。当我们  调用 myArray 的 push 方法时,JavaScript 并没有在 myArray 对象上发现 push 方法,而是在 myArray 的原型对象上找到了,于是 JavaScript 调用此方法(见图 3)。

3

  上面所描述的行为是指一个对象本身继承了原型上的任何方法或属性。JavaScript 中其实不需要使用类语法也能实现继承。就像从赛车原型上继承了相应的技术的车,一个 JavaScript 对象也可以从原型对象上继承功能特性。

  图 3 还展示了每个数组对象同时也可以维护自身的状态和成员。在请求得到 myArray 的 length 属性的情况下,JavaScript 会取得 myArray 中 length 属性的值,而不会去读取原型中的对应值。我们可以通过向对象上添加 push 这样的方法来“重写”push 方法。这样就会有效地隐藏原型中的 push 方法实现。

 共享原型

  JavaScript 中原型的真正神奇之处是多个对象如何维持对同一个原型对象的引用。例如,如果我们创建了这样的两个数组:

var myArray = [1, 2];
var yourArray = [4, 5, 6];

  那么这两个数组将共享同一个原型对象,而下面的代码计算结果为 true:

Object.getPrototypeOf(myArray) === Object.getPrototypeOf(yourArray);

  如果我们引用两个数组对象上的 push 方法,JavaScript 会去寻找原型上共享的 push 方法。

4

  JavaScript 中的原型对象提供继承功能,同时也就实现了该方法实现的共享。原型也是链式的。换句话说,因为原型对象只是一个对象,所以一个原型对象可以维持到另一个原型对象的引用。如果你重新审视图 2 便可以看到,原型的 __proto__ 属性是一个指向另一个原型的非空值。当 JavaScript 查找像 push 方法这样的成员时,它会循着原型引用链检查每一个对象,直到找到该成员,或者抵达原型链的末端。原型链为继承和共享开辟了一条灵活的途径。

  你可能会问的下一个问题是:我该如何设置那些自定义对象的原型引用呢?例如前面所使用的点对象,如何才能将 add 方法添加到原型对象中,并从多个点对象中继承方法呢?在回答这个问题之前,我们需要看看函数。

 有关函数

  JavaScript 中的函数也是对象。这样的表述带来了几个重要的结果,而我们并不会在本文中涉及所有的事项。这其中,能将一个函数赋值给一个变量,并且将一个函数作为参数传递给另一个函数的能力构成了现代 JavaScript 编程表达的基本范式。

  我们需要关注的是,函数本身就是对象,因此函数可以有自身的方法,属性,并且引用一个原型对象。让我们来讨论下面的代码的含义。

// 这将返回 true:
typeof (Array) === "function"
// 这样的表达式也是:
Object.getPrototypeOf(Array) === Object.getPrototypeOf(function () { })
// 这样的表达式同样:
Array.prototype != null

  代码中的第一行证明, JavaScript 中的数组是函数。稍后我们将看到如何调用 Array 函数创建一个新的数组对象。下一行代码,证明了 Array 对象使用与任何其他函数对象相同的原型,就像我们看到数组对象间共享相同的原型一样。最后一行代码证明了 Array 函数都有一个 prototype 属性,而这个 prototype 属性指向一个有效的对象。这个 prototype 属性十分重要。

  JavaScript 中的每一个函数对象都有 prototype 属性。千万不要混淆这个 prototype 属性的 __proto__ 属性。他们用途并不相同,也不是指向同一个对象。

// 返回 true
Object.getPrototypeOf(Array) != Array.prototype

  Array.__proto__ 提供的是 数组原型 – 请把它当作 Array 函数所继承的对象。

  而 Array.protoype,提供的的是 所有数组的原型对象。也就是说,它提供的是像 myArray 这样数组对象的原型对象,也包含了所有数组将会继承的方法。我们可以写一些代码来证明这个事实。

// true
Array.prototype == Object.getPrototypeOf(myArray)
// 也是 true
Array.prototype == Object.getPrototypeOf(yourArray);

  我们也可以使用这项新知识重绘之前的示意图。

5

  基于所知道的知识,请想象创建一个新的对象,并让新对象表现地像数组的过程。一种方法是使用下面的代码。

// 创建一个新的空对象
var o = {};
// 继承自同一个原型,一个数组对象
o.__proto__ = Array.prototype;
// 现在我们可以调用数组的任何方法...
o.push(3);

  虽然这段代码很有趣,也能工作,可问题在于,并不是每一个 JavaScript 环境都支持可写的 __proto__ 对象属性。幸运的是,JavaScript 确实有一个创建对象内建的标准机制,只需要一个操作符,就可以创建新对象,并且设置新对象的 __proto__ 引用 – 那就是“new”操作符。

var o = new Array();
o.push(3);

  JavaScript 中的 new 操作符有三个基本任务。首先,它创建新的空对象。接下来,它将设置新对象的 __proto__ 属性,以匹配所调用函数的原型属性。最后,操作符调用函数,将新对象作为“this”引用传递。如果要扩展最后两行代码,就会变成如下情况:

var o = {};
o.__proto__ = Array.prototype;
Array.call(o);
o.push(3);

  函数的 call 方法允许你在调用函数的情况下在函数内部指定“this”所引用的对象。当然,函数的作者在这种情况下需要实现这样的函数。一旦作者创建了这样的函数,就可以将其称之为构造函数。

  构造函数

  构造函数和普通的函数一样,但是具有以下两个特殊性质。

  1. 通常构造函数的首字母是大写的(让识别构造函数变得更容易)。
  2. 构造函数通常要和 new 操作符结合,用来构造新对象。

  Array 就是一个构造函数的例子。Array 函数需要和 new 操作符一起使用,而且 Array 的首字母是大写的。JavaScript 将 Array 作为内置函数包括在内,而任何人都可以写出自己的构造函数。事实上,我们最后可以为先前创建的点对象编写出构造函数。

var Point = function (x, y) {
  this.x = x;
  this.y = y;
  this.add = function (otherPoint) {
    this.x += otherPoint.x;
    this.y += otherPoint.y;
  }
}
var p1 = new Point(3, 4);
var p2 = new Point(8, 6);
p1.add(p2);

  在上面的代码中,我们使用了 new 操作符和 Point 函数来构造点对象,这个对象带有 x 属性和 y 属性和一个 add 方法。你可以将最后的结果想象成图 6 的样子。

6

  现在的问题是我们的每个点对象中仍然有单独的 add 方法。使用我们学到的原型和继承的知识,我们更希望将点对象的 add 方法从每个点实例中转移到 Point.prototype 中。要达到继承 add 方法的效果,我们所需要做的,就是修改 Point.prototype 对象。

var Point = function (x, y) {
  this.x = x;
  this.y = y;
}
Point.prototype.add = function (otherPoint) {
  this.x += otherPoint.x;
  this.y += otherPoint.y;
}
var p1 = new Point(3, 4);
var p2 = new Point(8, 6);
p1.add(p2);

  大功告成!我们刚刚在 JavaScript 中完成原型式的继承模式!

7

  总结

  我希望这篇文章能够帮助你揭开 JavaScript 原型概念的神秘面纱。开始看到的是原型怎样让一个对象从其他对象中继承功能,然后看到怎样结合 new 操作符和构造函数来构建对象。这里所提到的,只是开启对象原型力量和灵活性的第一步。本文鼓励你自己发现学习有关原型和 JavaScript 语言的新信息。

  同时,请小心驾驶。你永远不会知道这些行驶在路上的车辆会从他们的原型继承到什么(有缺陷)的技术。

  原文链接: Script Junkie   翻译: 伯乐在线 - 埃姆杰

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