摘要:然后將構(gòu)造函數(shù)的原型設(shè)為,便實現(xiàn)了對象繼承。首先,我們定義一個構(gòu)造函數(shù),并在其中定義一個局部變量。這里的是局部變量,其作用域仍然存在是閉包現(xiàn)象,而非對象屬性。
Javascript是動態(tài)的,弱類型的,解釋執(zhí)行的程序設(shè)計語言。
Javascript極其靈活,支持多種程序設(shè)計范式:面向?qū)ο?、指令式、函?shù)式。JavaSCript最初被用于瀏覽器腳本,現(xiàn)在已經(jīng)是所有主流瀏覽器的默認(rèn)腳本語言。瀏覽器腳本的作用包括用戶交互、DOM操作、以及與服務(wù)器通信。Node.js的流行已經(jīng)將JavaScript從瀏覽器端引入到服務(wù)器端,并顯示出卓越的并發(fā)性能。
Javascript 由 Netscape 瀏覽器首次引入到Web文檔中。Javascript的出現(xiàn)使web文檔變得更像動態(tài)的App,而不是靜態(tài)的文檔?,F(xiàn)在主流的瀏覽器都有javascript支持。Javascript核心被標(biāo)準(zhǔn)化為ECMAScript。
JavaScript作為一門通用編程語言(general purpose),其語法、內(nèi)置對象、以及工具庫需要大量的精力才能掌握,但JavaScript有著直觀的語法,是一門非常容易上手的編程語言。本章中介紹JavaScript最基本的語法,然后編寫一個簡單的Javascript程序。
1.Hello World使用你喜歡的文本編輯器,建立一個javascript文件(后綴命名為.js即可),輸入以下的內(nèi)容:
console.log("Hello, world!")
console為多數(shù)JavaScript的宿主環(huán)境都會提供的全局對象,它的log方法用來輸出(到標(biāo)準(zhǔn)輸出)一個字符串。為了運(yùn)行這段程序,可以打開Chrome瀏覽器的開發(fā)者工具(右上角的設(shè)置按鈕->更多工具->開發(fā)者工具),進(jìn)入Console頁面。
拷貝這段代碼進(jìn)去,按下回車,你會看到在控制臺的輸出:Hello, world!。
2.數(shù)據(jù)類型Javascript 是一門弱類型和動態(tài)類型的語言。弱類型意味著允許隱式類型轉(zhuǎn)換,不同類型的變量間可以直接賦值和運(yùn)算;動態(tài)類型則意味著只在運(yùn)行時才會進(jìn)行類型檢查,于是包含錯誤的文件仍然能夠得到執(zhí)行,直到運(yùn)行至包含類型錯誤的語句,JavaScript程序才異常退出。
Javascript中包含5種基本類型(primitive types):
字符串(String):其取值為字符序列。字符串可以用單引號或雙引號分隔,二者完全等價。
var str1 = "str1", str2 = ""str1" is a string";
數(shù)字(Number):它是實數(shù)?整數(shù)?無符號數(shù)?這些都不需要考慮!Javascript中,數(shù)字就是數(shù)字!
var n = 1; n = n/3; // 0.33333...
布爾類型(Boolean):只有兩種取值:true, false。
空(Null):該類型只有一個值可取,它就是null。在多數(shù)Javascript運(yùn)行時中,null是一種特殊的object。
未定義(Undefined):該類型也只有一個值可取,它就是undefined。有趣的是,你可以把一個變量聲明為未定義:
var foo = undefined;
除了基本數(shù)據(jù)類型,還有一種對象(object)類型,Object是JavaScript中所有其他對象的原型(如果你熟知Java等基于類繼承的語言,可以理解為Object是其他所有對象的公共父類)。日期、數(shù)組、函數(shù)都是JavaScript內(nèi)置的特殊對象。ECMAScript將對象定義為屬性的集合(Collection),創(chuàng)建一個對象的語法就像創(chuàng)建一個集合:
var student = {name: "小明", age: 23};
我們看到:
所有的變量都是以var聲明的,如果你忘記了寫var,那么這個變量將成為整個運(yùn)行時的全局變量。
另外,Javascript與C++使用同樣的注釋風(fēng)格,即/**/用于多行注釋,//用于單行注釋。
3.對象Javascript是面向?qū)ο蟮木幊陶Z言,對象是javascript中最重要的概念。上一節(jié)我們通過一對大括號創(chuàng)建了一個對象:
var student = {name: "小明", age: 23};
如果你熟悉構(gòu)造函數(shù),JavaScript也支持這種方式:
function Student(name){ this.name = name; this.age = 23; } var student = new Student("小明"); student.school = "PKU";
在Javascript中,通過this來訪問自己的屬性。有趣的是,訪問屬性前不需聲明:
如果直接讀取未聲明屬性,會得到undefined;
如果直接寫入未聲明屬性,則會聲明并用指定的值初始化該屬性。
如果你熟悉設(shè)計模式,你可能會需要這樣的對象創(chuàng)建方式:
function StudentFactory(_school){ var school = _school; this.create = function(name, age){ return { name: name, age : age, school: school } } } var factory = new StudentFactory("PKU"); var student = factory.create("小明", 23);
上述代碼中,首先聲明了一個工廠對象(function是一種特殊的對象),并用school來配置該工廠。此后用該工廠生產(chǎn)一個學(xué)生對象:小明。
現(xiàn)在,你已經(jīng)熟悉了Javascript中對象的創(chuàng)建。如果你曾開發(fā)過C++或者Java,你可能會關(guān)心Javascript中如何進(jìn)行繼承,以及實現(xiàn)多態(tài)。
不錯,Javascript是面向?qū)ο笳Z言。但沒有類的聲明和實例化機(jī)制,Javascript使用原型繼承的方式(prototype)??吹较旅娴睦右苍S就清楚了:
// file: prototype.js var Person = { sayhi : function(){ console.log("hi! I"m", this.name) } }; function Student(name){ this.name = name; this.school = "pku"; } Student.prototype = Person; var student = new Student("alice"); // hi! I"m alice student.sayhi();
上述代碼中,首先聲明了Person對象,他有一個sayhi屬性,該屬性的值為function類型(還記得嗎?函數(shù)是一種特殊的對象)。然后將構(gòu)造函數(shù)Student的原型設(shè)為Person,便實現(xiàn)了對象繼承。
4.函數(shù)prototype對象是個模板,要實例化的對象都以這個模板為基礎(chǔ)(屬性和方法都被傳遞給那個類的所有實例)。prototype是可以傳遞的,最終形成原型鏈。
JavaScript 最令人感興趣的可能莫過于函數(shù)實際上是功能完整的對象,通過function來聲明函數(shù)。在JavaScript中的函數(shù)是一級公民,這意味著函數(shù)和其他變量一樣,可以被傳參、賦值、以及返回。函數(shù)有如下幾個默認(rèn)屬性:
1.length:參數(shù)個數(shù)
function func(arg1, arg2){} // 2 console.log(func.length)
2.toString:這是所有對象共有的方法,將會輸出函數(shù)的源代碼
// function func(arg1, arg2){} console.log(func.toString())5.arguments
在函數(shù)體中,JavaScript提供了一個特殊對象 arguments,它是當(dāng)前函數(shù)被調(diào)用時傳入的參數(shù)數(shù)組,通過訪問arguments不僅可以訪問所有實參,還可以獲得實參的數(shù)目,從而實現(xiàn)其他編程語言中可變參數(shù)的機(jī)制:
/ file: arguments.js function sayHi(){ if(arguments.length == 1) console.log("I"m", arguments[0]); if(arguments.length == 2) console.log("I"m", arguments[0], "aged", arguments[1]); } // I"m alice sayHi("alice"); // I"m alice aged 23 sayHi("alice", 23);
6.call & applyJavascript并未提供函數(shù)重載、默認(rèn)參數(shù)機(jī)制。但我們可以直接訪問arguments參數(shù)列表,sayHi其實模仿了函數(shù)重載。
call和apply是函數(shù)對象的兩個函數(shù)屬性,一般用于對象冒充、包裝或者代理。可能在教學(xué)中不會用到太多,但在JS庫的開發(fā)中會經(jīng)常用到。這兩個函數(shù)的功能均為用一個對象去調(diào)用一個函數(shù),而該函數(shù)并非該對象的屬性,例如:
// file: call.js function sayHi(age){ console.log("I"m", this.name, "aged", age) } var student = { name: "alice" }; // I"m alice aged 23 sayHi.call(student, 23)
call的第一個參數(shù)是用作this的對象。其他參數(shù)都直接傳遞給函數(shù)自身。既然如此,如果我們希望實現(xiàn)一個自己的log函數(shù),來包裝console.log:
// file: log.js function log(){ var str = ""; for(var i = 0; i < arguments.length; i++){ str += " " + arguments[i] } console.log(str); } // a b c log("a", "b", "c");
這里我們不能使用call函數(shù),因為我們不知道參數(shù)個數(shù)。這便是apply的用武之地:
// file: log.js function log(){ console.log.apply(this, arguments) } // a b c log("a", "b", "c");
apply接受的第二個參數(shù)為參數(shù)數(shù)組,而非call的參數(shù)列表。這便為上述的情況提供了便利。
7.回調(diào)與異步因Javascript最初運(yùn)行于瀏覽器端,與服務(wù)器的通信必然要異步執(zhí)行(否則,將會阻塞主控制流,此時瀏覽器會不響應(yīng)用戶操作)。于是,Javascript天生就是異步的,Node.js也是采用異步事件而大幅提高I/O密集型任務(wù)的效率。
異步是計算機(jī)在執(zhí)行任務(wù)的過程中,某些任務(wù)可以獨立于主控制流執(zhí)行,使得主控制流得以繼續(xù)執(zhí)行,是一種非阻塞的控制方案。
在Javascript中有多種異步的實現(xiàn)方案,它們無一例外地需要回調(diào)函數(shù)。回調(diào)函數(shù)就是將函數(shù)作為參數(shù)傳遞到其他代碼(例如:某一個異步任務(wù)),這一設(shè)計允許了該異步任務(wù)完成時執(zhí)行傳入的函數(shù)。例如:
// file: async.js function taskFinished(){ console.log("Task finished!") } setTimeout(taskFinished, 3000);
上述代碼中,首先定義了一個任務(wù)完成時需要執(zhí)行的函數(shù)taskFinished。setTimeout是Javascript提供的一個全局對象(還記得嗎?函數(shù)是一種特殊的對象),它是一個計時器,在到達(dá)指定的時間后調(diào)用某個函數(shù)。上述例子中,在5000毫秒后運(yùn)行taskFinished函數(shù)。
8.變量作用域與閉包Javascript的變量作用域不同于其他的主流編程語言。包括變量的定義方式、作用域的劃分、作用持續(xù)時間等。
在javascript中使用賦值語句即可定義一個對象,而不需要對象聲明(也不需要指定變量的類型)。使用var可以定義局部變量,而省略var則可以定義全局變量。例如:
// 定義局部變量 foo var foo = "bar" // 定義全局變量 bar bar = "foo
上述的全局變量作用域不是當(dāng)前對象,也不是當(dāng)前文件,而是整個運(yùn)行時進(jìn)程!實際項目中,應(yīng)盡量避免引入全局變量。眾所周知,全局變量會使得代碼高度耦合、難以復(fù)用和維護(hù)、復(fù)雜化團(tuán)隊協(xié)作。
那么,局部變量的作用域是怎樣的呢?局部作用域即當(dāng)前函數(shù),不同于C++或Java的當(dāng)前代碼塊(以大括號分隔)。正因為javascript的這一點特殊性質(zhì),引出了javascript中的一個重要概念:閉包(closure)。閉包是指是引用了自由變量的函數(shù)。這個被引用的自由變量將和這個函數(shù)一同存在,即使已經(jīng)離開了它最初被創(chuàng)建的環(huán)境。
閉包之所以如此重要,是因為它完成了javascript的諸多面向?qū)ο筇匦裕?strong>數(shù)據(jù)封裝、運(yùn)行時多態(tài)。我們看一個閉包:
// file: closure.js function Count(){ var num=0; this.add = function(){ console.log(num++); } } var c = new Count(); c.add(); c.add();
Count是函數(shù),而函數(shù)是對象,對象可以有屬性和方法。add是方法屬性(類型為方法的屬性),而num則只是函數(shù)的局部變量。
首先,我們定義一個構(gòu)造函數(shù)Count,并在其中定義一個局部變量num。接著,我們調(diào)用該構(gòu)造函數(shù)創(chuàng)建一個對象c,并調(diào)用兩次它的add方法。不出所料,程序的輸出將是:
0 1
你可能已經(jīng)注意到:第二次調(diào)用add方法時,add中引用的num仍然保持著上次調(diào)用后的值。事實上,num是被add方法引用的一個自由變量,其作用域會一直跟隨add而存在。
不要混淆局部變量和對象屬性。這里的num是局部變量,其作用域仍然存在是閉包現(xiàn)象,而非對象屬性。對象屬性需要用this關(guān)鍵字來定義。
版權(quán)聲明如果你希望進(jìn)一步學(xué)習(xí)Javascript,Mozilla Developer和W3C School: Javascript教程都提供了很好文檔。如果你愿意深究Javascript語法規(guī)則,請參考ECMAScript標(biāo)準(zhǔn):ECMAScript-262。
本文由Harttle創(chuàng)作,轉(zhuǎn)載需署名作者且注明文章出處
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/87727.html
摘要:所有全局對象函數(shù)以及變量均自動成為對象的成員。消息彈出窗口對象使用簡單方便,但因其交互方式生硬,通常只用于調(diào)試。可以在中創(chuàng)建三種消息框警告框確認(rèn)框提示框。習(xí)慣上,通過選擇符得到的集合一般命名為以起始的字符串,例如。 avascript 等于 ECMAScript + 宿主環(huán)境。宿主環(huán)境提供了一系列的全局對象,例如Node.js提供的require, console等;瀏覽器提供的DOM...
摘要:簡介是對象表示法的縮寫是一種數(shù)據(jù)格式而不是一種編程語言用來表示結(jié)構(gòu)化數(shù)據(jù)是的一個嚴(yán)格子集并不從屬于很多編程語言都可以用數(shù)據(jù)格式語法語法可以表示以下三種類型的值簡單值字符串?dāng)?shù)值布爾值但是不支持對象數(shù)組不支持變量函數(shù)對象實例簡單值字符串?dāng)?shù)字布爾 JSON 簡介 JSON是JavaScript Object Notation(JavaScript對象表示法)的縮寫 JSON是一種數(shù)據(jù)格式,...
摘要:簡介是的新增特性。我們統(tǒng)一把這些操作稱為副作用,或者簡稱為作用。由于副作用函數(shù)是在組件內(nèi)聲明的,所以它們可以訪問到組件的和。副作用函數(shù)還可以通過返回一個函數(shù)來指定如何清除副作用。目前為止,有兩種主流方案來解決這個問題高階組件和。 Hook 簡介 Hook 是 React 16.8 的新增特性。它可以讓你在不編寫 class 的情況下使用 state 以及其他的 React 特性。 us...
摘要:用聲明的變量,其可用范圍在當(dāng)前執(zhí)行上下文。在函數(shù)外聲明的變量,其作用范圍是全局。拋出語法錯誤錯誤信息未捕獲的異常標(biāo)識符已經(jīng)被聲明過。例如輸出照例,命名規(guī)范指出我們應(yīng)該用大寫字母聲明常量。為了讓代碼長期可維護(hù),命名規(guī)范還是值得遵守的。 轉(zhuǎn)載請注明出處:葡萄城官網(wǎng),葡萄城為開發(fā)者提供專業(yè)的開發(fā)工具、解決方案和服務(wù),賦能開發(fā)者。原文出處:https://dzone.com/articles/...
摘要:但是在中,可以通過關(guān)鍵字來實現(xiàn)類的繼承的使用可以使得繼承意義更加明確并且值得一提的是,如果你使用來定義的組件,那么可以在類的構(gòu)造器里面,用簡單的的聲明方式來替代方法。 原文:The 10 min ES6 course for the beginner React Developer譯者:Jim Xiao 著名的80/20定律可以用來解釋React和ES6的關(guān)系。因為ES6增加了超過75...
閱讀 1020·2021-11-22 13:52
閱讀 1450·2021-11-19 09:40
閱讀 3184·2021-11-16 11:44
閱讀 1279·2021-11-15 11:39
閱讀 3913·2021-10-08 10:04
閱讀 5374·2021-09-22 14:57
閱讀 3108·2021-09-10 10:50
閱讀 3191·2021-08-17 10:13