亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

JavaScript執行順序詳細介紹

2019-11-20 21:31:34
字體:
來源:轉載
供稿:網友

之前從JavaScript引擎的解析機制來探索JavaScript的工作原理,下面我們以更形象的示例來說明JavaScript代碼在頁面中的執行順序。如果說,JavaScript引擎的工作機制比較深奧是因為它屬于底層行為,那么JavaScript代碼執行順序就比較形象了,因為我們可以直觀感覺到這種執行順序,當然JavaScript代碼的執行順序是比較復雜的,所以在深入JavaScript語言之前也有必要對其進行剖析。
1.1  按HTML文檔流順序執行JavaScript代碼
首先,讀者應該清楚,HTML文檔在瀏覽器中的解析過程是這樣的:瀏覽器是按著文檔流從上到下逐步解析頁面結構和信息的。JavaScript代碼作為嵌入的腳本應該也算做HTML文檔的組成部分,所以JavaScript代碼在裝載時的執行順序也是根據腳本標簽<script>的出現順序來確定的。例如,瀏覽下面文檔頁面,你會看到代碼是從上到下逐步被解析的。

復制代碼 代碼如下:

<script>
alert("頂部腳本");
</script>
<html><head>
<script>
alert("頭部腳本");
</script>
<title></title>
</head>
<body>
<script>
alert("頁面腳本");
</script>
</body></html>
<script>
alert("底部腳本");
</script>

如果通過腳本標簽<script>的src屬性導入外部JavaScript文件腳本,那么它也將按照其語句出現的順序來執行,而且執行過程是文檔裝載的一部分。不會因為是外部JavaScript文件而延期執行。例如,把上面文檔中的頭部和主體區域的腳本移到外部JavaScript文件中,然后通過src屬性導入。繼續預覽頁面文檔,你會看到相同的執行順序。

復制代碼 代碼如下:

<script>

alert("頂部腳本");

</script>

<html>

<head>

<script src="http://www.49028c.com/head.js"></script>

<title></title>

</head>

<body>

<script src="http://www.49028c.com/body.js"></script>

</body>

</html>

<script>

alert("底部腳本");

</script>

1.2  預編譯與執行順序的關系

在Javascript中,function才是Javascript的第一型。當我們寫下一段函數時,其實不過是建立了一個function類型的實體。
就像我們可以寫成這樣的形式一樣:

復制代碼 代碼如下:

functionHello()
{
alert("Hello");
}
Hello();
varHello = function()
{
alert("Hello");
}
Hello();

其實都是一樣的。 但是當我們對其中的函數進行修改時,會發現很奇怪的問題。
復制代碼 代碼如下:

<scripttype="text/javascript">       
functionHello() {        
alert("Hello");  
}       
Hello();     
functionHello() {  
alert("Hello World");   
}       
Hello();   
</script>

我們會看到這樣的結果:連續輸出了兩次Hello World。
而非我們想象中的Hello和Hello World。
這是因為Javascript并非完全的按順序解釋執行,而是在解釋之前會對Javascript進行一次“預編譯”,在預編譯的過程中,會把定義式的函數優先執行,也會把所有var變量創建,默認值為undefined,以提高程序的執行效率。
也就是說上面的一段代碼其實被JS引擎預編譯為這樣的形式:
復制代碼 代碼如下:

<scripttype="text/javascript">       
varHello = function() {          
alert("Hello");       
}      
Hello = function() {
           alert("Hello World"); 
      }     
Hello();    
Hello(); 
</script>

我們可以通過上面的代碼很清晰地看到,其實函數也是數據,也是變量,我們也可以對“函數“進行賦值(重賦值)。
當然,我們為了防止這樣的情況,也可以這樣:
復制代碼 代碼如下:

<scripttype="text/javascript">     
functionHello() {       
alert("Hello");      
}       
Hello(); 
</script>  
<scripttype="text/javascript">  
functionHello() {         
alert("Hello World");    
}      
Hello(); 
</script>

這樣,程序被分成了兩段,JS引擎也就不會把他們放到一起了。   

當JavaScript引擎解析腳本時,它會在預編譯期對所有聲明的變量和函數進行處理。

做如下處理:

1. 在執行前會進行類似“預編譯”的操作:首先會創建一個當前執行環境下的活動對象,并將那些用var申明的變量設置為活動對象的屬性,但是此時這些變量的賦值都是undefined,并將那些以function定義的函數也添加為活動對象的屬性,而且它們的值正是函數的定義。

2. 在解釋執行階段,遇到變量需要解析時,會首先從當前執行環境的活動對象中查找,如果沒有找到而且該執行環境的擁有者有prototype屬性時則會從prototype鏈中查找,否則將會按照作用域鏈查找。遇到var a = ...這樣的語句時會給相應的變量進行賦值(注意:變量的賦值是在解釋執行階段完成的,如果在這之前使用變量,它的值會是undefined) 所以,就會出現當JavaScript解釋器執行下面腳本時不會報錯:

復制代碼 代碼如下:

alert(a);                            // 返回值undefined

var a =1;

alert(a);                            // 返回值1


由于變量聲明是在預編譯期被處理的,所以在執行期間對于所有代碼來說,都是可見的。但是,你也會看到,執行上面代碼,提示的值是undefined,而不是1。這是因為,變量初始化過程發生在執行期,而不是預編譯期。在執行期,JavaScript解釋器是按著代碼先后順序進行解析的,如果在前面代碼行中沒有為變量賦值,則JavaScript解釋器會使用默認值undefined。由于在第二行中為變量a賦值了,所以在第三行代碼中會提示變量a的值為1,而不是undefined。

同理,下面示例在函數聲明前調用函數也是合法的,并能夠被正確解析,所以返回值為1。

復制代碼 代碼如下:

f();                                 // 調用函數,返回值1

function f(){

    alert(1);

}

但是,如果按下面方式定義函數,則JavaScript解釋器會提示語法錯誤。

復制代碼 代碼如下:

f();                                 // 調用函數,返回語法錯誤

var f = function(){

    alert(1);

}

這是因為,上面示例中定義的函數僅作為值賦值給變量f,所以在預編譯期,JavaScript解釋器只能夠為聲明變量f進行處理,而對于變量f的值,只能等到執行期時按順序進行賦值,自然就會出現語法錯誤,提示找不到對象f。

再見一些例子:

復制代碼 代碼如下:

<script type="text/javascript">

/*在預編譯過程中func是window環境下的活動對象中的一個屬性,值是一個函數,覆蓋了undefined值*/

alert(func); //function func(){alert("hello!")}

var func = "this is a variable"

function func(){

alert("hello!")

}

/*在執行過程中遇到了var重新賦值為"this is a variable"*/

alert(func);  //this is a variable

</script>

復制代碼 代碼如下:

<script type="text/javascript">
var name = "feng"; function func()
{
/*首先,在func環境內先把name賦值為undefined,然后在執行過程中先尋找func環境下的活動對象的name屬性,此時之前已經預編譯值為undefined,所以輸出是undefined,而不是feng*/
alert(name);  //undefined var name = "JSF";
alert(name);  //JSF
}
func();
alert(name);
//feng
</script>

雖然變量和函數聲明可以在文檔任意位置,但是良好的習慣應該是在所有JavaScript代碼之前聲明全局變量和函數,并對變量進行初始化賦值。在函數內部也是先聲明變量,然后再引用。

1.3  按塊執行JavaScript代碼

所謂代碼塊就是使用<script>標簽分隔的代碼段。例如,下面兩個<script>標簽分別代表兩個JavaScript代碼塊。

復制代碼 代碼如下:

<script>

// JavaScript代碼塊1

var a =1;

</script>

<script>

// JavaScript代碼塊2

function f(){

    alert(1);

}

</script>

JavaScript解釋器在執行腳本時,是按塊來執行的。通俗地說,就是瀏覽器在解析HTML文檔流時,如果遇到一個<script>標簽,則JavaScript解釋器會等到這個代碼塊都加載完后,先對代碼塊進行預編譯,然后再執行。執行完畢后,瀏覽器會繼續解析下面的HTML文檔流,同時JavaScript解釋器也準備好處理下一個代碼塊。

由于JavaScript是按塊執行的,所以如果在一個JavaScript塊中調用后面塊中聲明的變量或函數就會提示語法錯誤。例如,當JavaScript解釋器執行下面代碼時就會提示語法錯誤,顯示變量a未定義,對象f找不到。

復制代碼 代碼如下:

<script>

// JavaScript代碼塊1

alert(a);

f();

</script>

<script>

// JavaScript代碼塊2

var a =1;

function f(){

    alert(1);

}

</script>

雖然說,JavaScript是按塊執行的,但是不同塊都屬于同一個全局作用域,也就是說,塊之間的變量和函數是可以共享的。

1.4  借助事件機制改變JavaScript執行順序

由于JavaScript是按塊處理代碼,同時又遵循HTML文檔流的解析順序,所以在上面示例中會看到這樣的語法錯誤。但是當文檔流加載完畢,如果再次訪問就不會出現這樣的錯誤。例如,把訪問第2塊代碼中的變量和函數的代碼放在頁面初始化事件函數中,就不會出現語法錯誤了。

復制代碼 代碼如下:

<script>

// JavaScript代碼塊1

window.onload = function(){        // 頁面初始化事件處理函數

    alert(a);

    f();

}

</script>

<script>

// JavaScript代碼塊2

var a =1;

function f(){

    alert(1);

}

</script>

為了安全起見,我們一般在頁面初始化完畢之后才允許JavaScript代碼執行,這樣可以避免網速對JavaScript執行的影響,同時也避開了HTML文檔流對于JavaScript執行的限制。

注意

如果在一個頁面中存在多個windows.onload事件處理函數,則只有最后一個才是有效的,為了解決這個問題,可以把所有腳本或調用函數都放在同一個onload事件處理函數中,例如:

復制代碼 代碼如下:

window.onload = function(){

    f1();

    f2();

    f3();

}

而且通過這種方式可以改變函數的執行順序,方法是:簡單地調整onload事件處理函數中調用函數的排列順序。

除了頁面初始化事件外,我們還可以通過各種交互事件來改變JavaScript代碼的執行順序,如鼠標事件、鍵盤事件及時鐘觸發器等方法,詳細講解請參閱第14章的內容。

1.5  JavaScript輸出腳本的執行順序

在JavaScript開發中,經常會使用document對象的write()方法輸出JavaScript腳本。那么這些動態輸出的腳本是如何執行的呢?例如:

復制代碼 代碼如下:

document.write('<script type="text/javascript">');

document.write('f();');

document.write('function f(){');

document.write('alert(1);');

document.write('}');

document.write('</script>');


運行上面代碼,我們會發現:document.write()方法先把輸出的腳本字符串寫入到腳本所在的文檔位置,瀏覽器在解析完document.write()所在文檔內容后,繼續解析document.write()輸出的內容,然后才按順序解析后面的HTML文檔。也就是說,JavaScript腳本輸出的代碼字符串會在輸出后馬上被執行。

請注意,使用document.write()方法輸出的JavaScript腳本字符串必須放在同時被輸出的<script>標簽中,否則JavaScript解釋器因為不能夠識別這些合法的JavaScript代碼,而作為普通的字符串顯示在頁面文檔中。例如,下面的代碼就會把JavaScript代碼顯示出來,而不是執行它。

復制代碼 代碼如下:

document.write('f();');

document.write('function f(){');

document.write('alert(1);');

document.write(');');

但是,通過document.write()方法輸出腳本并執行也存在一定的風險,因為不同JavaScript引擎對其執行順序不同,同時不同瀏覽器在解析時也會出現Bug。

Ø 問題一,找不到通過document.write()方法導入的外部JavaScript文件中聲明的變量或函數。例如,看下面示例代碼。

復制代碼 代碼如下:

document.write('<script type="text/javascript" src="http://www.49028c.com/test.js">

</script>');

document.write('<script type="text/javascript">');

document.write('alert(n);');  // IE提示找不到變量n

document.write('</script>');

alert(n+1);                          // 所有瀏覽器都會提示找不到變量n

外部JavaScript文件(test.js)的代碼如下:

復制代碼 代碼如下:

var n = 1;

分別在不同瀏覽器中進行測試,會發現提示語法錯誤,找不到變量n。也就是說,如果在JavaScript代碼塊中訪問本代碼塊中使用document.write()方法輸出的腳本中導入的外部JavaScript文件所包含的變量,會顯示語法錯誤。同時,如果在IE瀏覽器中,不僅在腳本中,而且在輸出的腳本中也會提示找不到輸出的導入外部JavaScript文件的變量(表述有點長和繞,不懂的讀者可以嘗試運行上面代碼即可明白)。

Ø 問題二,不同JavaScript引擎對輸出的外部導入腳本的執行順序略有不同。例如,看下面示例代碼。

復制代碼 代碼如下:

<script type="text/javascript">

document.write('<script type="text/javascript" src="http://shaozhuqing.com/test1.js">

</script>');

document.write('<script type="text/javascript">');

document.write('alert(2);')

document.write('alert(n+2);');

document.write('</script>');

</script>

<script type="text/javascript">

alert(n+3);

</script>


外部JavaScript文件(test1.js)的代碼如下所示。
復制代碼 代碼如下:

var n = 1;

alert(n);

在IE瀏覽器中的執行順序如圖1-6所示。

圖1-6  IE 7瀏覽器的執行順序和提示的語法錯誤

在符合DOM標準的瀏覽器中的執行順序與IE瀏覽器不同,且沒有語法錯誤,如圖1-7所示的是在Firefox 3.0瀏覽器中的執行順序。

圖1-7  Firefox 3瀏覽器的執行順序和提示的語法錯誤

解決不同瀏覽器存在的不同執行順序,以及可能存在Bug。我們可以把凡是使用輸出腳本導入的外部文件,都放在獨立的代碼塊中,這樣根據上面介紹的JavaScript代碼塊執行順序,就可以避免這個問題。例如,針對上面示例,可以這樣設計:

復制代碼 代碼如下:

<script type="text/javascript">

document.write('<script type="text/javascript" src="http://www.49028c.com/test1.js"></script>');

</script>

<script type="text/javascript">

document.write('<script type="text/javascript">');

document.write('alert(2);') ; // 提示2

document.write('alert(n+2);'); // 提示3

document.write('</script>');

alert(n+3); // 提示4

</script>

<script type="text/javascript">

alert(n+4); // 提示5

</script>

這樣在不同瀏覽器中都能夠按順序執行上面代碼,且輸出順序都是1、2、3、4和5。存在問題的原因是:輸出導入的腳本與當前JavaScript代碼塊之間的矛盾。如果單獨輸出就不會發生沖突了。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
精品国产区一区二区三区在线观看| 亚洲iv一区二区三区| 欧美成人精品激情在线观看| 国产精品海角社区在线观看| 午夜精品国产精品大乳美女| 亚洲日本成人女熟在线观看| 欧美大片在线看免费观看| 久久国产精品99国产精| 精品成人乱色一区二区| 日韩精品一二三四区| 亚洲激情在线观看| 一本色道久久88综合亚洲精品ⅰ| 日韩av影片在线观看| 国产综合在线看| 欧美日韩爱爱视频| 欧日韩在线观看| 亚洲直播在线一区| 日韩专区在线观看| 久久99青青精品免费观看| 亚洲色图综合久久| 久久影视电视剧免费网站清宫辞电视| 最近2019中文字幕大全第二页| 色综合91久久精品中文字幕| 欧美精品久久久久久久免费观看| 亚洲第一色中文字幕| 久久精品中文字幕| 日韩综合视频在线观看| 欧美日韩第一视频| 精品久久久国产| 91在线色戒在线| 欧美成人精品在线| 亚洲欧洲在线免费| 国产精品美女久久久免费| 国产91精品久| 91亚洲一区精品| 国产国产精品人在线视| 97色在线视频观看| 国产一区二区三区在线观看网站| 成人免费淫片视频软件| 国产精品高潮呻吟久久av野狼| 亚洲韩国欧洲国产日产av| 96国产粉嫩美女| 亚洲热线99精品视频| 91高清在线免费观看| 992tv成人免费影院| 久久成人国产精品| 日韩av网址在线观看| 国产999在线观看| 亚洲天堂精品在线| 亚洲精品国产精品自产a区红杏吧| 97超碰国产精品女人人人爽| 国产做受69高潮| 亚洲欧美日韩国产中文| 欧美日韩国产色| 欧美成人午夜剧场免费观看| 亚洲精品电影久久久| 欧美日韩精品二区| 欧美在线播放视频| 亚洲2020天天堂在线观看| 一本色道久久综合亚洲精品小说| 成人欧美一区二区三区黑人孕妇| 日韩欧美中文字幕在线播放| 欧美丰满少妇xxxxx做受| 国产精品国产三级国产aⅴ浪潮| 国产精品久久久精品| 亚洲国产精品女人久久久| 亚洲成人网久久久| 国产欧美精品日韩| 日韩电影中文字幕一区| 国产精品久久久久久影视| 久久手机免费视频| 精品国产999| 欧美成人午夜剧场免费观看| 国产欧美韩国高清| 日韩视频永久免费观看| 国产91免费看片| 亚洲性视频网址| 狠狠色香婷婷久久亚洲精品| 亚洲午夜久久久影院| 亚洲激情第一页| 欧美日韩另类在线| 国产日韩精品在线观看| 亚洲free性xxxx护士白浆| 中文综合在线观看| 国产精品久久久av| 一区二区三区国产视频| xx视频.9999.com| 亚洲一区二区中文字幕| 精品久久香蕉国产线看观看亚洲| 日韩有码在线观看| 欧美性猛交xxxx乱大交极品| 久色乳综合思思在线视频| 日韩激情av在线播放| 欧美亚洲日本黄色| 亚洲欧美在线看| 欧美日韩免费看| 福利视频导航一区| 久久精品99无色码中文字幕| 亚洲自拍小视频| 成人美女av在线直播| 福利一区福利二区微拍刺激| 亚洲成人精品视频| 夜色77av精品影院| 国内精品伊人久久| 国产精品美腿一区在线看| 亚洲人精选亚洲人成在线| 日韩av在线看| 欧美性生交xxxxxdddd| 日韩免费在线电影| 福利精品视频在线| 精品亚洲一区二区| 亚洲自拍偷拍福利| 欧美最猛性xxxxx免费| 久久久久久亚洲精品中文字幕| 欧美日韩黄色大片| 亚洲电影免费观看| 欧美精品免费在线| 亚洲的天堂在线中文字幕| 日韩电影大片中文字幕| 久久久精品在线观看| 日韩黄色高清视频| 日韩黄色高清视频| 欧美性色xo影院| 国产成人一区三区| 91麻豆国产精品| 久久久久久噜噜噜久久久精品| 国产日韩亚洲欧美| 狠狠爱在线视频一区| 亚洲激情视频网站| 日本精品久久中文字幕佐佐木| 一区二区欧美亚洲| 国产精品精品久久久| 日本久久久久久久久| 日本三级韩国三级久久| 色吧影院999| 欧美日韩激情网| 亚洲最大福利视频网站| 美日韩精品免费视频| 国产成人a亚洲精品| 国产精品成人一区二区| 91久久精品久久国产性色也91| 国产日韩在线看| 成人网在线视频| 最近2019免费中文字幕视频三| 亚洲精品美女视频| 91精品啪在线观看麻豆免费| 精品福利在线观看| 成人免费在线视频网址| 在线视频精品一| 日本久久精品视频| 亚洲乱码国产乱码精品精天堂| 97精品一区二区三区| 成人a在线视频| 超碰日本道色综合久久综合| 精品中文字幕在线观看| **欧美日韩vr在线| 欧美精品一区二区免费| 亚洲偷熟乱区亚洲香蕉av| 亚洲级视频在线观看免费1级| 久久久综合av| 欧日韩不卡在线视频| 亚洲成人动漫在线播放| 韩国一区二区电影| 亚洲视频在线免费看|