javascript
JavaScript面向对象精要(一)
數據類型
在JavaScript中,數據類型分為兩類:
原始類型
保存一些簡單數據,如true,5等。JavaScript共有5中原始類型:
- boolean:布爾,值為true或false
- number:數字,值為任何整型會浮點數值
- string:字符串,值為由單引號或雙引號括出的單個字符或連續字符(JavaScript不區分字符類型)
- null:空類型,其僅有一個值:nulll
- undefined:未定義,其僅有一個值:undefined
原始類型的值是直接保存在變量中,并可以用 typeof 進行檢測。但是typeof對null的檢測是返回object,而不是返回null:
//彈出Not null if(typeof null){alert("Not null"); }else{alert("null"); }所以檢測null時,最好用全等于(===),其還能避免強制類型轉換:
console.log("21" === 21); //false console.log("21" == 21); //true console.log(undefined == null); //true console.log(undefined === null); //false對于字符串、數字或者布爾值,其都有對應的方法,這些方法來自于對應的原始封裝類型:String、Number和Boolean。原始封裝類型將被自動創建。
var name = "Pomy"; var char = name.charAt(0); console.log(char); //"P"在JavaScript引擎中發生的事情:
var name = "Pomy"; var temp = new String(name); var char = temp.charAt(0); temp = null; console.log(char); //"P"字符串對象的引用在用完之后立即被銷毀,所以不能給字符串添加屬性,并且instanceof檢測對應類型時均返回false:
var name = "Pomy"; name.age = 21; console.log(name.age); //undefined console.log(name instanceof String); //false引用類型
保存為對象,實質是指向內存位置的引用,所以不在變量中保存對象。除了自定義的對象,JavaScript提供了6中內建類型:
- Array:數組類型,以數字為索引的一組值的有序列表
- Date:日期和時間類型
- Error:運行期錯誤類型
- Function:函數類型
- Object:通用對象類型
- RegExp:正則表達式類型
可以用new來實例化每一個對象,或者用字面量形式來創建對象:
var obj = new Object; var own = {name:"Pomy",blog:"http://www.ido321.com","my age":22}; console.log(own.blog); //訪問屬性 console.log(own["my age"]); obj = null; //解除引用obj 并不包含對象實例,而是一個指向內存中實際對象所在位置的指針(或者說引用)。因為typeof對所有非函數的引用類型均返回object,所以需要用instanceof來檢測引用類型。
函數
在JavaScript中,函數就是對象。使函數不同于其他對象的決定性特性是函數存在一個被稱為[[Call]]的內部屬性。內部屬性無法通過代碼訪問而是定義了代碼執行時的行為。
創建形式
1、函數聲明:用function關鍵字,會被提升至上下文
2、函數表達式:不能被提升
3、實例化Function內建類型
參數
JavaScript函數的另外一個獨特之處在于可以給函數傳遞任意數量的參數。函數參數被保存在arguments類數組對象中,其自動存在函數中,能通過數字索引來引用參數,但它不是數組實例:
alert(Array.isArray(arguments)); //false類數組對象arguments 保存的是函數的實參,但并不會忽略形參。因而,arguments.length返回實參列表的長度,arguments.callee.length返回形參列表的長度。
function ref(value){return value; } console.log(ref("Hi")); console.log(ref("Hi",22)); console.log(ref.length); //1函數中的this
關于this的問題,可參考此文:JavaScript中的this。
JavaScript提供了三個方法用于改變this的指向:call、apply和bind。三個函數的第一個參數都是指定this的值,其他參數均作為參數傳遞給函數。
對象
對象是一種引用類型,創建對象常見的兩種方式:Object構造函數和對象字面量形式:
var per1 = {name:"Pomy",blog:"http://www.ido321.com" }; var per2 = new Object; per2.name = "不寫代碼的碼農";屬性操作
在JavaScript中,可以隨時為對象添加屬性:
per1.age = 0; per1.sayName = function(){alert(this.name); //"Pomy" }因而,在檢測對象屬性是否存在時,常犯的一個錯誤是:
//結果是false if(per1.age){alert(true) }else{alert(false); }per1.age 是存在的,但是其值是0,所以不能滿足if條件。if判斷中的值是一個對象、非空字符串、非零數字或true時,判斷會評估為真;而當值是一個null、undefined、0、false、NaN或空字符串時評估為假。
因而,檢測屬性是否存在時,有另外的兩種方式:in和hasOwnProperty(),前者會檢測原型屬性和自有(實例)屬性,后者只檢測自有(實例)屬性。
console.log("age" in per1); //true console.log(per1.hasOwnProperty("age")); //true console.log("toString" in per1); //true console.log(per1.hasOwnProperty("toString")); //false對象per1并沒有定義toString,該屬性繼承于Object.prototype,所以in和hasOwnProperty()檢測該屬性時出現差異。如果只想判斷一個對象屬性是不是原型,可以利用如下方法:
function isPrototypeProperty(obj,name){return name in obj && !obj.hasOwnProperty(name); }若要刪除一個屬性,用delete操作符,用于刪除自有屬性,不能刪除原型屬性。
per1.toString = function(){console.log("per1對象"); }; console.log(per1.hasOwnProperty("toString")); //true per1.toString(); //"per1對象" delete per1.toString; console.log(per1.hasOwnProperty("toString")); //false console.log(per1.toString()); //[object Object]有時需要枚舉對象的可枚舉屬性,也有兩種方式:for-in循環和Object.keys(),前者依舊會遍歷出原型屬性,后者只返回自有屬性。所有可枚舉屬性的內部屬性[[Enumerable]]的值均為true。
var per3 = {name:"Pomy",blog:"http://www.ido321.com",age:22,getAge:function(){return this.age;} };實際上,大部分原生屬性的[[Enumerable]]的值均為false,即該屬性不能枚舉。可以通過propertyIsEnumerable()檢測屬性是否可以枚舉:
console.log(per3.propertyIsEnumerable("name")); //true var pros = Object.keys(per3); //返回可枚舉屬性的名字數組 console.log("length" in pros); //true console.log(pros.propertyIsEnumerable("length")); //false屬性name是自定義的,可枚舉;屬性length是Array.prototype的內建屬性,不可枚舉。
屬性類型
屬性有兩種類型:數據屬性和訪問器屬性。二者均具有四個屬性特征:
- 數據屬性:[[Enumerable]]、[[Configurable]]、[[Value]]和[[Writable]]
- 訪問器屬性:[[Enumerable]]、[[Configurable]]、[[Get]]和[[Set]]
**[[Enumerable]] :**布爾值,屬性是否可枚舉,自定義屬性默認是true。
**[[Configurable]] :**布爾值,屬性是否可配置(可修改或可刪除),自定義屬性默認是true。它是不可逆的,即設置成false后,再設置成true會報錯。
**[[Value]]:**保存屬性的值。
**[[Writable]]:**布爾值,屬性是否可寫,所有屬性默認可寫。
**[[Get]]:**獲取屬性值。
**[[Set]]:**設置屬性值。
ES 5提供了兩個方法用于設置這些內部屬性:
Object.defineProperty(obj,pro,desc_map) 和 Object.defineProperties(obj,pro_map)。利用這兩個方法為per3添加一個屬性和創建一個新對象per4:
需要注意的是,通過這兩種方式來定義新屬性時,如果不指定特征值,則默認是false,也不能創建同時具有數據特征和訪問器特征的屬性。可以通過Object.getOwnPropertyDescriptor()方法來獲取屬性特征的描述,接受兩個參數:對象和屬性名。若屬性存在,則返回屬性描述對象。
var desc = Object.getOwnPropertyDescriptor(per4,"name"); console.log(desc.enumerable); //false console.log(desc.configurable); //false console.log(desc.writable); //true根據屬性的屬性類型,返回的屬性描述對象包含其對應的四個屬性特征。
禁止修改對象
對象和屬性一樣具有指導其行為的內部特征。其中,[[Extensible]]是一個布爾值,指明改對象本身是否可以被修改([[Extensible]]值為true)。創建的對象默認都是可以擴展的,可以隨時添加新的屬性。
ES5提供了三種方式:
- Object.preventExtensions(obj):創建不可擴展的obj對象,可以利用Object.isExtensible(obj)來檢測obj是否可以擴展。嚴格模式下給不擴展對象添加屬性會報錯,非嚴格模式下則添加失敗。
- Object.seal(obj):封印對象,此時obj的屬性變成只讀,不能添加、改變或刪除屬性(所有屬性都不可配置),其[[Extensible]]值為false,[[Configurable]]值為false。可以利用Object.isSealed(obj)來檢測obj是否被封印。
- Object.freeze(obj):凍結對象,不能在凍結對象上添加或刪除屬性,不能改變屬性類型,也不能寫入任何數據類型。可以利用Object.isFrozen(obj)來檢測obj是否被凍結。
注意:凍結對象和封印對象均要在嚴格模式下使用。
注意,禁止修改對象的三個方法只對對象的自有屬性有效,對原型對象的屬性無效,仍然可以在原型上添加或修改屬性。
function Person(name){this.name = name; } var person1 = new Person("Pomy"); var person2 = new Person("dwqs"); Object.freeze(person1); Person.prototype.Hi = function(){console.log("Hi"); }; person1.Hi(); //"Hi"; person2.Hi(); //"Hi";轉載于:https://www.cnblogs.com/ddlove/p/6294359.html
總結
以上是生活随笔為你收集整理的JavaScript面向对象精要(一)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: FLAC5.0出错时提示未赋予杨氏模量的
- 下一篇: Paxos 实现日志复制同步(Multi