首先我們先來瞭解一下ASP頁面執行的流程
1.IIS找到ASP檔案,提交給ASP引擎(一般是ASP.DLL)處理。
2.引擎開啟這個ASP檔案,找出<%和%>之間的內容,當然還有<script runAt="server">和對應的</script>之間的內容,這些內容稱為指令碼塊。只有指令碼塊裡的內容被引擎解析,其他內容不管,作為沒有意義的字元插在指令碼塊之間。有必要說明一下的是,其實被解析的內容還不止這些,<!--#include ***-->類的伺服器端包含檔案也是由引擎包含進來並加以處理的。如果你讀的程式比較多,你還會知道有的runAt屬性標註為"Server"的<object>對象也是會被處理的,這裡不做深入討論。
3.引擎執行指令碼塊中的指令碼,這些伺服器端的指令碼是作為一個整體被執行的,也就是說,可以寫出如下的代碼:
<%
Dim i
For i=1 to 5
%> Hello World!
<% Next %>
引擎並不會將這些指令碼塊分開解析,而使兩個指令碼塊都發生語法錯誤。所以我們得到如下結論:並非所有非伺服器指令碼的代碼都會被發送到用戶端,有可能這段非伺服器指令碼的代碼被指令碼塊限制了。伺服器是一定不會操心用戶端指令碼的執行問題的,但是可以通過伺服器端的指令碼輸出不同的用戶端指令碼。
4.最終引擎產生了一個文字資料流,或者說是指令碼的執行結果,可以認為這是一個字串,就是發送到用戶端瀏覽器的網頁的代碼。用戶端瀏覽器將頁面顯示出來,此時頁面的原始碼(源檔案)是不包含伺服器端的指令碼的,但包含了伺服器端指令碼的執行結果(這是顯然的)。
<% … %> 與 <script runat="server">…</script>
它們都是伺服器端的指令碼,同時被處理執行。他們執行時是作為一個整體的。
<% … %> 與 <script language="…">…</script>
前者是伺服器端指令碼,後者是用戶端指令碼。前者先執行,後者後執行。
其實也不盡然,二者的指令碼是有可能在同時被執行的,但空間不同,仍然是:前者在伺服器上執行,後者在用戶端瀏覽器裡執行。前者在邏輯上一定提前於後者執行。同時我們也得到結論:在同一個頁面的執行中,用戶端指令碼無論如何不能反饋給伺服器端指令碼,也就是說,用戶端瀏覽你的留言本並且提交新留言或者是任何用戶端指令碼擷取的值都不可能在同一次伺服器響應中被處理。
關於組件的調用
注意伺服器端指令碼和用戶端指令碼都是指令碼,自然都可以建立xmlhttp組件、ADODB.Connection組件等,但是並不是放在哪裡都可以的。
xmlhttp如果用於伺服器的抓取網頁(比如採集)就要在伺服器指令碼裡建立了,而如果是用於用戶端的ajax無重新整理而後台訪問伺服器端的頁面,那麼就是運行於用戶端的了,自然在用戶端建立。
ADODB.Connection組件用於訪問資料庫,一般來說在伺服器端建立,畢竟是伺服器端的asp程式在跑資料庫的資料,但如果你的資料庫真的是在用戶端串連的,那麼就毫無疑問在用戶端指令碼裡建立了。
總之,矛盾著的事物及其每一個側面各有其特點。不同事物有不同的矛盾;同一事物在發展的不同過程和不同階段上有不同的矛盾;同一事物中的不同矛盾、同一矛盾的兩個不同方面各有其特殊性(看不懂的可以略去不看……)。這一原理要求我們堅持具體問題具體分析原則,在矛盾普遍性原理的指導下,具體分析矛盾的特殊性,並找出解決矛盾的正確方法。反對千篇一律地採用一種方法解決不同事物的矛盾。“一把鑰匙開一把鎖,到什麼山唱什麼歌”講的就是這個道理。
伺服器端VBScript指令碼建立對象使用Server.CreateObject(className)方法,用戶端VBScript指令碼建立對象使用CreateObject(className)方法。
典型錯誤
<%
Function TSize(b)
'這是我自訂的函數
TSize="中國"
end function
%>
<a href="javascript:<%TSize('變數')%>" >點這裡要使用我定義的函數</a>
錯誤分析:
混淆了伺服器端指令碼和用戶端指令碼的區別。實際執行時我們會發現,用戶端根本沒有收到什麼TSize之類的代碼,因為TSize是伺服器端的程式,被引擎處理之後(注意引擎對於函數的處理,純粹是給伺服器端指令碼調用的,不會發回到用戶端)就消失了,不可能在用戶端起作用。這就是說,用戶端指令碼無法直接調用伺服器端指令碼的函數。
事實上,這個程式是有語法錯誤的,引擎處理這段內容的時候先找到了<%和%>之間的內容,也就是<%TSize('變數')%>,顯然這段內容不符合VBScript的文法規則。嗯,改成<%=TSize("變數")%>在伺服器端指令碼就沒有語法錯誤了,這時TSize函數可以正常傳回值"中國",於是用戶端收到的href屬性是這樣寫的:"javascript:中國",是無法執行的。