从 PHP 迁移到 ASP.NET(三)

2017 年 6 月 6 日2720

表 9:Foreach 循环

PHP Visual Basic .NET
$i = 0;
foreach($a as $v) {

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

";

$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

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
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" %>

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://http://www.zjjv.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 相比,这些功能可以显著减少完成相同任务所需的代码数量。

gigi_miao

本文来源:AspCool

责任编辑:王晓易_NE0011

0 0