ASP物件導向編程探討及比較 選擇自 RAINMAN_NET 的 Blog
關鍵字 ASP物件導向編程探討及比較
出處
ASP是Microsoft於較早期推出的動態網頁編程技術,但其結合ADO對資料庫方便快捷的訪問、結合XML、COM/ActiveX等其它技術 實現伺服器多層結構的功能使它在今天還有著頑強的生命力,並且依然有著一定的發展。ASP.Net雖然在架構上完全不同於ASP,但它很多內建對象也是基 於ASP進行擴充的。網上有無數的介紹ASP的文章,卻鮮有介紹ASP物件導向以及與其它語言比較的,這也就是我下決心寫這篇文章的原因。
因為是早期的版本,ASP只提供了很弱的物件導向的介面。眾所周知,ASP的實現語言分為VBScript和JavaScript/JScript: 在VBScript中有Class關鍵字,可以用來聲明一個自訂類;JavaScript就比較怪,它用一個函數來“聲明”類,然後在該函數裡通過 this.prototype定義屬性,this.func定義方法。這裡將以VBScript為主進行討論,VBScript的類聲明是這樣的:
Class name
statements
End Class
這裡statements裡可以聲明公有或私人的成員,包括函數、成員和屬性。關於屬性,不得不贊一下微軟的get和set方法,這個在COM中出現 的理念,直到.Net中一直被沿用下來,個人認為對程式員而言,比Java用getProp()、setProp()兩個方法來實現同樣效果要方便直觀得 多。
相比之下,VBScript中的類與PHP4中的類各有千秋(當然跟最新的PHP5沒法比),VBScript中的類保持了VB的不完全物件導向的 “特性”,它僅僅實現了最基本的構造/解構函式、成員函數、變數、屬性,甚至建構函式不能帶參數。PHP4中則還實現了繼承、函數重載等類的重要性質,也 只有實現了這些,才能稱之為物件導向,才有可能為實現多態提供基礎。但二者均沒有實作類別的靜態(static)成員等功能。儘管可以用其它一些變通達到同 樣的功效,但從物件導向的思想出發,這都是不徹底的(由於PHP非常靈活,PHP4中可以通過成員函數的靜態變數來間接實作類別的靜態變數;而“::”—— 可以實作類別的靜態函數訪問的操作符——在PHP4中沒有嚴格檢查。換句話說,所有的成員函數都可以當成靜態函數訪問,只要你在該函數裡不使用成員變數就不 會出錯。VBScript根本沒有實現static,只能用Session或Application來實現)。所以在平常的使用中,你可以使用 VBScript的自訂類來封裝一些操作,但不要指望它像C++ / Java / .Net那樣為你的物件導向思想服務。
VBScript同樣發揚了VB中預設的參數或變數是引用的好風格。這樣,儘管Script語言中對類型不敏感,但它還能夠達到C/C++裡指標/引用同樣的功效,完成很多事情。最基本的,比如說用它定義一個列表(List)的節點類ListNode:
<%
Class ListNode
Public Content
Public NextNode
Private Sub Class_Initialize()
Content="Node"
Set NextNode=Nothing
End Sub
End Class
%>
呵呵,就這麼簡單,但不要感到鄙夷,也不要忘記對變數初始值。VB中也差不多,聲明時加上類型就行了。而使用時:
<%
Set nh=new ListNode
Set nh.NextNode=new ListNode
'其它語句……
'遍曆列表
Set n=nh
While Not n is Nothing
Response.Write n.Content+"<br />"
Set n=n.NextNode
Wend
%>
如果不加其它代碼,上面的運行結果是兩個“node”。VBScript的自訂類和對象也不外如是,只要你掌握基本的概念,對它有一定瞭解,就再簡單不過了。再次強調,用Set語句來對對象進行賦值,相當於Java裡的賦值,都是獲得一個引用。這比PHP4裡預設對象賦值是調用拷貝建構函式來建立一 個新的對象好多了(甚至連obj=new Obj;這樣的語句都會建立兩個對象!如果你想獲得引用的話,要在等號後變數前顯示地加上&),而似乎PHP5也不想修改PHP4的這種做法。
ASP中的Session本身是可以儲存對象的,它可以儲存基本變數,數組,Automation 物件(Automation Object)等,但在儲存自訂類的對象時會碰到問題。如下面的代碼:
<%
If isempty(Session("node")) Then Set Session("node")=New ListNode
Set n=Session("node")
Response.Write n.Content
%>
還是上面的ListNode這個類,這段代碼意圖在一個使用者會話中只保留一個ListNode的對象。所以在使用者第一次訪問該網頁時,會產生ListNode的一個對象,並儲存在Session(“node”)中;後面訪問該網頁時,因為Session(“node”)不為空白了,所以不會產生一個新的對象,而是到 Session(“node”)中取出儲存的對象。理論上應該也會輸出100,但是問題來了,ASP一直會報錯:
Microsoft VBScript runtime error '800a01b6'
Object doesn't support this property or method: 'n.Content'
用n.Type也會出錯。同樣的代碼翻譯成PHP,運行卻是可以通過的。為什嗎?
個人分析下來,認為Session可以儲存對象是沒錯,只是VBScript中類型轉換的機制太弱,而且沒有顯式的強制類型轉換供使用者使用,無法將 Session(“node”)正確轉換為ListNode類型。因為是自訂的類,我們只能在每個頁面中都出現類的定義語句,這樣在ASP看來,每次讀 取這個頁面時,ListNode類都是一個新類,所以就不認得Session中的這個類的對象了。
結論:盡量不要想到用Session或Application來儲存ASP中自訂類的對象。如果確實需要,可以考慮用COM來編寫類,然後在VBScript中用:Set Session("obj") = Server.CreateObject("YourApp.YourClass")來建立一個對象,然後即可實現上面預想的功能了。