从 PHP 迁移到 ASP.NET
发布时间:2006-10-14 3:06:44   收集提供:gaoqian

简介

本文讨论了从 PHP (PHP:Hypertext Preprocessor 4) 到 ASP.NET 的迁移,并对 PHP 和 Microsoft® Visual Basic® .NET 的基本语法以及基本功能和体系结构进行了对比。

尽管使用 PHP 和 ASP.NET 都可以构建复杂的 Web 应用程序(例如电子商务站点、Intranet 和企业门户),但两者之间却存在若干重要差别。与 PHP 不同,ASP.NET 并不是一种语言或分析器,而是 Microsoft .NET Framework 中的一组用于构建 Web 应用程序和 XML Web Service 的技术。同 PHP 一样,Microsoft ASP.NET 页面在服务器上执行并生成标记(例如 HTML、WML 或 XML),标记将发送给桌面应用程序或移动应用程序。但不同的是,ASP.NET 为开发 Web 页面提供了强大的、面向对象的以及由事件驱动的编程模型,同时仍保留了 PHP 开发人员所习惯的那种简单性。

ASP.NET 应用程序是基于强大的面向对象编程 (OOP) 模式,而不是基于脚本模式。这样便可以获得更为严谨的 OOP 功能,例如继承、封装和反射。虽然大多数基本的简单操作可以轻松地从 PHP 转换到 ASP.NET,但要转换更为复杂的应用程序却并不这么简单,这需要仔细的规划和考虑,并采用更符合 OOP 的方法。

本文假设读者具有 PHP 方面的经验,并且对编程和软件开发有着基本了解。本文首先从代码方面对两者的基础体系结构的差别进行简单比较,并简要介绍一下 OOP 开发模型,然后比较两者的功能,最后将使用 PHP 和 ASP.NET 开发 Web 应用程序时的语法和常见任务作以对比。

注意:如果您希望跳过有关迁移的详细信息,而只想尝试 ASP.NET 的使用,则可以直接跳到建议的后续操作部分。

体系结构对比

阅读完本文后,通过比较语法和语言,您将了解 PHP 和 ASP.NET 是很相似的,它们具有类似的功能和语法。但是,从更基本的体系结构层面来讲,PHP 与 ASP.NET 却有着很大差别。PHP 基于独立于平台的处理器/引擎,该处理器/引擎用于分析 PHP 脚本、提供数据库连接、符合 Internet 协议以及提供大多数 Web 应用程序平台所常见的众多其他任务。

ASP.NET 是基于一系列技术(如 CLR)构建的框架,提供了大量以很好的方式组织的类库,这些类库可满足要在 Web 应用程序中使用的大多数功能的需要。您还可以轻松地创建各种组件,对该框架进行扩展。

虽然 PHP 也提供了类似的功能(例如 PEAR 库),但 PHP 和 ASP.NET 并非完全相同,因为 ASP.NET 框架是完全基于 OOP 模式和 OOP 概念构建的,而 PHP 则不然。这种差别在 PHP 和 ASP.NET 中类和对象的访问方式方面表现得犹为明显。

PHP 和 ASP.NET 中的面向对象编程

虽然 PHP 和 ASP.NET 都为应用程序开发提供了 OOP 模式,但它们对各种 OOP 概念(例如封装和多态)的支持却存在差别。例如,PHP 仅支持部分封装(例如,支持在类中声明方法和字段)和部分多态(没有重载和抽象)。此外,由于 PHP 的类中不存在私有 (private)、公有 (public) 或受保护 (protected ) 函数等概念以及重载,因此 PHP 也不支持这些概念及其访问。OOP 纯粹主义者可能会说,ASP.NET 以及各种语言也没有完全支持 OOP 模式中的所有概念,对于使用 OOP 的大多数语言(如 C++ 和 Java)来说,情况确实如此。

这既有积极的一面,也有消极的一面。消极的一面是:对于某些 Web 开发人员来说,学习 ASP.NET 要比学习 PHP 困难些,因为 PHP 提供的是开发人员创建 Web 站点时所惯用的脚本模式。但是,使用过 OOP 语言和/或 V 的开发人员会发现 ASP.NET 比较亲切且易于学习。

ASP.NET 支持 OOP 概念的积极的一面是,ASP.NET 应用程序的大多数部分都会得到很好的设计,并且能够对内容、逻辑和数据进行清楚的分隔,因而从应用程序生命周期的长远角度看,这通常更容易为其提供支持。此外,由于 ASP.NET 本身支持企业技术,例如消息队列、事务(请参阅 .NET Framework 的 System.EnterpriseServices 类)SNMP 和 Web 服务,因而可以很容易地开发具有高度可缩放性的强大的应用程序。

要大致了解面向对象编程(从 Visual Basic 的角度)的主要概念,请参阅 Object-Oriented Programming in Visual Basic

编译

PHP

请求 PHP 页面时,HTML 和内联 PHP 脚本将编译为 Zend Opcode(Zend 操作代码)。Opcode 是低级的二进制指令,是提供 PHP 页面时实际使用的代码。编译后,Zend 引擎将运行 Opcode(与 Java 的运行时引擎运行字节码类似),然后生成 HTML 并提供给客户端。

有许多商业产品可用来通过优化这些 Opcode 来加快 PHP 页面的执行速度。其他提高 PHP 脚本性能的方法还包括对 Opcode 和生成的 HTML 进行缓存。

ASP.NET

当向 IIS(Internet 信息服务)或其他 Web 服务器请求 .aspx 页面(或 ASP.NET 支持的任何其他扩展名)时,该请求将被传递到 ASP.NET 进行处理。如果是第一次请求该页面,ASP.NET 会将其编译为 MSIL(Microsoft 中间语言)。然后,CLR(公共语言运行库)会将此 MSIL 代码转换为机器码。最后使用这个编译后的代码运行该请求。后续的请求将使用相同的机器码运行(假设该页面未被修改)。

需要注意的是,因为由 CLR 生成的二进制代码已经最大限度地进行了优化,所以不必再使用其他产品来帮助提高性能。

此外还要注意,ASP.NET 中的所有内容在运行前都被编译为机器码。甚至 HTML 文本也被转换为字符串常值控件,并按照适当的顺序插入到控件树中。

功能对比

表 1 对 PHP 和 ASP.NET 中的某些主要功能进行了比较。

表 1:PHP 和 ASP.NET 的功能对比

功能 PHP ASP.NET
编码语言 具有旧 ASP 样式标记的 C、C++ 样式脚本语言。支持某些 OOP 概念。 支持超过 25 种语言,但最常用的两种语言是 Visual Basic .NET 和 C#。大多数开发人员都选择一种语言,但可以使用用任何其他支持的语言编写的组件。
编译的应用程序逻辑 可编译并且可以作为可执行程序运行。 在动态编译和预编译模式下均支持。
全部页面输出缓存 本身并不支持。 支持,基于一个或多个 URL 参数、浏览器类型、自定义函数或任意组合来缓存页面的不同版本。
部分页面输出缓存 本身并不支持。 通过使用用户控件提供内置支持。可使用缓存 API 和复杂的过期规则来缓存数据和其他对象。
数据库访问 为市场上的大多数数据库以及开放源代码的数据库提供了驱动程序。 直接支持 OLE-DB 和 ODBC,本身还包含了 Microsoft SQL Server™ 和 Oracle 的驱动程序。
数据库输出 数据集作为 PHP 变量返回,并且可以像任何其他变量那样将其输出。 以模板化的方式绑定到服务器端控件以简化开发;如果需要,也可以手动循环。
外部组件 可以调用各种软件包,并且开发人员可以直接修改 Zend 引擎(因为它是开放源代码的)。负责 Zend 的编程人员还可以使 .NET 对象能够从 PHP 中调用。 充分支持本身的 C 库和 COM 对象以及用任何与 .NET 兼容的语言(包括托管 C++ [Managed C++])编写的程序集。对 CORBA 对象或 Java 类未提供内置支持。
XML/XSLT 在附加软件包和库中支持 为 XML DOM、XSLT、验证和 XML 文档的轻量级面向流的分析提供了广泛和易于使用的支持。
XML Web Service 当前正在开发 PHP 软件包以支持基于 Apache AXIS 引擎以及其他引擎的 Web 服务。 当前版本提供了广泛、灵活、符合标准的支持,使得发布和使用 Web 服务都非常容易。
会话状态 基于 Cookie 的会话管理 基于 Cookie 或无 Cookie 的会话状态(使用单个服务器内存存储、中央状态服务器或数据库后端)。此外,可扩展的体系结构还允许使用自定义的会话状态模块替换内置选项。要启用无 Cookie 的会话,只需更改配置。
内置功能 PHP 的内置功能涵盖了基于 Web 的应用程序可能需要执行的许多常见任务。通过某些额外的处理,它还可以访问 Java 类库。 ASP.NET 可以直接访问整个 .NET Framework 类库,这些类库包含了大量功能。
正则表达式 支持 POSIX 和 Perl 兼容的正则表达式语法。 支持 Perl-5 兼容的正则表达式,以及一些其他功能,例如从右向左匹配、预编译表达式、命名组及完全的 Unicode 支持。此外,还允许用户指定在正则表达式替换操作过程中调用的函数。
调试 PHP 未提供广泛的调试功能,但可以通过各种第三方产品和 Zend 增强调试和测试功能。 包含广泛的跟踪和环境信息,这些信息可包含在页面中或在单独的页面中显示。使用 Microsoft Visual Studio® .NET 除了可以调试客户端脚本和 SQL Server 存储过程外,还可以轻松地在执行页面时对其进行交互式调试。
错误处理 不支持错误捕获,但提供了各种错误处理功能和日志。 支持结构化的异常处理(其中新增了一个“finally”代码块,无论是否发生错误,都将执行该块),支持引发自定义异常,还支持为不同类型的不可恢复错误指定自定义错误页面。
图像处理 无内置支持,但可以使用第三方组件。 包含大量用于创建和处理图像的工具。(请参阅 .NET Framework 的 System.Drawing 类。)
代码重用 用户定义的函数、创建类的功能以及包含文件。 用户控件、服务器控件、自定义类以及包含文件。
线程 PHP 具有很好的线程模型。 完全支持。不但可以在单独的线程中同时请求多个页面,而且每个页面(如果需要)还可以生成自己的线程以执行同步或异步操作。
数据缓存 具有有限的数据缓存(仅限于数据本身)。 包含大量的缓存 API,能够存储几乎任何类型的数据(包括数据库查询结果),并且能够基于时间、使用情况、文件或其他缓存项而过期。还允许在从缓存中删除给定项时调用用户定义的函数。
国际化 完全支持 Unicode。 支持 Unicode 字符串和各种字符编码。日期、数字和货币函数均可区分文化环境,并根据当前文化环境的设置更改输出(而不要求调用其他函数)。支持使用资源文件对应用程序进行动态本地化。
SMTP、HTTP、FTP、POP3 本身支持各种 Internet 协议。 SMTP 支持依赖于 IIS SMTP 服务。能够很好地支持 HTTP。.NET 不支持 FTP 和 POP3,但是有一些免费的和销售的第三方组件可用来实现此功能。
集成开发环境 可以使用大量提供了广泛功能的开发工具,价格从零到几百美元不等。 Microsoft 提供了一个名为 ASP.NET Web Matrix 的免费 ASP.NET 开发工具。

最常用的工具是 Visual Studio.NET,它为所有 .NET 语言、用于创建 SQL 以及测试数据库的数据库工具、Web 设计工具、与版本控制的集成、高级调试功能以及许多其他功能提供了充分的支持。要获得完整的功能列表,请访问 MSDN® Visual Studio Developer Center

其他工具(包括 Borland C# Builder 和 Macromedia Dreamweaver MX)也支持 ASP.NET。

Web 服务器支持 支持几乎所有 Web 服务器。 支持 IIS 以及 Covalent 的商业版 Apache 2.0。
操作系统支持 支持几乎所有常用的操作系统,包括 Microsoft Windows®、Mac、OS X、Amiga、Solaris、Free BSD、Linux 以及 AIX 等。 当前支持 Windows 2000、Windows XP 和 Windows Server 2003

语法和常见任务对比

以下各部分比较了 PHP 和 .NET 的语法,并介绍了如何完成某些非常常见的编程任务。

注释

PHP 允许您使用 C、C++ 和 Unix shell 样式语法在代码中插入注释,注释指示器中的任何内容都不会被执行。

一般来讲,要在 ASP.NET 中注释出 Visual Basic .Net 代码,需要使用 <%-- 打开注释块,并使用 --%> 关闭注释块。

表 2 显示了这两种环境中的注释。

表 2:服务器端注释

PHP ASP.NET
/*

这是被注释出的

文本块

*/

<%--

这是注释

--%>

变量

虽然 PHP 和 Visual Basic .NET 具有类似的语言构造,但语法却截然不同。由于 Visual Basic .NET 是基于 OOP 模型构建的,因而变量的声明要比 PHP 中变量的声明更严谨(在 PHP 中声明变量时只需在变量名之前添加一个 $ 符号)。

在 Visual Basic .NET 中声明变量时,要指定它的变量名和特性。变量的声明语句是 Dim 关键字。它的位置和内容决定了变量的特性。变量具有级别(例如局部变量和模块变量)、数据类型、生存期以及可访问性。

虽然这种方法最初看上去要比在 PHP 中指定变量复杂,但实际上它会给开发人员带来方便。ASP.NET 致力于帮助开发人员创建可靠的应用程序,而指定数据类型可以使诸如变量清理、调试、异常和错误处理以及代码维护等任务变得更加简单。

表 3 显示了在两种环境中声明变量的示例。

表 3:变量声明

PHP Visual Basic .NET
$head_count

$foo

$X

$obj

Dim head_count AS Integer

Dim foo As String

Dim X as Date

Dim Obj as object

声明数据类型

声明语句中的 AS 子句用于定义所声明变量的数据类型或对象类型。您可以为变量指定以下任意类型:

  • 基本数据类型,例如 BooleanLong Decimal
  • 复合数据类型,例如数组或结构
  • Visual Basic 或其他应用程序中的对象类型或类,例如 Label TextBox

您可以在一个语句中声明多个相同类型的变量而不必重复数据类型。下面的语句将变量 numStudentsnumGTA numProfessors 声明为 Integer 类型:

Dim numStudents, numGTA , numProfessors AS Integer  ' 三个变量都是 Integer 类型。

有关数据类型的详细信息,请参阅 Data Types。有关面向对象编程的详细信息,请参阅 Object-Oriented Programming in Visual Basic

声明生存期

变量的生存期是指变量可供使用的时间段。使用 Dim 语句声明的局部变量仅在其执行过程中存在。过程终止时,该过程的所有局部变量都将消失,这些变量的值也将丢失。

生存期这个概念很有用,因为它使开发人员在创建应用程序时不必考虑在大型应用程序中出现的许多问题,例如有效的内存管理。通过为变量选择正确的生存期,可以使 .NET 对未在使用的变量执行清理操作。

有关生存期的详细信息,请参阅 Lifetime

声明范围

局部变量是在过程(过程与函数类似)中声明的变量。非局部变量是在过程之外但在类或结构中声明的变量。

在类或结构中,非局部变量的类别取决于该变量是否被共享。使用 Shared 关键字声明的变量是共享变量,它只有一个副本并且在类或结构的所有实例中共享。如果不使用 Shared 关键字,则该变量为实例变量,并且针对类或结构的每个实例都将创建一个单独的变量副本。实例变量的给定副本仅可用于为之创建该副本的实例。

变量的范围是指可以引用它而不必限定其名称的所有代码集。变量的范围取决于声明变量的位置。位于给定区域中的代码可以使用在该区域中定义的变量而不必限定其名称。声明范围时,将遵循以下规则:

  • 共享变量或实例变量的范围是在其中声明了该变量的结构或类。
  • 局部变量的范围是在其中声明了该变量的过程。

但是,如果您在块中声明一个局部变量,则它的范围将只限于该块。局部变量在定义它的控制块中处于活动状态。控制块可以是过程、if 语句以及循环语句等。

有关范围的详细信息,请参阅 Scope

声明可访问性

.NET 支持变量的可访问性,从而使开发人员可以控制哪些代码能够访问特定的变量。例如,如果要为某个公式设置一些常数,并且要确保这些常数不会被它们所在类以外的其他代码更改,则可以使用 private 按如下所示声明该变量:

Private myConstant As Integer

变量的可访问性由声明语句中使用的一个或多个关键字 DimPublicProtectedFriendProtected Friend Private 确定。开发过程中通常只使用 public 和 private。

您可以使用上述任意关键字来声明模块、结构、类或实例变量。过程中只允许使用 Dim 关键字,并且可访问性始终为 private。

输出

PHP 中输出数据的典型方式是使用 echo() 语言构造。ASP.NET 中与此最相似的是 Response.Write() 方法(或 <%= %> 构造,它只是 Response.Write() 的简写形式)。表 4 显示了向页面写入值的基本语法。

表 4:基本输出

PHP Visual Basic .NET
<?php

$hello = "hi how are you\n";

echo $hello;

?

>

<%

Dim Hello As String = "Hi how are you" & vbcrlf

Response.Write(Hello)

%>

然而,这些用于向浏览器发送输出的方法主要是为了保持与传统 ASP 的向后兼容性。ASP.NET 提供了新的基于控件且面向事件的模型,您只需在服务器控件上设置属性即可将数据输出到浏览器中。通过这种技术可以将布局与代码清楚地分开,从而简化了维护工作,并且在复杂情况下所需的代码要比 PHP 少得多。

<script language="VB" runat="server">
Sub Page_Load(sender As Object, e As EventArgs)
   TheDate.Text = DateTime.Now 
End Sub
</script>
The current date is: <asp:Label id="TheDate" runat="server"/>

此示例声明了一个名为 TheDate 的服务器端标签 (Label) 控件,并在该页面的 Load 事件中将此标签的 Text 属性设置为当前的日期和时间。此代码的 HTML 输出与其他两个版本相同,不同的是 Label 控件将自身呈现为 span 标记(其中包含所设置的任何标签文本)。

条件处理

IF/ELSE

PHP 具有若干条件处理表达式,例如 forwhileswitch foreach,但最常用的是 if/else 表达式。Visual Basic .NET 有几个非常类似的构造,其语法也很类似。表 5 对 PHP 和 Visual Basic .NET 中的等价条件逻辑进行了比较。

表 5:基本条件逻辑

PHP Visual Basic .NET
if ($a > $b) {

print "a is bigger than b";

} elseif ($a == $b) {

print "a is equal to b";

} else {

print "a is smaller than b";

}

If a > b

Response.write ("a is bigger

than b")

ElseIf a = b Then

Response.write ("a is equal to

b")

Else

Response.write ("a is smaller

than b")

End If

切换

切换语句是大多数编程语言的常见语言构造,用于通过多个值对单个表达式进行测试。它常用于替换包含多个 elseif/else 块的 if 语句。

图 6 对 PHP 的 switch 语句与 Visual Basic 的 Select Case 语句进行了比较。

表 6:switch select case 语句的对比

PHP Visual Basic .NET
switch ($i) {

case 0:

print "i equals 0";

break;

case 1:

print "i equals 1";

break;

case 2:

print "i equals 2";

break;

default:

print "i is not equal to 0,

1 or 2";

}

Select Case Number i

Case 0

description = "0"

response.write ("i equals 0")

Case 1

description = "1"

response.write ("i equals 1")

Case 2

description = "2"

response.write ("i equals 2")

Case Else

description = " i is not equal

to 0, 1 or 2"

response.write ("i is not

equal to 0, 1 or 2 ")

End Select

循环

另一个非常常见的控制构造是循环。PHP 和 .NET 都支持若干不同的常见循环类型。

表 7:For 循环

PHP Visual Basic .NET
for ($i = 1; $i <= 100; $i++) {

print $i;

}

Dim sum As Integer = 0

Dim counter As Integer

For counter = 1 To 100 Step 5

sum += counter

Next

For i = 1 to 100

Response.write (i)

Next I

在 Visual Basic 中,这种循环类型被称为 For...Next 循环,而在 PHP 中则称为 For 循环。在此示例中,我们使用了 += 运算符作为 sum = sum + counter 的简写方式。在 PHP 中,可以使用 Break; 语句来中断循环。而 For...Next 循环可以通过 Exit For 语句中断。

条件循环

只要某个条件计算为 true,条件循环便会持续重复一组指令。表 8 显示了这两种语言中的基本条件循环示例。

表 8:条件循环

PHP Visual Basic .NET
$i = 1;

while ($i <= 10):

print $i;

$i++;

endwhile;

Dim counter i As Integer = 1

Do While counter i <= 10

Response.write(i)counter i += 1

Loop

在 Visual Basic 中,这种循环类型称为 Do...Loop 语句,或 while 循环。PHP 也支持 Do...While 循环,该循环与 while 循环非常类似,不同之处在于它是在每个循环的结尾而不是开始检查真假表达式。 Do...While 循环与常规 while 循环的主要差别是前者会确保运行第一个循环(因为只在该循环的结尾检查真假表达式),而常规的 while 循环则不一定运行。(因为它在每个循环的开始检查真假表达式,如果它一开始便计算为 FALSE,该循环将立即结束执行。)

下面给出了一个 PHP 中这种情况的示例:

$i = 0;
do {
   print $i;
} 
while ($i>0);

此循环只运行一次,因为在第一次循环后检查真假表达式时,表达式计算为 FALSE$i 不大于 0),所以循环将停止执行。

在 Visual Basic .NET 中,您可以执行同样的操作:

Dim counter i As Integer = 0
Do
   response.write (i)counter
Loop While counter i > 0

但是,Visual Basic .NET 支持内置的循环功能,这是 PHP 所不支持的。该功能将对条件进行计算,直到条件为真时为止。

Foreach 循环

PHP 4(而非 PHP 3)包含一个 foreach 构造,它与 ASP.NET 以及某些其他语言中的构造非常类似。它只是为循环数组提供了一个简单的途径。foreach 只适用于数组,如果试图将其用于其他数据类型的变量或未初始化的变量,则会导致错误。Visual Basic .NET 中与此等价的循环是 For Each...Next 语句。表 9 显示了在这两种语言中循环数组的示例。

表 9:Foreach 循环

PHP Visual Basic .NET
$i = 0;

foreach($a as $v) {

print "\$Key[$i]$v \n";

$i++;

}

for each v in a

response.write a(v) (v & vbcrlf)

next

数组

PHP 中的数组的功能与 Visual Basic .NET 中的数组的功能截然不同。PHP 中的数组实际上是关联数组,但是可以像索引或关联数组那样使用。而 Visual Basic .NET 中的数组是索引数组。Visual Basic .NET 不支持关联数组(但您可以自己创建它们,请参阅下面的“集合”)。其他 ASP.NET 语言支持这些类型的数组,但 Visual Basic .NET 并不支持,这可能给 PHP 开发人员提出了一个难题,因为他们并不习惯使用这种更常见的索引数组,并且可能会希望在 Visual Basic .NET 中创建关联数组模型。表 10 显示了 PHP 和 Visual Basic .NET 中的简单数组示例。

表 10:创建数组的示例

PHP Visual Basic .NET
$a = array (0,1, 2); Dim MySingleArraya() As Integer =

New Integer (2) {0,1,2}

与其他变量一样,Visual Basic .NET 数组中的变量也是使用 Dim 语句声明的。变量名后面带有一对或多对括号,表明它是数组而不是标量(包含单个值的变量)。

创建 Visual Basic .NET 数组时,还必须将其声明为保存特定类型的数据。如果指定的类型是 Object(作为 .NET 中所有对象类型的基础的普通类型),则此数组可以保存任何类型的数据,但是从数组中检索值时,必须将值转换为原来的类型。

Visual Basic .NET 数组可以是嵌套的数组,也可以是多维数组。Visual Basic .NET 中有许多与 PHP 相对应的用于处理数组的函数,但有一个例外。由于 Visual Basic .NET 不支持关联数组,因而不存在用于根据数组的“主键”来访问、索引或执行任何操作的函数。Visual Basic .NET 中不存在这样的“主键”。

虽然本文多次提到 Visual Basic .NET 不支持关联数组,但是可以创建集合来代替数组。集合与关联数组有些相似,可以使用集合来解决某些类似的问题。

在某些情况下,将项目存储在集合中要比存储在数组中更有效。

如果要处理一组小型的动态项目,则可以使用集合。要创建集合,只需如下面的示例代码所示声明并实例化一个 Collection

Dim myCollection As New Collection()

然后,您可以使用 Add 方法向集合中添加成员。在此示例中,我们创建了四个字符串并将它们添加到集合中。您还可以选择添加一个唯一的 String 值作为集合成员的主键。该值作为 Add 方法的第二个参数传递给集合。

Dim w, x, y, z As String
w = "key1"
x = "key2"
y = "key3"
z = "key4"
myCollection.Add(w, "1")
myCollection.Add(x, "2")
myCollection.Add(y, "3")
myCollection.Add(z, "4")

虽然这看起来类似于在 PHP 中创建关联数组,但集合是一种截然不同的事物,因为它本身就是一个对象。对于转到 ASP 的 PHP 开发人员,我们建议在尝试在 Visual Basic .NET 中创建关联数组模型之前先了解一下 Microsoft 的 Visual Basic .NET Language Specification

管理状态

任何 Web 应用程序中都很常见的一个任务就是管理状态,该任务通常是使用 Cookie 或应用程序状态管理构造(如 Session 变量)来完成的。Visual Basic .NET 具有与 PHP 类似的用于处理状态的方法。

设置和检索 Cookie

在这两种环境中设置 Cookie 都是比较繁琐的。表 11 显示了在这两种语言中编写然后读取 Cookie 的示例。

表 11:设置和检索 Cookie

PHP Visual Basic .NET
<?php

$value = 'something from

somewhere';

setcookie ("TestCookie",

$value,time()+3600); /*

expire in 1 hour */

?>

and to retive the set cookie

<?

echo $_COOKIE["TestCookie"];

?>

Dim value as string = "something from

somewhere"

Dim myCookie As New HttpCookie = New

HttpCookie("Something from

somewhereTestCookie")

Dim now as DateTime = DateTime.Now

myCookie.Value = now.ToString()value

myCookie.Expires = now.AddHour(1)

Response.Cookies.Add(myCookie)

'and to retrieve the set cookie

Response.Write(Request.Cookies["What we

setTestCookie"}.Value}

设置和检索 Session 变量

ASP.NET 中的 Session 变量与 PHP 中的 Session 变量非常类似。这两种环境中的 Session 变量都为确保 Web 应用程序访问中的一致性提供了相应处理和 Cookie 操作。

最重要的一个差别是,当从 ASP.NET 的 Session 对象中检索值时,该值将作为普通的 System.Object 类型返回,而这种类型可以保存任何类型的数据。要使用该值,必须先将其重新转换为原来的基本类型。表 12 显示了一些 Session 变量的使用示例。

表 12:Session 变量的使用

PHP Visual Basic .NET
<?PHP

session_start();

session_register('today');

$today = getdate();

?>

<?= $today ?>

Session("Today") = DateTime.Now

Dim today As Date

today = CDate(Session("Today"))

Response.Write(today)

Response.write (session("Today"))

ASP.NET 还提供了另一种形式的状态管理,即“应用程序状态”,它与 Session 变量类似,但只是在应用程序的生存期中保持一致。这使您可以存储各种内容,例如配置信息或数据库连接字符串。这些内容在应用程序运行期间不会发生变化。

有关此主题的详细信息,请参阅《.NET Framework Development Guide》中的 Application State 一节。

正则表达式

ASP.NET 支持其他正则表达式实现(例如 Perl 和 awk 中的正则表达式)的大多数常见功能。这实际上是为了保持与 Perl 5 正则表达式兼容。ASP.NET 还支持其他正则表达式实现中未提供的正则表达式功能,例如从右向左匹配以及实时编译。由于 ASP.NET 与 Perl 正则表达式兼容,并且大多数 PHP 开发人员都使用了 Perl 兼容的正则表达式,因而通常无须将语法从一种形式转换为另一种形式。有关 .NET 正则表达式支持的详细信息,请参阅 .NET Framework Regular Expressions

异常处理

ASP.NET 框架通过一种人们所熟悉的语言构造 Try/Catch 提供了结构化的异常处理,从而能够捕获代码中可能出现的异常。PHP 没有提供此功能,但 PHP 5 中将添加它。

下面的示例说明了如何在 Visual Basic .NET 中进行异常处理:

Try
   ' 此处是可能导致错误的代码
Catch e As ExceptionType
   ' 用于处理错误的代码
' 可选:可在此处放置更多 Catch 块
Finally
   ' 始终执行的代码
End Try

要注意的是,Try 块可以包含一个或多个 Catch 块,或包含一个 Finally 块,也可以同时包含两者。也就是说,如果您知道无法更正错误,但仍需要清除某些对象而不管是否发生错误,则可以使用不包含 Catch 语句的 Try...Finally 块。

查询数据库

在 PHP 中,通常可以通过两种常见方法访问数据库:使用数据库的特定扩展名或使用独立于数据库的 PEAR DB 库。

在 ASP.NET 中,可以通过一组称为 ADO.NET 的对象来访问数据库,这些对象与 PEAR DB 库的功能基本相同。数据库查询实际上是通过一组连接、命令、参数和数据适配器对象完成的。其中的每个对象都有多个版本,具体情况取决于所访问的数据库的类型。例如,有一组对象用于具有 OLE-DB 驱动程序的数据库(例如 Microsoft Access),还有一组对象用于具有 ODBC 驱动程序而没有 OLE-DB 驱动程序的数据库。针对 Oracle 和 Microsoft SQL Server 也有专门的数据提供者,它们已经过优化,可以分别为每种特定的数据库提供高性能的访问。某些第三方还提供了其他数据库支持,例如 MySQL。本节中的示例将使用 SQL Server 对象,因为它是 ASP.NET 最常用的数据库。

System.Data、System.Data.SqlClient 和 System.Data.oledb 是用于在 ADO.NET 中定义数据库访问的命名空间。要使页面能够访问这些类,您需要将 System.Data System.Data.SqlClient 命名空间导入到页面中。

<%@ Import Namespace="System.Data" %>
<%@ Import Namespace="System.Data.SqlClient" %>

图 13 显示了在这两种语言中执行查询的示例。在 PHP 中,我们显示了一个使用 PEAR 的连接,PEAR 不仅是用于连接 DB 的最常用方法之一,但还是与 ADO.NET 最为类似的方法。

表 13:执行查询

PHP Visual Basic .NET
<?php

//connect

require_once('DB.PHP');

$db=DB::connect

("mysql://mydbvie

w:user@localhost/mydb");

if (DB::iserror($db)) {

die($db->getMessage());

$sql = "select * from mytable";

$q= $db->query($sql);

if (DB::iserror($q)) {

die($q->getMessage());

}

<tr>

<td><?= $row[0] ?></td>

<td><?= $row[1] ?></td>

<td><?= $row[2] ?></td>

</tr>

<script runat="server">

Sub Page_Load(Sender As Object, E As

EventArgs)

Dim myConnection As New

SqlConnection("server=(local)\NetSDK;

database=mydb;Trusted_Connection

=yes")

Dim myCommand As New

SqlDataAdapter("select * from

mytable", myConnection)

Dim ds As New DataSet()

myCommand.Fill(ds, "myDataset")

myDataGrid.DataSource = myDataset

myDataGrid.DataBind()

End Sub

</script>

<%--- 输出结果 ---%>

<form runat="server">

<asp:DataGrid id="myDataGrid"

runat="server" />

</form>

在 PHP 中,查询结果存储在一个称为 result set(结果集)的变量中,而在 ADO.NET 中,该变量称为 Dataset 对象。结果集是返回数据的只读视图,而 .NET Dataset 实际上是返回数据在内存中的读写视图,这使 .NET 开发人员能够轻松地处理从数据源返回的数据。

输出数据时,ASP.NET 提供了若干方法,用于向用户或客户端显示数据。第一种方法与 PHP 类似,即使用 SQLDataReader 对象遍历结果集以写出我们要通过查询显示的数据。更常用的方法是 ASP.NET 的数据绑定(PHP 中没有与此对应的方法)。通过该方法,开发人员可以构建用户界面并显示可在应用程序中使用和重用的控件,还可以进一步根据数据和逻辑提取显示内容。数据绑定的语法很灵活,您不仅可以绑定到数据源,还可以绑定到简单的属性、集合、表达式,甚至是方法调用返回的结果。

要使用数据绑定,您需要向某个数据识别服务器控件(例如 DataGrid)的 DataSource 属性指定某个数据源(例如查询结果)。您还可以选择为每个列提供一些附加的格式化信息并调用 DataBind() 方法。服务器控件将负责完成剩余的工作。

例如,在表 13 中,我们使用了数据绑定语法输出查询结果,如下所示:

<%--- 在页面中加载事件
---%>
myDataGrid.DataSource = myDataset
myDataGrid.DataBind()
<%--- 输出结果 ---%>
<asp:DataGrid id="myDataGrid"
   runat="server" />

数据识别服务器控件将提供其他功能,例如支持对显示的数据进行分页或内联编辑。有关详细信息和示例,请参阅 Data Binding Server Controls

数据缓存和页面缓存

对经常访问的数据进行缓存可以显著缩短站点的响应时间,因为不需要等待数据库查询便可进行页面处理。而缓存由页面请求生成的 HTML 可以进一步缩短响应时间,因为根本就无需处理缓存的页面。这两种环境均支持上述两种缓存策略,但与 PHP 相比,ASP.NET 提供了更多用于缓存和管理数据的方法,从而使开发人员可以针对自己应用程序的性能需要选择相应的方法和策略。

页面缓存

缓存某个页面请求的 HTML 输出是减少 Web 应用程序的负载的一种常用方法。PHP 本身不支持页面缓存,但是可以通过编程的方式或下载第三方软件包来执行此操作。通常可以在服务器上通过多种方法执行页面缓存,这些方法可用于缓存编译的代码,直至将页面输出写出到一个单独的文件中(该文件将随代码的更新而更新)。

在 ASP.NET 中,可以通过低级别的 OutputCache API 或高级别的 @ OutputCache 指令来执行页面缓存。启用输出缓存后,将在对页面进行第一个 GET 请求时创建输出缓存条目。后续的 GET HEAD 请求将由输出缓存条目支持,直到缓存的请求过期。

输出缓存将检查页面的过期策略和验证策略。如果页面位于输出缓存中并标记了过期策略(指示该页面自缓存时开始 60 分钟后过期),则 60 分钟后,该页面将从输出缓存中删除。如果在过期后收到另一个请求,将执行页面代码并再次缓存该页面。这种过期策略称为绝对过期,即页面在某个时间之前有效。

除了对整个页面进行输出缓存外,ASP.NET 还提供了一个简单方法,使您可以只缓存页面的特定部分。这种方法称为拆分缓存。您可以使用用户控件 (User Control) 来指定页面区域,并使用前面介绍的 @ OutputCache 指令将这些区域标记为缓存区域。该指令指定了用户控件的输出内容在服务器上缓存的时间(以秒为单位)以及影响这一时间的任何可选条件。

有关 ASP.NET 的输出缓存的详细信息,请参阅 Caching ASP.NET Pages

数据缓存

在 PHP 中,可以通过多种方法以编程方式将查询结果缓存到环境中,但 PHP 本身并未提供这种方法。在 PHP 中,可以创建数据缓存类或系统。对于少量信息,只需使用 Session 变量和/或 Cookie;对于大量或复杂的信息,可以创建您自己的数据缓存类。但问题是,当处理包含很多类型的复杂数据时,这种方法的效率会很低且容易出错,在一定程度上会加大编程的难度。

ASP.NET 通过 Page.Cache 对象提供了一个系统范围的数据缓存方法,可用于数据集 (DataSet)、数组、集合、XML 对象等。对于需要更复杂功能的应用程序,ASP.NET 缓存还支持三种特定类型的缓存:过期、清理以及文件和关键项相关性。

  • 过期类型的缓存使开发人员可以控制缓存项目的过期时间。可以将该时间定义为特定的时间(例如 01:00),也可以将其定义为相对于上次使用项目时的时间,例如自上次访问该项目时开始 20 分钟后过期。过期后的项目将从缓存中删除,以后再试图检索该项目时,将返回空值(除非再次将其插入到缓存中)。
  • 清理类型的缓存会在内存不足时尝试删除不经常使用或不重要的项目。开发人员可以控制清理的执行方式,并且可以在将项目插入到缓存中时向清理程序提供提示,以指示创建该项目的相对系统开销以及为使该项目有效而必须对其进行访问的相对频率。
  • 文件和关键项相关性类型的缓存将基于某个外部文件或另一个缓存项目确定该缓存项目的有效性。如果相关项发生变化,缓存项目将失效并从缓存中删除。例如,假设您有一个较大的需要定期更新并下载给员工的报告,则可以在应用程序中使用此功能。应用程序处理文件和报告中的数据,然后缓存该数据并插入对该数据的来源文件(从该文件读取数据)的相关性。当该文件更新时,数据将从缓存中删除,应用程序可以重新读取它并重新插入更新的数据副本。

ASP.NET 数据缓存为编程人员提供了许多不同的方法来管理应用程序并提高应用程序的响应速度和效率。有关详细信息,请参阅 Cache Class 文档以了解 Cache 对象。

发送电子邮件

PHP 和 ASP.NET 都为以编程方式发送电子邮件提供了内置的支持。要在此示例中使用 ASP.NET 发送电子邮件,您需要设置 IIS SMTP 服务。因为 .NET 中的内置邮件对象依赖于该服务包含的对象,所以您必须安装它。尽管如此,与 PHP 一样,.NET 也允许您使用任何 SMTP 服务器或邮件服务器。表 14 对这两种环境的基本语法进行了比较。

表 14:发送电子邮件

PHP Visual Basic .NET
$to = "test@atnoaddress.com";

$from =

"me@nosuchaddress.com";

$subject = "hi";

$message = "just wanted to say

hi";

mail($to,$subject,$message,

$from)

Dim myMail As MailMessage = New

MailMessage()

myMail.From = "me@nosuchaddress.com"

myMail.To = "test@atnoaddress.com"

myMail.Subect = "hi"

myMail.Body = "just wanted to say hi"

SmtpMail.Send(myMail)

XML 处理和 Web 服务

在 PHP 中,为分析和处理 XML 所提供的内置支持功能非常薄弱。虽然开发人员可以用它来分析和处理 XML,但它不支持 DOM 分析(该分析尽管比 PHP 的 SAX 分析器速度慢,但更容易使用)。PHP 也不支持根据 DTD 或 XML SCHEME 对 XML 文档进行本地验证,它也不支持 XSL/XSLT 以及市场上许多 Web 应用程序常用的各种其他技术。虽然 PHP 可以通过很多 PHP 软件包完成许多与 XML 相关的任务,但 .NET 和 ASP.NET 为处理 XML 提供了广泛的内置支持。XML 是 .Net 平台的核心技术之一。有关 Web 服务的详细信息,请参阅 How ASP.NET Web Services Work

.NET Framework 为 W3C 定义的所有 XML 建议提供了广泛的支持,此外还支持 XSL/XSLT、XPath、XQuery 以及各种其他技术,例如用于 Web 服务的 UDDI、WSDL 以及 SOAP。

虽然可以在 PHP 中创建 XML-RPC 类型的机制,但创建 Web 服务(使开发人员能够使用常用协议和标准来交换数据和过程并将其用于查找、数据绑定和描述)。.NET 为 Web 服务和相关技术(如 SOAP、WSDL 和 UDDI)提供了广泛的支持。使用 .NET,开发人员还可以轻松地创建和开发 Web 服务。例如,以下代码将创建一个简单的 hello world Web 服务:

<%@ WebService Language="VB" Class="HelloWorld" %>

Imports System
Imports System.Web.Services

Public Class HelloWorld :Inherits WebService

     <WebMethod()> Public Function SayHelloWorld() As String
          Return("Hello World")
     End Function

End Class

利用 .NET Framework SDK,您可以使用命令行 Web 服务描述语言工具 (WSDL.exe) 生成代理类。要为上述示例创建一个名为 HelloWorld.cs 的代理类,可以输入:

WSDL http://someDomain.com/someFolder/HelloWorld.asmx?WSDL

此类与前面创建的类非常相似。它包含一个名为 SayHelloWorld 的方法,用于返回字符串。将此代理类编译到应用程序中,然后调用此代理类的方法,使该代理类通过 HTTP 将 SOAP 请求打包并接收 SOAP 编码的响应,该响应随后被封装为字符串。

对于客户端,该代码是非常简单的,如以下示例所示:

Dim myHelloWorld As New HelloWorld()
Dim sReturn As String = myHelloWorld.SayHelloWorld()

这就是创建一个简单 Web 服务的全过程。有关 XML 的一般信息以及 Web 服务的特定信息,请参阅 Employing XML in the .NET Framework

小结

对于简单的小型应用程序,从 PHP 迁移到 ASP.NET 通常并不复杂。对于更复杂的应用程序,由于基础体系结构的不同以及 ASP.NET 的 OOP 模式,需要进行事先规划和精心设计,以利用 ASP.NET 提供的将显示与逻辑和数据明确分开的功能,以及可以节省时间的内置功能。与 PHP 相比,这些功能可以显著减少完成相同任务所需的代码数量。

 
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50