倔強(qiáng),倔的正確讀音是什么?
jué第二聲,
“倔”,普通話讀音為jué、juè,初見于傳抄古文。在讀jué時,“倔”的基本含義為頑強(qiáng),固執(zhí),如倔強(qiáng)(jiàng);引申含義為“倔”古同“崛”,突出。在讀juè時,“倔”的含義為言語粗直,態(tài)度不好,如那老頭真倔。
在日常使用中,“倔”也常做形容詞,表示奇譎,如倔傀、倔佹。
半條命怎么換第三人稱模式?
控制臺中輸入sv_cheats1然后保存、讀取再次輸入thirdperson就OK但這種模式效果極差我相信你還會再輸入一次firstperson的
如何理解Javascript的原型和原型鏈?
Javascript中的原型和原型鏈都是實現(xiàn)OOP的手段,OOP在Javascript中的具體實現(xiàn)如下:
對象(Object)就是屬性(Property)的集合,特別的,稱值(Value)為函數(shù)(Function)的屬性為方法(Method)。將相似對象的共有屬性提取出來聚集在一起就形成了類(Class),這些對象稱為該類的實例(Instance)。同樣,將相似類的共有屬性提取出來聚集在一起也形成新的類,這個類是前面那些類的超類(SuperClass),前面那些類是這個類的子類(SubClass)。多個超類還可以作為子類聚集出一個新的超類,這個過程會一持續(xù)下去,直到出現(xiàn)名為Object的類,它的超類為空(Null)。
類除了是共有屬性的聚集外,還擔(dān)負(fù)對象工廠(ObjectFactory)的職責(zé)。一個類的實例對象由類的構(gòu)造函數(shù)(Constructor)負(fù)責(zé)創(chuàng)建。構(gòu)造函數(shù)負(fù)責(zé)兩件事:
創(chuàng)建對象;
初始化該對象;
因為前者的實現(xiàn)已經(jīng)由方法提供,所以構(gòu)造函數(shù)真正需要完成的就是初始化對象,這里又分為兩件事情:
讓對象具有類所聚集的共有屬性;
根據(jù)參數(shù),對某些對象的屬性進(jìn)行特化;
對于第二件事情,沒什么說的,就是將特化的屬性添加到待初始化的對象中去。對于第一件事,也可以仿照后者的實現(xiàn)方法,但是這不是一個明智的選擇,因為這些共有屬性的值在大多數(shù)情況下是不會發(fā)生改變的。Javascript選擇的方法是:
以這些共有屬性為屬性并賦予默認(rèn)屬性值,創(chuàng)建一個原型(Prototype)對象;
初始化時,將原型對象賦予待初始化對象的特殊屬性:__proto__;
也就是說,一個類對應(yīng)一個原型對象,在初始化時,用__proto___將實例對象和原型對象連接起來。
特殊屬性__proto__不僅負(fù)責(zé)連接實例和原型,還負(fù)責(zé)連接子類和超類的原型對象,以實現(xiàn)類之間的繼承關(guān)系。這樣以來,一個對象的類原型,超類原型,超類的超類原型,...就由__proto__連接成一個“鏈”,稱為該對象的原型鏈。允許,一個對象的__proto__屬性為null,這表明該對象沒有原型鏈,Object類的原型就是這樣的。
為了讓原型初始化實例的方法真正得以實現(xiàn),必須在對象的屬性訪問上進(jìn)行配合:
讀取屬性值:先在對象中查找該屬性,如果存在則返回其值,否則,在原型對象中查找,如果存在則返回其值,否則,在原型對象的原型對象中查找,...,直到原型鏈為null,表示該屬性未定義,返回undefined;
給屬性賦值:在對象中查找該屬性,如果存在則對其賦值,如果不存在則在對象中創(chuàng)建該屬性然后對其賦值;
刪除屬性:如果該屬性在對象中存在則刪除它,否則什么都不做。
這套訪問機(jī)制保證了:對象屬性可以覆蓋(去覆蓋)原型屬性,但是不會改變原型屬性,這就是OOP的多態(tài)性。
構(gòu)造函數(shù)在創(chuàng)建對象時需要用到原型對象,它是通過prototype屬性知道其對應(yīng)類的原型對象的。另外,為讓實例對象知道是誰創(chuàng)建了它,它的constructor屬性會“抓著”構(gòu)造函數(shù)。類的原型對象也被認(rèn)為是該類的構(gòu)造函數(shù)構(gòu)創(chuàng)建的。
接下來我們看一下實現(xiàn)OOP的具體代碼:
首先,不考慮繼承關(guān)系,聲明一個類的范例代碼如下:
注:特殊屬性__proto__是undocumented應(yīng)該避免直接使用,正式的做法是調(diào)用方法,它的參數(shù)就是所要創(chuàng)建對象的原型對象。注:遵照OOP語言的傳統(tǒng),構(gòu)造函數(shù)的名字就是類的名字。
當(dāng)一個函數(shù)被調(diào)用時,如果this上下文(Context)綁定的是一個普通對象(而非null或全局對象window),則這個函數(shù)就是作為該對象的方法被調(diào)用。
當(dāng)我們用new表達(dá)式創(chuàng)建對象時,構(gòu)造函數(shù)就是以方法的被new調(diào)用:
上面范例代碼中構(gòu)造函數(shù)開始和結(jié)束部分所作的事情,new表達(dá)式,就替我們干了:
new會創(chuàng)建一個空白對象,讓其,原型鏈綁定構(gòu)造函數(shù)的prototype屬性,讓其,constructor屬性綁定構(gòu)造函數(shù);然后以該對象為this上下文調(diào)用構(gòu)造函數(shù),如果構(gòu)造函數(shù)沒有返回值,則以空白對象作為創(chuàng)建的對象。寫成代碼就是:
被new調(diào)用的構(gòu)造函數(shù),已經(jīng)轉(zhuǎn)變?yōu)闃?gòu)造方法,但為了讓其還保留構(gòu)造函數(shù)的能力,一般這樣實現(xiàn):
接下來,考慮類的繼承。
一個實例對象的初始化過程是:先被超類的構(gòu)造函數(shù)初始化,之后才被子類的構(gòu)造函數(shù)初始化,這樣才能達(dá)到子類覆蓋超類的要求。基于此,范例代碼如下:
寫到這里,我們發(fā)現(xiàn)又是一堆不得不寫的規(guī)范代碼。于是早期很多前端框架,都紛紛的提供了以上代碼的封裝方案,旦各自為政,沒有統(tǒng)一的解決方法,直到ES6直接提供了class語法,整個事情才算告一段落:
注:Javascript中的屬性分為存儲屬性和訪問屬性(分別對應(yīng)傳統(tǒng)OOP語言中的字段(Field)和屬性),class中只能聲明原型中方法和訪問屬性,而在原型中聲明存儲屬性還得是老辦法。
class表達(dá)式只是語法層面的封裝,最終依然是基于原型和原型鏈這套實現(xiàn)。
雖然我們現(xiàn)在已經(jīng)不需要按照那套復(fù)雜的規(guī)范聲明類了,但是了解原型和原型鏈對應(yīng)深入理解Javascript的OOP機(jī)制依然十分重要。
最后,給出Javascript內(nèi)建對象之間的原型鏈關(guān)系圖(粗箭頭是__proto__屬性,細(xì)箭頭是prototype屬性,虛箭頭是constructor屬性):
(Value:Number,String,Boolean;Symbol,Container:Array,Set,Map)