ASP面向对象编程探讨及比较
【
信息来源】 发布日期:
5-8 10:30:00 文章分类:财经资讯
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中也差不多,声明时加上类型就OK了。而使用时:
<%
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 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")来建立一个对象,然后即可实现上面预想的功能了。