Rumah hujung hadapan web tutorial js 关于JavaScript继承的多种方式以及优缺点的详细介绍

关于JavaScript继承的多种方式以及优缺点的详细介绍

May 14, 2017 am 10:12 AM

这篇文章主要介绍了深入理解JavaScript继承的多种方式和优缺点,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

写在前面

本文讲解JavaScript各种继承方式和优缺点。

注意:

跟《JavaScript深入之创建对象》一样,更像是笔记。

哎,再让我感叹一句:《JavaScript高级程序设计》写得真是太好了!

1.原型链继承

function Parent () {
  this.name = 'kevin';
}

Parent.prototype.getName = function () {
  console.log(this.name);
}

function Child () {

}

Child.prototype = new Parent();

var child1 = new Child();

console.log(child1.getName()) // kevin
Salin selepas log masuk

问题:

1.引用类型的属性被所有实例共享,举个例子:

function Parent () {
  this.names = ['kevin', 'daisy'];
}

function Child () {

}

Child.prototype = new Parent();

var child1 = new Child();

child1.names.push('yayu');

console.log(child1.names); // ["kevin", "daisy", "yayu"]

var child2 = new Child();

console.log(child2.names); // ["kevin", "daisy", "yayu"]
Salin selepas log masuk

2.在创建 Child 的实例时,不能向Parent传参

2.借用构造函数(经典继承)

function Parent () {
  this.names = ['kevin', 'daisy'];
}

function Child () {
  Parent.call(this);
}

var child1 = new Child();

child1.names.push('yayu');

console.log(child1.names); // ["kevin", "daisy", "yayu"]

var child2 = new Child();

console.log(child2.names); // ["kevin", "daisy"]
Salin selepas log masuk

优点:

1.避免了引用类型的属性被所有实例共享

2.可以在 Child 中向 Parent 传参

举个例子:

function Parent (name) {
  this.name = name;
}

function Child (name) {
  Parent.call(this, name);
}

var child1 = new Child('kevin');

console.log(child1.name); // kevin

var child2 = new Child('daisy');

console.log(child2.name); // daisy
Salin selepas log masuk

缺点:

方法都在构造函数中定义,每次创建实例都会创建一遍方法。

3.组合继承

原型链继承和经典继承双剑合璧。

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {

  Parent.call(this, name);
  
  this.age = age;

}

Child.prototype = new Parent();

var child1 = new Child('kevin', '18');

child1.colors.push('black');

console.log(child1.name); // kevin
console.log(child1.age); // 18
console.log(child1.colors); // ["red", "blue", "green", "black"]

var child2 = new Child('daisy', '20');

console.log(child2.name); // daisy
console.log(child2.age); // 20
console.log(child2.colors); // ["red", "blue", "green"]
Salin selepas log masuk

优点:融合原型链继承和构造函数的优点,是 JavaScript 中最常用的继承模式。

4.原型式继承

function createObj(o) {
  function F(){}
  F.prototype = o;
  return new F();
}
Salin selepas log masuk

就是 ES5 Object.create 的模拟实现,将传入的对象作为创建的对象的原型。

缺点:

包含引用类型的属性值始终都会共享相应的值,这点跟原型链继承一样。

var person = {
  name: 'kevin',
  friends: ['daisy', 'kelly']
}

var person1 = createObj(person);
var person2 = createObj(person);

person1.name = 'person1';
console.log(person2.name); // kevin

person1.firends.push('taylor');
console.log(person2.friends); // ["daisy", "kelly", "taylor"]
Salin selepas log masuk

注意:修改person1.name的值,person2.name的值并未发生改变,并不是因为person1person2有独立的 name 值,而是因为person1.name = 'person1',给person1添加了 name 值,并非修改了原型上的 name 值。

5. 寄生式继承

创建一个仅用于封装继承过程的函数,该函数在内部以某种形式来做增强对象,最后返回对象。

function createObj (o) {
  var clone = object.create(o);
  clone.sayName = function () {
    console.log('hi');
  }
  return clone;
}
Salin selepas log masuk

缺点:跟借用构造函数模式一样,每次创建对象都会创建一遍方法。

6. 寄生组合式继承

为了方便大家阅读,在这里重复一下组合继承的代码:

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {
  Parent.call(this, name);
  this.age = age;
}

Child.prototype = new Parent();

var child1 = new Child('kevin', '18');

console.log(child1)
Salin selepas log masuk

组合继承最大的缺点是会调用两次父构造函数。

一次是设置子类型实例的原型的时候:

Child.prototype = new Parent();
Salin selepas log masuk

一次在创建子类型实例的时候:

var child1 = new Child('kevin', '18');
Salin selepas log masuk

回想下 new 的模拟实现,其实在这句中,我们会执行:

Parent.call(this, name);
Salin selepas log masuk

在这里,我们又会调用了一次 Parent 构造函数。

所以,在这个例子中,如果我们打印 child1 对象,我们会发现 Child.prototype 和 child1 都有一个属性为colors,属性值为['red', 'blue', 'green']。

那么我们该如何精益求精,避免这一次重复调用呢?

如果我们不使用 Child.prototype = new Parent() ,而是间接的让 Child.prototype 访问到 Parent.prototype 呢?

看看如何实现:

function Parent (name) {
  this.name = name;
  this.colors = ['red', 'blue', 'green'];
}

Parent.prototype.getName = function () {
  console.log(this.name)
}

function Child (name, age) {
  Parent.call(this, name);
  this.age = age;
}

// 关键的三步
var F = function () {};

F.prototype = Parent.prototype;

Child.prototype = new F();


var child1 = new Child('kevin', '18');

console.log(child1);
Salin selepas log masuk

最后我们封装一下这个继承方法:

function object(o) {
  function F() {}
  F.prototype = o;
  return new F();
}

function prototype(child, parent) {
  var prototype = object(parent.prototype);
  prototype.constructor = child;
  child.prototype = prototype;
}

// 当我们使用的时候:
prototype(Child, Parent);
Salin selepas log masuk

引用《JavaScript高级程序设计》中对寄生组合式继承的夸赞就是:

这种方式的高效率体现它只调用了一次Parent构造函数,并且因此避免了在 Parent.prototype 上面创建不必要的、多余的属性。与此同时,原型链还能保持不变;因此,还能够正常使用 instanceof 和 isPrototypeOf。开发人员普遍认为寄生组合式继承是引用类型最理想的继承范式。

Atas ialah kandungan terperinci 关于JavaScript继承的多种方式以及优缺点的详细介绍. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan Laman Web ini
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

Alat AI Hot

Undresser.AI Undress

Undresser.AI Undress

Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover

AI Clothes Remover

Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool

Undress AI Tool

Gambar buka pakaian secara percuma

Clothoff.io

Clothoff.io

Penyingkiran pakaian AI

AI Hentai Generator

AI Hentai Generator

Menjana ai hentai secara percuma.

Artikel Panas

R.E.P.O. Kristal tenaga dijelaskan dan apa yang mereka lakukan (kristal kuning)
3 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Tetapan grafik terbaik
3 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Cara Memperbaiki Audio Jika anda tidak dapat mendengar sesiapa
3 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25: Cara Membuka Segala -galanya Di Myrise
4 minggu yang lalu By 尊渡假赌尊渡假赌尊渡假赌

Alat panas

Notepad++7.3.1

Notepad++7.3.1

Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina

SublimeText3 versi Cina

Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1

Hantar Studio 13.0.1

Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6

Dreamweaver CS6

Alat pembangunan web visual

SublimeText3 versi Mac

SublimeText3 versi Mac

Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Penjelasan terperinci tentang warisan fungsi C++: Bagaimana untuk menggunakan 'penunjuk kelas asas' dan 'penunjuk kelas terbitan' dalam warisan? Penjelasan terperinci tentang warisan fungsi C++: Bagaimana untuk menggunakan 'penunjuk kelas asas' dan 'penunjuk kelas terbitan' dalam warisan? May 01, 2024 pm 10:27 PM

Dalam pewarisan fungsi, gunakan "penunjuk kelas asas" dan "penunjuk kelas terbitan" untuk memahami mekanisme pewarisan: apabila penuding kelas asas menghala ke objek kelas terbitan, transformasi ke atas dilakukan dan hanya ahli kelas asas diakses. Apabila penuding kelas terbitan menghala ke objek kelas asas, hantaran ke bawah dilakukan (tidak selamat) dan mesti digunakan dengan berhati-hati.

Mesti membaca sebelum membeli sistem: Analisis kelebihan dan kekurangan Win11 dan Win10 Mesti membaca sebelum membeli sistem: Analisis kelebihan dan kekurangan Win11 dan Win10 Mar 28, 2024 pm 01:33 PM

Dalam era maklumat hari ini, komputer peribadi memainkan peranan penting sebagai alat yang sangat diperlukan dalam kehidupan seharian kita. Sebagai salah satu perisian teras komputer, sistem pengendalian mempengaruhi pengalaman penggunaan dan kecekapan kerja kami. Di pasaran, sistem pengendalian Windows Microsoft sentiasa menduduki kedudukan yang dominan, dan kini orang ramai menghadapi pilihan antara Windows 11 terkini dan Windows 10 lama. Bagi pengguna biasa, apabila memilih sistem pengendalian, mereka tidak hanya melihat nombor versi, tetapi juga memahami kelebihan dan kekurangannya.

Apakah kelebihan dan kekurangan templat? Apakah kelebihan dan kekurangan templat? May 08, 2024 pm 03:51 PM

Templat: Kebaikan dan Keburukan Templat ialah teknik pengaturcaraan yang berkuasa yang membolehkan anda membuat blok kod yang boleh digunakan semula. Ia menawarkan pelbagai kelebihan, tetapi juga beberapa kelemahan. Kelebihan: Kebolehgunaan Semula Kod: Templat membolehkan anda mencipta kod biasa yang boleh digunakan semula sepanjang aplikasi anda, mengurangkan usaha pertindihan dan penyelenggaraan. Ketekalan: Templat memastikan bahawa coretan kod dilaksanakan dengan cara yang sama di lokasi yang berbeza, meningkatkan ketekalan dan kebolehbacaan kod. Kebolehselenggaraan: Perubahan pada templat ditunjukkan secara serentak dalam semua kod yang menggunakannya, memudahkan penyelenggaraan dan kemas kini. Kecekapan: Templat menjimatkan masa dan usaha kerana anda tidak perlu menulis kod yang sama berulang kali. Fleksibiliti: Templat membolehkan anda membuat blok kod yang boleh dikonfigurasikan yang boleh disesuaikan dengan mudah kepada keperluan aplikasi yang berbeza. kelemahan

Apakah kelebihan dan kekurangan cara Java Servlets berfungsi? Apakah kelebihan dan kekurangan cara Java Servlets berfungsi? Apr 16, 2024 pm 03:18 PM

JavaServlet ialah kelas Java yang digunakan untuk membina halaman web dinamik dan berfungsi sebagai jambatan antara klien dan pelayan. Prinsip kerja: menerima permintaan, memulakan Servlet, memproses permintaan, menjana respons dan menutup Servlet. Kelebihan: Mudah alih, berskala, selamat dan mudah digunakan. Kelemahan: Overhed, gandingan dan pengurusan negeri. Kes praktikal: Cipta Servlet ringkas untuk memaparkan "Hello, Servlet!"

Perbandingan kelebihan dan kekurangan rangka kerja PHP: Mana satu yang lebih baik? Perbandingan kelebihan dan kekurangan rangka kerja PHP: Mana satu yang lebih baik? Jun 04, 2024 pm 03:36 PM

Pilihan rangka kerja PHP bergantung pada keperluan projek dan kemahiran pembangun: Laravel: kaya dengan ciri dan komuniti aktif, tetapi mempunyai keluk pembelajaran yang curam dan overhed prestasi tinggi. CodeIgniter: ringan dan mudah dipanjangkan, tetapi mempunyai fungsi terhad dan kurang dokumentasi. Symfony: Bermodul, komuniti yang kuat, tetapi kompleks, isu prestasi. ZendFramework: Gred perusahaan, stabil dan boleh dipercayai, tetapi besar dan mahal untuk dilesenkan. Langsing: rangka kerja mikro, pantas, tetapi dengan fungsi terhad dan keluk pembelajaran yang curam.

Bagaimanakah warisan dan polimorfisme mempengaruhi gandingan kelas dalam C++? Bagaimanakah warisan dan polimorfisme mempengaruhi gandingan kelas dalam C++? Jun 05, 2024 pm 02:33 PM

Pewarisan dan polimorfisme mempengaruhi gandingan kelas: Pewarisan meningkatkan gandingan kerana kelas terbitan bergantung pada kelas asas. Polimorfisme mengurangkan gandingan kerana objek boleh bertindak balas kepada mesej secara konsisten melalui fungsi maya dan penunjuk kelas asas. Amalan terbaik termasuk menggunakan warisan dengan berhati-hati, menentukan antara muka awam, mengelakkan penambahan ahli data pada kelas asas dan menyahgandingkan kelas melalui suntikan kebergantungan. Contoh praktikal yang menunjukkan cara menggunakan polimorfisme dan suntikan pergantungan untuk mengurangkan gandingan dalam aplikasi akaun bank.

Bahasa C dan Python: analisis senario yang berkenaan serta kelebihan dan kekurangan Bahasa C dan Python: analisis senario yang berkenaan serta kelebihan dan kekurangan Mar 22, 2024 am 11:24 AM

Bahasa C dan Python: analisis senario yang boleh digunakan serta kelebihan dan kekurangan Dalam bidang pengaturcaraan komputer, bahasa C dan Python adalah dua bahasa pengaturcaraan yang sangat popular, masing-masing mempunyai kelebihan dan kekurangan yang unik dan sesuai untuk senario yang berbeza. Artikel ini akan menjalankan analisis mendalam bahasa C dan Python, membincangkan senario, kelebihan dan keburukan yang berkenaan. 1. Senario terpakai bahasa C: Bahasa C ialah bahasa pengaturcaraan berorientasikan proses dengan kecekapan tinggi dan prestasi cemerlang Ia sesuai untuk pembangunan perisian sistem, pemacu dan sistem terbenam yang memerlukan tahap kawalan dan kecekapan yang tinggi.

Penjelasan terperinci tentang warisan fungsi C++: Bagaimana untuk menyahpepijat ralat dalam warisan? Penjelasan terperinci tentang warisan fungsi C++: Bagaimana untuk menyahpepijat ralat dalam warisan? May 02, 2024 am 09:54 AM

Petua penyahpepijatan ralat warisan: Pastikan perhubungan warisan yang betul. Gunakan penyahpepijat untuk melangkah melalui kod dan memeriksa nilai pembolehubah. Pastikan anda menggunakan pengubah suai maya dengan betul. Periksa masalah berlian warisan yang disebabkan oleh warisan tersembunyi. Semak fungsi maya tulen yang tidak dilaksanakan dalam kelas abstrak.

See all articles