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 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")来创建一个对象,然后即可实现上面预想的功能了。