欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JS原型和原型鏈原理與用法實例詳解

 更新時間:2020年02月05日 10:40:40   作者:米彩  
這篇文章主要介紹了JS原型和原型鏈原理與用法,結(jié)合實例形式詳細分析了js原型和原型鏈概念、原理、使用方法及相關(guān)操作注意事項,需要的朋友可以參考下

本文實例講述了JS原型和原型鏈原理與用法。分享給大家供大家參考,具體如下:

Javascript語言的繼承機制一直很難被人理解。

它沒有"子類"和"父類"的概念,也沒有"類"(class)和"實例"(instance)的區(qū)分,全靠一種很奇特的"原型鏈"(prototype chain)模式,來實現(xiàn)繼承。

Brendan Eich設(shè)計javascript之初是為了實現(xiàn)網(wǎng)頁與瀏覽器之間交互的一種簡單的腳本語言

如果真的是一種簡易的腳本語言,其實不需要有"繼承"機制。但是,Javascript里面都是對象,必須有一種機制,將所有對象聯(lián)系起來。所以,Brendan Eich最后還是設(shè)計了"繼承"。

背景介紹

1.構(gòu)造函數(shù)

構(gòu)造函數(shù) ,是一種特殊的方法。主要用來在創(chuàng)建對象時初始化對象。每個構(gòu)造函數(shù)都有prototype(原型)屬性

2.原型模式

每個函數(shù)都有prototype(原型)屬性,這個屬性是一個指針,指向一個對象,這個對象的用途是包含特定類型的所有實例共享的屬性和方法,即這個原型對象是用來給實例共享屬性和方法的。

而每個實例內(nèi)部都有一個指向原型對象的指針。

原型鏈

每個構(gòu)造函數(shù)都有一個原型對象,原型對象都包含一個指向構(gòu)造函數(shù)的指針,而實例都包含指向原型對象內(nèi)部的指針。我們讓原型對象的實例(1)等于另一個原型對象(2),

此時原型對象(2)將包含一個指向原型對象(1)的指針,

再讓原型對象(2)的實例等于原型對象(3),如此層層遞進就構(gòu)成了實例和原型的鏈條,這就是原型鏈的概念

構(gòu)造函數(shù)

構(gòu)造函數(shù) ,是一種特殊的方法。主要用來在創(chuàng)建對象時初始化對象。 即為對象變量賦初始值。每個構(gòu)造函數(shù)的實例都將共享構(gòu)造函數(shù)的初始值。 構(gòu)造函數(shù)的出現(xiàn)是為了解決使用Object構(gòu)造函數(shù)和字面量表示法不方便創(chuàng)建大量重復(fù)對象的問題。

傳統(tǒng)創(chuàng)建對象實例的方法

  var person={
    name:'張女士',
    age:'80',
    gender:'女'
  };
 console.log(person)

注:這個方法如果用于創(chuàng)建大量相同屬性和方法的對象時,會產(chǎn)生大量重復(fù)代碼

構(gòu)造函數(shù)的方法

//構(gòu)造函數(shù)方法創(chuàng)建對象實例
function Person(name,age,gender) {
this.name=name;
this.age=age;
this.gender=gender;
this.say=function () {
    alert(this.name)
  }
}
var person1=new Person('鐘女士',80,'女');
var person2=new Person('張女士',80,'女');
console.log(person2)
console.log(person1)

原型模式

使用構(gòu)造函數(shù)的問題是,每個方法都要在每個實例上重新創(chuàng)建一遍,即在構(gòu)造函數(shù)的不同實例上的同名函數(shù)是不相等的。而我們創(chuàng)建每個構(gòu)造函數(shù)都有一個prototype(原型)屬性,這個屬性是個指針,指向一個對象,而這個對象的用途是包含可以由特定類型的所有實例共享的屬性和方法,我們使用這個原型對象來共享實例的屬性和方法的模式就叫原型模式

//原型模式創(chuàng)建對象
function Person(){
}
Person.prototype.name='鐘女士';
Person.prototype.age=80;
Person.prototype.gender='女';
var person1= new Person();
console.log(person1)
//簡寫原型模式
Person.prototype={
  constructor:Person
  name:'鐘女士',
  age:80,
  gender:'女'
}

注:每個原型對象都有constructor屬性,由于簡寫模式重寫了默認的prototype對象,所以constructor也會被重新定義,不再指向他的構(gòu)造函數(shù),所以可以自己寫一個constructor屬性指向他的構(gòu)造函數(shù)

原型鏈

每個構(gòu)造函數(shù)都有原型對象,每個構(gòu)造函數(shù)實例都包含一個指向原型對象的內(nèi)部指針(proto),如果我們讓第一個構(gòu)造函數(shù)的原型對象等于第二個構(gòu)造函數(shù)的實例,結(jié)果第一個構(gòu)造函數(shù)的原型對象將包含一個指向第二個原型對象的指針,再然第三個原型對象等于第一個構(gòu)造函數(shù)的實例,這樣第三個原型對象也將包含指向第一個原型對象的指針,以此類推,就夠成了實例于原型的鏈條,這就是原型鏈的基本概念

function One(){
}
function Two(){
}
function Three(){
}
Two.prototype=new One();
Three.prototype=new Two();
var three=new Three();
console.log(three);
console.log(three.__proto__===Three.prototype) //true
console.log(three.__proto__.__proto__===Two.prototype) //true
console.log(three.__proto__.__proto__.__proto__===One.prototype) //true
console.log(three.__proto__.__proto__.__proto__.__proto__===Object.prototype) //true

在對象實例中,訪問對象原型的方法

  • 1、使用proto屬性

此屬性是瀏覽器支持的一個屬性,并不是ECMAScript里的屬性

  • 2.Object.getPrototypeOf
  • 3.使用constructor.prototype的方法

對于不支持proto的瀏覽器,可以使用constructor,訪問到對象的構(gòu)造函數(shù),在用prototype訪問到原型

使用原型鏈解釋ANUGLAR作用域

在開發(fā)過程中,我們可能會出現(xiàn)控制器的嵌套,看下面這段代碼:

  <div ng-controller="OuterCtrl">
    <span>{{a}}</span>
     <div ng-controller="InnerCtrl">
      <span>{{a}}</span>
     </div>
   </div>
  <script>
  function OuterCtrl($scope) {
  $scope.a = 1;
  }
  function InnerCtrl($scope) {
  }
  </script>

我們可以看到界面顯示了兩個1,而我們只在OuterCtrl的作用域里定義了a變量,但界面給我們的結(jié)果是,兩個a都有值,現(xiàn)在自控制器里的a是從父控制器里繼承過來的

我們可以父子級的作用域看成兩個原型對象,其中一個原型對象繼承另一個原型對象的實例

function Outer() {
  this.a = 1;
}
function Inner() {
}
var outer = new Outer();
Inner.prototype=new Outer();
var inner = new Inner();
console.log(outer.a)
console.log(inner.a)

Angular的實現(xiàn)機制其實也就是把這兩個控制器中的$scope作了關(guān)聯(lián),外層的作用域?qū)嵗蔀榱藘?nèi)層作用域的原型。

既然作用域是通過原型來繼承的,自然也就可以推論出一些特征來。比如說這段代碼,點擊按鈕的結(jié)果是什么?

<div ng-controller="OuterCtrl">
  <span>{{a}}</span>
  <div ng-controller="InnerCtrl">
    <span>{{a}}</span>
    <button ng-click="a=a+1">a++</button>
  </div>
</div>
<script>
function OuterCtrl($scope) {
  $scope.a = 1;
}
 
function InnerCtrl($scope) {
}
</script>

點了按鈕之后,兩個a不一致了,里面的變了,外面的沒變,這是為什么?

function Outer() {
  this.a = 1;
}
function Inner() {
}
var outer = new Outer();
Inner.prototype=new Outer();
var inner = new Inner();
inner.a = inner.a + 1;
console.log(outer.a)
console.log(inner.a)

因為在原型鏈中,訪問一個實例屬性時,會在實例本身查找,如果找不到,則搜索實例的原型,如果再搜索不到,則繼續(xù)沿著原型鏈往上查找。找到之后則會賦給該實例,所以inner上面就被賦值了一個新的a,outer里面的仍然保持原樣,這也就導(dǎo)致了剛才看到的結(jié)果。

上下級共享變量

比如說,我們就是想上下級共享變量,不創(chuàng)建新的,該怎么辦呢?

function Outer() {
  this.data = {
    a: 1
  };
}
function Inner() {
}
var outer = new Outer();
Inner.prototype = outer;
var inner = new Inner();
console.log(outer.data.a);
console.log(inner.data.a);
inner.data.a += 1;
console.log(outer.data.a);
console.log(inner.data.a);

我們可以把a寫在一個對象里,當(dāng)inner找到對象data并賦值到自己身上時,其實是復(fù)制了對象的指針(參考高程第4章復(fù)制引用類型和基本類型的區(qū)別),我們對對象里的屬性的改動都會反映到所有引用該對象的元素上。
反映到AngularJs,我們可以這么寫

<div ng-controller="OuterCtrl">
  <span>{{data.a}}</span>
  <div ng-controller="InnerCtrl">
    <span>{{data.a}}</span>
    <button ng-click="data.a=data.a+1">increase a</button>
  </div>
</div>
<script>
function OuterCtrl($scope) {
  $scope.data = {
    a: 1
  };
}
function InnerCtrl($scope) {
}
</script>

這樣點擊按鈕兩個控制器的a都會+1

感興趣的朋友可以使用在線HTML/CSS/JavaScript代碼運行工具http://tools.jb51.net/code/HtmlJsRun測試上述代碼運行效果。

更多關(guān)于JavaScript相關(guān)內(nèi)容可查看本站專題:《JavaScript常用函數(shù)技巧匯總》、《javascript面向?qū)ο笕腴T教程》、《JavaScript查找算法技巧總結(jié)》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》及《JavaScript數(shù)學(xué)運算用法總結(jié)

希望本文所述對大家JavaScript程序設(shè)計有所幫助。

相關(guān)文章

  • top.location.href 沒有權(quán)限 解決方法

    top.location.href 沒有權(quán)限 解決方法

    以前好像沒有遇到這問題,也可能是沒有在意吧,我的blog內(nèi)容頁都是有判斷的,規(guī)則是,如果top.location不是內(nèi)容頁的話就跳到內(nèi)容頁
    2008-08-08
  • 聊聊KeyCode被棄用后的問題

    聊聊KeyCode被棄用后的問題

    今天咱們就來聊聊KeyCode被棄用后的問題,希望對大家有所啟發(fā),如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • JavaScript字符串檢索字符的方法

    JavaScript字符串檢索字符的方法

    這篇文章主要為大家詳細介紹了JavaScript字符串檢索字符的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • Javascript簡單改變表單元素背景的方法

    Javascript簡單改變表單元素背景的方法

    這篇文章主要介紹了Javascript簡單改變表單元素背景的方法,涉及javascript操作頁面元素樣式的基本技巧,非常簡單實用,需要的朋友可以參考下
    2015-07-07
  • JavaScript數(shù)組操作之旋轉(zhuǎn)二維數(shù)組

    JavaScript數(shù)組操作之旋轉(zhuǎn)二維數(shù)組

    這篇文章主要介紹了JavaScript數(shù)組操作之旋轉(zhuǎn)二維數(shù)組,主要從兩個方面展開文章介紹,一是通過對數(shù)組的操作熟練度;二是(鏡像反轉(zhuǎn))比實現(xiàn)一更優(yōu),減少了空間復(fù)雜度,內(nèi)容介紹具有一定的參考價值,需要的小伙伴可以參考一下
    2022-04-04
  • 使用Xcache緩存器加速PHP網(wǎng)站的配置方法

    使用Xcache緩存器加速PHP網(wǎng)站的配置方法

    從訪問速度上來看,一般要比放于國內(nèi)的網(wǎng)站慢2-3倍,所以便想辦法對網(wǎng)站做了一些簡單的優(yōu)化,比如使用緩存系統(tǒng)來提升網(wǎng)站頁面訪問速度
    2017-04-04
  • 基于mpvue小程序使用echarts畫折線圖的方法示例

    基于mpvue小程序使用echarts畫折線圖的方法示例

    這篇文章主要介紹了基于mpvue小程序使用echarts畫折線圖的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-04-04
  • js中遞歸函數(shù)的使用介紹

    js中遞歸函數(shù)的使用介紹

    所謂的遞歸函數(shù)就是在函數(shù)體內(nèi)調(diào)用本函數(shù)。使用遞歸函數(shù)一定要注意,處理不當(dāng)就會進入死循環(huán)。遞歸函數(shù)只有在特定的情況下使用 ,比如階乘問題
    2012-10-10
  • javascript中數(shù)組的concat()方法使用介紹

    javascript中數(shù)組的concat()方法使用介紹

    數(shù)組的concat()方法想必大家比不陌生吧,在本文為大家介紹下javascript中數(shù)組的concat()方法的具體使用,感興趣的朋友可以參考下
    2013-12-12
  • JavaScript引用賦值與傳值賦值總結(jié)

    JavaScript引用賦值與傳值賦值總結(jié)

    這篇文章主要介紹了JavaScript引用賦值與傳值賦值總結(jié),在JavaScript中基本數(shù)據(jù)類型都是傳值賦值,復(fù)合數(shù)據(jù)類型都是引用賦值(傳地址)也叫引用傳址,下文更多相關(guān)資料,需要的小伙伴可以參考一下
    2022-05-05

最新評論