存档在 ‘后端技术’ 分类

如何通过Etag和Expires控制网页缓存

2008年10月13日

题记:本文对页面中Etag和Expires标识处理,使得页面更加有效被Cache。

摘要

1、Etag和Expires中Client 端Http Request Header及Server端Http Reponse Header工作原理。
2、静态下Apache、Lighttpd和Nginx中Etag和Expires配置
3、非实时交互动态页面中Etag和Expires处理

在客户端通过浏览器发出第一次请求某一个URL时,根据 HTTP 协议的规定,浏览器会向服务器传送报头(Http Request Header),服务器端响应同时记录相关属性标记(Http Reponse Header),服务器端的返回状态会是200,格式类似如下:

HTTP/1.1 200 OK

Date: Tue, 03 Mar 2009 04:58:40 GMT

Content-Type: image/jpeg

Content-Length: 83185

Last-Modified: Tue, 24 Feb 2009 08:01:04 GMT

Cache-Control: max-age=2592000

Expires: Thu, 02 Apr 2009 05:14:08 GMT

Etag: “5d8c72a5edda8d6a:3239″

客户端第二次请求此URL时,根据 HTTP 协议的规定,浏览器会向服务器传送报头(Http Request Header),服务器端响应并记录相关记录属性标记文件没有发生改动,服务器端返回304,直接从缓存中读取:

HTTP/1.x 304 Not Modified

Date: Tue, 03 Mar 2009 05:03:56 GMT

Content-Type: image/jpeg

Content-Length: 83185

Last-Modified: Tue, 24 Feb 2009 08:01:04 GMT

Cache-Control: max-age=2592000

Expires: Thu, 02 Apr 2009 05:14:08 GMT

Etag: “5d8c72a5edda8d6a:3239″

其中Last-Modified、Expires和Etag是标记页面缓存标识

一、Last-Modified、Expires和Etag相关工作原理

1、Last-Modified

在浏览器第一次请求某一个URL时,服务器端的返回状态会是200,内容是你请求的资源,同时有一个Last-Modified的属性标记(Http Reponse Header)此文件在服务期端最后被修改的时间,格式类似这样:

Last-Modified: Tue, 24 Feb 2009 08:01:04 GMT

客户端第二次请求此URL时,根据 HTTP 协议的规定,浏览器会向服务器传送 If-Modified-Since 报头(Http Request Header),询问该时间之后文件是否有被修改过:

If-Modified-Since: Tue, 24 Feb 2009 08:01:04 GMT

如果服务器端的资源没有变化,则自动返回 HTTP 304 (Not Changed.)状态码,内容为空,这样就节省了传输数据量。当服务器端代码发生改变或者重启服务器时,则重新发出资源,返回和第一次请求时类似。从而保证不向客户端重复发出资源,也保证当服务器有变化时,客户端能够得到最新的资源。

注:如果If-Modified-Since的时间比服务器当前时间(当前的请求时间request_time)还晚,会认为是个非法请求

2、Etag工作原理

HTTP 协议规格说明定义ETag为“被请求变量的实体标记” (参见14.19)。简单点即服务器响应时给请求URL标记,并在HTTP响应头中将其传送到客户端,类似服务器端返回的格式:

Etag: “5d8c72a5edda8d6a:3239″

客户端的查询更新格式是这样的:

If-None-Match: “5d8c72a5edda8d6a:3239″

如果ETag没改变,则返回状态304。

即:在客户端发出请求后,Http Reponse Header中包含 Etag: “5d8c72a5edda8d6a:3239″
标识,等于告诉Client端,你拿到的这个的资源有表示ID:5d8c72a5edda8d6a:3239。当下次需要发Request索要同一个URI的时候,浏览器同时发出一个If-None-Match报头( Http Request Header)此时包头中信息包含上次访问得到的Etag: “5d8c72a5edda8d6a:3239″标识。

If-None-Match: “5d8c72a5edda8d6a:3239“

,这样,Client端等于Cache了两份,服务器端就会比对2者的etag。如果If-None-Match为False,不返回200,返回304 (Not Modified) Response。

3、Expires

给出的日期/时间后,被响应认为是过时。如Expires: Thu, 02 Apr 2009 05:14:08 GMT

需和Last-Modified结合使用。用于控制请求文件的有效时间,当请求数据在有效期内时客户端浏览器从缓存请求数据而不是服务器端. 当缓存中数据失效或过期,才决定从服务器更新数据。

4、Last-Modified和Expires

Last-Modified标识能够节省一点带宽,但是还是逃不掉发一个HTTP请求出去,而且要和Expires一起用。而Expires标识却使得浏览器干脆连HTTP请求都不用发,比如当用户F5或者点击Refresh按钮的时候就算对于有Expires的URI,一样也会发一个HTTP请求出去,所以,Last-Modified还是要用的,而 且要和Expires一起用。

5、Etag和Expires

如果服务器端同时设置了Etag和Expires时,Etag原理同样,即与Last-Modified/Etag对应的Http Request Header:If-Modified-Since和If-None-Match。我们可以看到这两个Header的值和Web Server发出的Last-Modified,Etag值完全一样;在完全匹配If-Modified-Since和If-None-Match即检查完修改时间和Etag之后,服务器才能返回304.

6、Last-Modified和Etag

Last-Modified 和ETags请求的http报头一起使用,服务器首先产生 Last-Modified/Etag标记,服务器可在稍后使用它来判断页面是否已经被修改,来决定文件是否继续缓存

过程如下:

1. 客户端请求一个页面(A)。

2. 服务器返回页面A,并在给A加上一个Last-Modified/ETag。

3. 客户端展现该页面,并将页面连同Last-Modified/ETag一起缓存。

4. 客户再次请求页面A,并将上次请求时服务器返回的Last-Modified/ETag一起传递给服务器。

5. 服务器检查该Last-Modified或ETag,并判断出该页面自上次客户端请求之后还未被修改,直接返回响应304和一个空的响应体。

注:

1、Last-Modified和Etag头都是由Web Server发出的Http Reponse Header,Web Server应该同时支持这两种头。

2、Web Server发送完Last-Modified/Etag头给客户端后,客户端会缓存这些头;

3、客户端再次发起相同页面的请求时,将分别发送与Last-Modified/Etag对应的Http Request Header:If-Modified-Since和If-None-Match。我们可以看到这两个Header的值和Web Server发出的Last-Modified,Etag值完全一样;

4、通过上述值到服务器端检查,判断文件是否继续缓存;

二、Apache、Lighttpd和Nginx中针配置Etag和Expires,有效缓存纯静态如css/js/pic/页面/流媒体等文件。

A、Expires

A.1、Apache Etag

使用Apache的mod_expires 模块来设置,这包括控制应答时的Expires头内容和Cache-Control头的max-age指令

ExpiresActive On
ExpiresByType image/gif “access plus 1 month”

ExpiresByType image/jpg “access plus 1 month”

ExpiresByType image/jpeg “access plus 1 month”
ExpiresByType image/x-icon “access plus 1 month”

ExpiresByType image/bmp “access plus 1 month”
ExpiresByType image/png “access plus 1 month”
ExpiresByType text/html “access plus 30 minutes”
ExpiresByType text/css? “access plus 30 minutes”

ExpiresByType text/txt? “access plus 30 minutes”
ExpiresByType text/js?? ”access plus 30 minutes”
ExpiresByType application/x-javascript?? ”access plus 30 minutes”
ExpiresByType application/x-shockwave-flash???? ”access plus 30 minutes”

<ifmodule mod_expires.c>

<filesmatch “\.(jpg|gif|png|css|js)$”>

ExpiresActive on

ExpiresDefault “access plus 1 year”

</filesmatch>

</ifmodule>

当设置了expires后,会自动输出Cache-Control 的max-age 信息

具体关于 Expires 详细内容可以查看Apache官方文档

在这个时间段里,该文件的请求都将直接通过缓存服务器获取,当然如果需要忽略浏览器的刷新请求(F5),缓存服务器squid还需要使用 refresh_pattern 选项来忽略该请求

refresh_pattern -i \.gif$ 1440 100% 28800 ignore-reload

refresh_pattern -i \.jpg$ 1440 100% 28800 ignore-reload

refresh_pattern -i \.jpeg$ 1440 100% 28800 ignore-reload

refresh_pattern -i \.png$ 1440 100% 28800 ignore-reload

refresh_pattern -i \.bmp$ 1440 100% 28800 ignore-reload

refresh_pattern -i \.htm$ 60 100% 100 ignore-reload

refresh_pattern -i \.html$ 1440 50% 28800 ignore-reload

refresh_pattern -i \.xml$ 1440 50% 28800 ignore-reload

refresh_pattern -i \.txt$ 1440 50% 28800 ignore-reload

refresh_pattern -i \.css$ 1440 50% 28800 reload-into-ims

refresh_pattern -i \.js$ 60 50% 100 reload-into-ims

refresh_pattern . 10 50% 60

有关Squid中Expires的说明,请参考Squid官方中refresh_pattern介绍。

A.2、Lighttpd Expires

和Apache一样Lighttpd设置expire也要先查看是否支持了mod_expire模块,

下面的设置是让URI中所有images目录下的文件1小时后过期;

expire.url = ( “/images/” => “access 1 hours” )

下面是让作用于images目录及其子目录的文件;

$HTTP["url"] =~ “^/images/” {

expire.url = ( “” => “access 1 hours” )

}

也可以指定文件的类型;

$HTTP["url"] =~ “\.(jpg|gif|png|css|js)$” {

expire.url = ( “” => “access 1 hours” )

}

具体参考Lighttpd官方Expires解释

A.3、Nginx中Expires

location ~ .*\.(gif|jpg|jpeg|png|bmp|swf)$

{

expires 30d;

}

location ~ .*\.(js|css)?$

{

expires 1h;

}

这类文件并不常修改,通过 expires 指令来控制其在浏览器的缓存,以减少不必要的请求。 expires 指令可以控制 HTTP 应答中的“ Expires ”和“ Cache-Control ”的头标(起到控制页面缓存的作用)。其他请参考Nginx中Expires

B.1、Apache中Etag设置

在Apache中设置Etag的支持比较简单,只用在含有静态文件的目录中建立一个文件.htaccess, 里面加入:

FileETag MTime Size

这样就行了,详细的可以参考Apache的FileEtag文档页

B.2、Lighttpd Etag

在Lighttpd中设置Etag支持:

etag.use-inode: 是否使用inode作为Etag

etag.use-mtime: 是否使用文件修改时间作为Etag

etag.use-size: 是否使用文件大小作为Etag

static-file.etags: 是否启用Etag的功能

第四个参数肯定是要enable的, 前面三个就看实际的需要来选吧,推荐使用修改时间

B.3、 Nginx Etag

Nginx中默认没有添加对Etag标识. Igor Sysoev的观点”在对静态文件处理上看不出如何Etag好于Last-Modified标识。”

Note:

Yes, it’s addition,and it’s easy to add, however, I do not see how ETag is better than Last-Modified for static files. -Igor Sysoev

A nice short description is here:

http://www.mnot.net/cache_docs/#WORK

It looks to me that it makes some caches out there to cache the response from the origin server more reliable as in rfc2616 (ftp://ftp.rfc-editor.org/in-notes/rfc2616.txt) is written.

3.11 Entity Tags 13.3.2 Entity Tag Cache Validators 14.19 ETag

当然也有第三方nginx-static-etags 模块了,请参考

http://mikewest.org/2008/11/generating-etags-for-static-content-using-nginx

三、对于非实时交互动态页面中Epires和Etag处理

对数据更新并不频繁、如tag分类归档等等,可以考虑对其cache。简单点就是在非实时交互的动态程序中输出expires和etag标识,让其缓存。但需要注意关闭session,防止http response时http header包含session id标识;

3.1、Expires

如expires.php

<?php

header(’Cache-Control: max-age=86400,must-revalidate’);
header(’Last-Modified: ‘ .gmdate(’D, d M Y H:i:s’) . ‘ GMT’ );
header(”Expires: ” .gmdate (’D, d M Y H:i:s’, time() + ‘86400′ ). ‘ GMT’);

?>

以上信息表示该文件自请求后24小时后过期。

其他需要处理的动态页面直接调用即可。

3.2、Etag

根据Http返回状态来处理。当返回304直接从缓存中读取

如etag.php

<?php

cache();

echo date(”Y-m-d H:i:s”);

function cache()

{

$etag = “http://longrujun.name”;

if ($_SERVER['HTTP_IF_NONE_MATCH'] == $etag)

{

header(’Etag:’.$etag,true,304);

exit;

}

else header(’Etag:’.$etag);

}

?>

通过ADO连接各种数据库的字符串翠集

2006年6月11日

在网络编程过程中,ADO(Activex Data Object)编程往往是不可少的工作,特别是开发电子商务网站。既然讲到ADO数据对象,那么就顺便简单地介绍一下ADO数据对象及其功能。ADO数据对象共有七种独立的对象,他们分别是连接对象(Connection)、记录集对象(RecordSet)、域对象(Field)、命令对象(Command)、参数对象(Parameter)、属性对象(Property)和错误对象(Error)。功能请参见附录1。 
ADO数据对象用来连接数据库的方式有两种,分别是ODBC和OLE DB方式,下面就分别介绍用这两种方式连接数据库的例子。在网络编程过程中,ADO(Activex Data Object)编程往往是不可少的工作,特别是开发电子商务网站。既然讲到ADO数据对象,那么就顺便简单地介绍一下ADO数据对象及其功能。ADO数据对象共有七种独立的对象,他们分别是连接对象(Connection)、记录集对象(RecordSet)、域对象(Field)、命令对象(Command)、参数对象(Parameter)、属性对象(Property)和错误对象(Error)。功能请参见附录1。
ADO数据对象用来连接数据库的方式有两种,分别是ODBC和OLE DB方式,下面就分别介绍用这两种方式连接数据库的例子。
第一种:ODBC DSN-Less Connections
ODBC Driver for Access
ODBC Driver for dBASE
ODBC Driver for Excel
ODBC Driver for MySQL
ODBC Driver for Oracle
ODBC Driver for Paradox
ODBC Driver for SQL Server
ODBC Driver for Sybase
ODBC Driver for Sybase SQL Anywhere
ODBC Driver for Text
ODBC Driver for Teradata
ODBC Driver for Visual FoxPro
第二种:OLE DB Data Provider Connections
OLE DB Provider for Active Directory Service
OLE DB Provider for DB2
OLD DB Provider for Internet Publishing
OLE DB Provider for Index Server
OLE DB Provider for Microsoft Jet
OLE DB Provider for ODBC Databases
OLE DB Provider for Oracle (From Microsoft)
OLE DB Provider for Oracle (From Oracle)
OLE DB Provider for Simple Provider
OLE DB Provider for SQL Server
一、下面我就先讲利用ODBC DSN来访问数据库,要完成这项工作,必须先通过“控制面板”—“管理工具”—“ODBC数据源”来建立一个数据源名,例如叫做TestData_Resoure。
1) 通过系统数据源(System DSN)的连接
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”DSN=TestData_Resoure;” & _
“Uid=AdminAccount;” & _
“Pwd=PassWord;
oConn.Open strConn
%>
2) 通过文件数据源(File DSN)的连接
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”FILEDSN=c:\somepath\mydb.dsn;” & _
“Uid=AdminAccount;” & _
“Pwd=PassWord;”
oConn.Open strConn
%>
3) 通过连接池(DSN-Less)的连接(ODBC Driver for AS/400)
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
strConn= “Driver={Client Access ODBC Driver (32-bit)};” & _
“System=myAS400;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
oConn.Open strConn
%>
二、利用数据库驱动程序直接访问数据库的连接字符串。
1) ODBC Driver for Access
● 标准的也是比较常用的连接方法
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={Microsoft Access Driver (*.mdb)};” & _
“Dbq=”&Server.MapPath(“Testdb.mdb”); & _
“Uid=AdminAccount;” & _
“Pwd=Password;”
oConn.Open strConn
%>
● 假如是一个工作组的系统数据库,那么连接字符串如下
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={Microsoft Access Driver (*.mdb)};” & _
“Dbq=c:\datapath\Testdb.mdb;” & _
“SystemDB=c:\datapath\Testdb.mdw;”, _
“admin”, “”
oConn.Open strConn
%>
● 假如数据库(MDB)是网络上共享的,那么连接字符串如下
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={Microsoft Access Driver (*.mdb)};” & _
“Dbq=\\myServer\myShare\myPath\Testdb.mdb;”
oConn.Open strConn
%>
2) ODBC Driver for dBASE
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={Microsoft dBASE Driver (*.dbf)};” & _
“DriverID=277;” & _
“Dbq=c:\FilePath;”
oConn.Open strConn
%>
说明:这里要注意的一点就是,在SQL查询语句中要特别指定数据库文件名,例如:
oRs.Open “select * From Testdb.dbf”, oConn, , ,adCmdText
3) ODBC Driver for Excel
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={Microsoft Excel Driver (*.xls)};” & _
“DriverId=790;” & _
“Dbq=c:\filepath\myExecl.xls;”
oConn.Open strConn
%>
4) ODBC Driver for MySQL (通过 MyODBC驱动程序)
● 连接到本地数据库(local database)
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={mySQL};” & _
“Server=ServerName;” & _
“Option=16834;” & _
“Database=mydb;”
oConn.Open strConn
%>
● 连接远程数据库(remote databas)
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={mySQL};Server=db1.database.com;Port=3306;” & _
“Option=131072;Stmt=;Database=mydb;Uid=myUsername;Pwd=myPassword;”
oConn.Open strConn
%>
5) ODBC Driver for Oracle
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={Microsoft ODBC for Oracle};” & _
“Server=OracleServer.world;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
oConn.Open strConn
%>
6) ODBC Driver for Paradox
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={Microsoft Paradox Driver (*.db)};” & _
“DriverID=538;” & _
“Fil=Paradox 5.X;” & _
“DefaultDir=c:\dbpath\;” & _
“Dbq=c:\dbpath\;” & _
“CollatingSequence=ASCII;”
oConn.Open strConn
%>
7) ODBC Driver for SQL Server
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn=”Driver={SQL Server};” & _
“Server=MyServerName;” & _
“Database=myDatabaseName;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
oConn.Open strConn
%>
8) ODBC Driver for Sybase (通过Sybase System 11 ODBC Driver驱动程序)
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={SYBASE SYSTEM 11};” & _
“Srvr=myServerName;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
oConn.Open strConn
%>
9) ODBC Driver for Sybase SQL Anywhere
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “ODBC; Driver=Sybase SQL Anywhere 5.0;” & _
“DefaultDir=c:\dbpath\;” & _
“Dbf=c:\sqlany50\mydb.db;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
“Dsn=”"”";”
oConn.Open strConn
%>
10) ODBC Driver for Teradata
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Provider=Teradata;” & _
“DBCName=MyDbcName;” & _
“Database=MyDatabaseName;” & _
“Uid=myUsername;” & _
“Pwd=myPassword;”
oConn.Open strConn
%>
11) ODBC Driver for Text
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={Microsoft Text Driver (*.txt; *.csv)};” & _
“Dbq=c:\somepath\;” & _
“Extensions=asc,csv,tab,txt;” & _
“Persist Security Info=False”
oConn.Open strConn
%>
12) ODBC Driver for Visual FoxPro
●使用数据库容器(database container)连接方式
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={Microsoft Visual FoxPro Driver};” & _
“SourceType=DBC;” & _
“SourceDB=c:\somepath\mySourceDb.dbc;” & _
“Exclusive=No;”
oConn.Open strConn
%>
● 不使用数据库容器(database container)连接方式(即Free Table Directory方式)
<%
<%
Dim oConn,strConn
Set oConn=Server.createObject(“ADODB.Connection”)
StrConn= “Driver={Microsoft Visual FoxPro Driver};” & _
“SourceType=DBF;” & _
“SourceDB=c:\somepath\mySourceDbFolder;” & _
“Exclusive=No;”
oConn.Open strConn
%>

用SQL语句创建Access表

2006年6月8日

create table mytable (m_id integer identity(1,1) primary key ,–自增型,主键 
m_class varchar(50) not null default 'AAA' ,–文本,非空,默认值'AAA' 
m_int integer not null ,–长整型,非空
m_numeric NUMERIC(6,2) ,–小数型
m_money money not null default 0.00 ,–货币型,非空,默认值0.00 
m_memo text ,–备注型
m_date date default date() ,–日期型,默认为当前日期
m_boolean bit default yes ,–布尔型,默认为yes
m_blob OLEObject ,–BLOB型
m_double double ,–双精度型
m_float real) –单精度型


—————————————————————————-
类型名称 TYPE 备注
—————————————————————————-
自动编号 integer + identity(1,1)
文本 varchar(50) 括号中的数字为文本长度
长整型 integer
整型 short
双精度型 double,float
单精度型 real
字节型 byte
小数 NUMERIC(6,2)
货币 money
备注 text
日期/时间 date,time,datetime
是/否 bit
OLE 对象 OLEObject

—————————————————————————-

主键 primary key
必填 not null
默认值 default 当为日期型时为 default date()
—————————————————————————–

示例
表名 字段名 类型 附属属性 说明
——- ——— ———— ——————————— ——————-
create table mytable (m_id integer identity(1,1) primary key ,–自增型,主键
m_class varchar(50) not null default 'AAA' ,–文本,非空,默认值'AAA'
m_int integer not null ,–长整型,非空
m_numeric NUMERIC(6,2) ,–小数型
m_money money not null default 0.00 ,–货币型,非空,默认值0.00
m_memo text ,–备注型
m_date date default date() ,–日期型,默认为当前日期
m_boolean bit default yes ,–布尔型,默认为yes
m_blob OLEObject ,–BLOB型
m_double double ,–双精度型
m_float real) –单精度型
—————————————————————————————————————————-

创建索引

示例1
create index myindex on mytable (m_class [DESC, ASC], m_int)
示例2
create unique index myindex on mytable (m_class) –创建无重复索引
注意:主键字段会被自动建立为没有重复的索引

JAVA的语法 VS C#语法

2006年6月4日

个人感觉更喜欢C#的语法,比如getter和setter,namespace的代码组织方式,“继承”和“实现”直接使用“:”而不是使用关键字“extens”和“implements”。


还有C#没有所谓的包的概念,所以不在乎代码放在什么地方。


C#没有所谓默认访问权限。在C#中默认的就是public,而java在没有访问控制描述时,访问权限限制在包内,是一种区别于private、protected、public三种标准权限的另一种访问权限。


JAVA给人的感觉是虽然很强大,但是从美感方面总是有些瑕疵,比如JAVA的各种包,重叠、凌乱、庞大,难以学习,C#基于.NET框架,就这么一个类库,结构清晰,易学,与Windows系统结合紧密。


很多时候,JAVA给人的感觉就像是在一个光洁完美的表面上安装上一个丑陋的螺丝钉,虽然系统更加坚苦了,但是美感无存。而C#的完美无暇就像一个仙女,清新纯洁,让人恋恋不舍。

Recordset-BOF、EOF 属性

2006年3月13日

BOF、EOF 属性



  • BOF 指示当前记录位置位于 Recordset 对象的第一个记录之前。


  • EOF 指示当前记录位置位于 Recordset 对象的最后一个记录之后。

返回值


BOFEOF 属性返回布尔型值。


BOF、EOF 属性



  • BOF 指示当前记录位置位于 Recordset 对象的第一个记录之前。


  • EOF 指示当前记录位置位于 Recordset 对象的最后一个记录之后。

返回值


BOFEOF 属性返回布尔型值。


说明


使用 BOFEOF 属性可确定 Recordset 对象是否包含记录,或者从一个记录移动到另一个记录时是否超出 Recordset 对象的限制。


如果当前记录位于第一个记录之前,BOF 属性将返回 True (-1),如果当前记录为第一个记录或位于其后则将返回 False (0)。


如果当前记录位于 Recordset 对象的最后一个记录之后 EOF 属性将返回 True,而当前记录为 Recordset 对象的最后一个记录或位于其前,则将返回 False


如果 BOFEOF 属性为 True,则没有当前记录。


如果打开没有记录的 Recordset 对象,BOFEOF 属性将设置为 True,而 Recordset 对象的 RecordCount 属性设置为零。打开至少包含一条记录的 Recordset 对象时,第一条记录为当前记录,而 BOFEOF 属性为 False


如果删除 Recordset 对象中保留的最后记录,BOFEOF 属性将保持 False,直到重新安排当前记录。


以下表格说明不同 BOFEOF 属性组合所允许的 Move 方法。

































MoveFirst,
MoveLast
MovePrevious,
Move < 0

Move 0
MoveNext,
Move > 0
BOF=True,
EOF=False
允许 错误 错误 允许
BOF=False,
EOF=True
允许 允许 错误 错误
同时为 True 错误 错误 错误 错误
同时为 False 允许 允许 允许 允许


允许使用 Move 方法并不能保证该方法成功定位记录,只是意味着调用指定的 Move 方法不会产生错误。


下表说明当调用各种 Move 方法但未成功定位记录时 BOF EOF 属性设置所发生的情况。























BOF EOF
MoveFirst, MoveLast 设置为 True 设置为 True
Move 0 没有变化 没有变化
MovePrevious, Move < 0 设置为 True 没有变化
MoveNext, Move > 0 没有变化 设置为 True


www.51windows.Net





对于ASP编码问题的深入研究与最终解决方案[zz]

2006年2月26日

ASP乱码确实棘手,这个说明比较权威。有待研究。


哪的资料都不如官方资料权威。今天总算从MSDN中择出了ASP编码问题的解决方案。



ASP乱码确实棘手,这个说明比较权威。有待研究。


哪的资料都不如官方资料权威。今天总算从MSDN中择出了ASP编码问题的解决方案。









下面是MSDN中的一段话。


Setting @CODEPAGE explicitly affects literal strings in a single response. Response.CodePage affects dynamic strings in a single response, and Session.CodePage affects dynamic strings in all responses in a session.


这句话解释清楚了@CODEPAGEResponse.CodePage,Session.CodePage 分别的作用是什么。


@CODEPAGE作用于所有静态的字符串,比如某文件中的 const blogname=”我的家”


Response.CodePage,Session.CodePage作用于所有动态输出的字符串,比如<%=blogname%>


这句话很关键的是说明了Response.CodePage的作用范围是a single response,而SXNA中声明的Session.CodePage的作用范围是all responses in a session。







再看另外一句话。


If Response.CodePage is not explicitly set in a page, it is implicitly set by Session.CodePage, if sessions are enabled. If sessions are not enabled, Response.CodePage is set by @CodePage, if @CodePage is present in the page. If there is no @CodePage in the page, Response.CodePage is set by the AspCodePage metabase property. If the AspCodePage metabase property is not set, or set to 0, Response.CodePage is set by the system ANSI code page.


这句话我乍一看,把意思理解成了这样:在sessions are enabled的时候,如果Response.CodePage没有声明,则Response.CodePage会被Session.CodePage赋值。如果sessions are not enabled的时候, 如果@CodePage已声明,则Response.CodePage会被@CodePage赋值,等等………….


这句话解释了为什么从SXNA中出来以后进入一些别的页面比如oblog,z-blog等等容易出现乱码,因为其他程序没有声明Response.CodePage而恰巧SXNA声明了Session.CodePage,因此一进入SXNA,Session.CodePage立即被赋值(版本不同,有的版本赋了936有的版本赋了65001),而后进入其他程序的时候Response.CodePage马上被Session.CodePage赋值如果这时Response.CodePage与页面本身编码不一样的话,页面就会出现乱码。所以进入z-blog出现乱码的时候我查了当时的Session.CodePage和Response.CodePage都是936,而进入oblog出现乱码的时候Session.CodePage和Response.CodePage都是65001.就是说要想保证叶面不出现乱码,应该声明Response.CodePage,否则他就会按照Session.CodePage来解释网页(而不是按照@codepage解释网页).


如果仅仅按照上面的解释的话,我实际上是很糊涂的,因为我们都是用的中文操系统,当每一次进入浏览器的时候你可以尝试输出Session.CodePage,能看到他都是936!为什么进入Z-blog的时候他不把默认的Session.CodePage的936赋给Response.CodePage呢?反而把@CodePage给了Response.CodePage?什么情况下Session.CodePage才赋值给Response.CodePage呢?原文的sessions are enabled应该如何理解呢?


也许上面的话应该这样理解:







在Session.CodePage任何程序声明的时候,如果Response.CodePage没有声明,则Response.CodePage会被Session.CodePage赋值。如果Session.CodePage没有被任何程序声明的时候, 如果@CodePage已声明,则Response.CodePage会被@CodePage赋值,….,最后的页面动态内容部分按照Response.CodePage的值解释。


因为Zblog和Oblog都声明了@CodePage,所以,用户刚刚启动完机器然后进入浏览器浏览Zblog和Oblog的时候Response.CodePage会被@CodePage赋值,于是叶面显示正常。







这句话进一步解释了产生乱码的原因


If you set Response.CodePage or Session.CodePage explicitly, do so before sending non-literal strings to the client. If you use literal and non-literal strings in the same page, make sure the code page of @CODEPAGE matches the code page of Response.CodePage, or the literal strings are encoded differently from the non-literal strings and display incorrectly.


其中比较有用的一句话是说如果Response.CodePage@CODEPAGE不一样的话会产生乱码。也就是说当Z-blog的@CODEPAGE=65001而Z-blog的Response.CodePage被Session.CodePage赋为936的时候就会出现乱码,oblog反之亦然。


不知道上面说了这么多解释清楚没有-_-||







下面解释一下为什么SXNA有时会把Session.CodePage赋为936,我有一个版本是这样写的:


<% OriginalCodePage=Session.CodePage %>


…….


<% Session.CodePage=OriginalCodePage %>


当用户进入浏览器的时候Session.CodePage默认为936,这个时候的默认936不是程序声明的,因此不会赋给Response.CodePage,当进入SXNA的时候,Session.CodePage被上面那段代码一折腾就变成了程序声明的Session.CodePage=936,因此再进入Zblog的时候就把936给了Response.CodePage


至此,全部原因已经分析清楚了。


因此说,保证asp叶面一定不会出现乱码的代码应该是这样的:(假定是UTF-8的叶子)







<%@ CODEPAGE=65001 %>


<% Response.CodePage=65001%>


<% Response.Charset="UTF-8" %>


进一步说明为什么要加Response.Charset,因为MSDN说应该加…呵呵







If the code page is set in a page, then Response.Charset should also be set.


另外,文件的编码格式应该与@CODEPAGE一样:







The file format of a Web page must be the same as the @CODEPAGE used in the page.


这就是为什么zblog,pjblog等一些程序要吧文件存成UTF8编码格式的原因.


综上,如果所有的程序都声明了Response.CodePage就不会被Session.CodePage干扰而出现乱码了。所以Session.CodePage还是不能轻易用的!


 


参考文章:


http://msdn.microsoft.com/library/default.asp?url=/library/en-us/iissdk/html/268f1db1-9a36-4591-956b-d7269aeadcb0.asp


http://msdn.microsoft.com/library/default.asp?url=/library/en-us/iissdk/html/582e6f47-52eb-413e-8b5d-c99145cb61d8.asp

ASP判断浏览器是否支持cookie

2006年2月25日

网上有一些关于asp判断浏览器是否支持cookie的代码,一般都很有局限性,举其中最典型的一例:


网上有一些关于asp判断浏览器是否支持cookie的代码,一般都很有局限性,举其中最典型的一例:









以下是引用片段:http://www.pconline.com.cn/pcedu/empolder/wz/asp/10111/15131.html
Browser Capabilities 组件众所周知,并不是所有浏览器都支持现今 Internet 技术的方方面面。有一些特性,某些浏览器支持而另一些浏览器却不支持,如 : ActiveX 控件、影像流、动态 HTML、 Flash 以及脚本程序等。使用 ASP 的 Browser Capabilities 组件,就能够设计“智能”的 Web 页,以适合浏览器性能的格式呈现内容。 Browser Capabilities 组件能够创建一个 BrowserType 对象,该对象提供带有客户端网络浏览器的功能说明的用户脚本。该组件之所以能识别客户浏览器的版本等信息,主要是因为当客户浏览器向服务器发送页面请求时,会自动发送一个 User Agent HTTP 标题,该标题是一个声明浏览器及其版本的 ASCII 字符串。 Browser Capabilities 组件将 User Agent 映射到在文件 Browscap.ini 中所注明的浏览器 , 并通过 BrowserType 对象的属性来识别客户浏览器。若该对象在 browscap.ini 文件中找不到与该标题匹配的项,那么将使用默认的浏览器属性。若该对象既未找到匹配项且 browscap.ini 文件中也未指定默认的浏览器设置,则它将每个属性都设为字符串 “UNKNOWN”。在默认情况下, browscap.ini 文件被存放在 WINDOWS\SYSTEM\INERSRV( 如果是 95/98+PWS4) 或 NT\SYSTEM32\INERSRV( 如果是 NT) 目录中,你可以自己编辑这个文本文件,以添加自己的属性或者根据最新发布的浏览器版本的更新文件来修改该文件。请看以下 checkCookie() 过程,使用 BrowserCap 对象的 Cookie 属性来判断客户端浏览器是否支持 Cookie,并返回信息 :

< %
Sub checkCookie()
Set BrowserCap=Server.CreateObject(“MSWC.BrowserType”)
if BrowserCap.Cookie=True then
response.write “你的浏览器支持 Cookie!”
else
response.write “对不起,你所使用的浏览器不支持 Cookie!”
end if
end Sub
%>


   使用此代码的条件是服务器必须支持Browser Capabilities 组件,而一般的主机是否支持此组件并不清楚,本博客的主机就不支持该组件,我的IE5都提示为unknown。另外即使浏览器支持cookie,用户也未必开启了该功能。


我写了另一个代码,也就是本站所采用的代码,直接测试浏览器是否启用了cookie。依赖于Response.redirect(URL)
假设使用该代码的页面为yourpage.asp,查询字符串为qstr,开页面开头假如如下代码:
IF Request.cookies(“Cookiename”)(“CookieTest”)<>“CookieTest” Then
   IF Request.QueryString(“cookie”)=”cookie” Then
      Response.Redirect(“no_cookies.html”)
   Else
      Request.cookies(“Cookiename”)(“CookieTest”)=”CookieTest” 
      Response.Redirect(“yourpage.asp?”&qstr&”&cookie=cookie”)
   End IF
End IF

ADO中使用设定了密码的Access数据库

2006年2月25日

Access2003以前的版本就有设置数据库密码的功能,同时在Access2003中提供了一个加密数据库的功能(这个功能 不知在Access2003之前是否存在),是给数据库进行编码,此种加密是防止用于以Access之外的工具打开数据库,使用Access或者其他ms提供的数据驱动程序打开数据库和加密之前没有区别,建议即给数据库设定密码,同时对数据库进行编码加密。


在AOD中使用设定了密码的,应这样设置:




Dim Conn
Set Conn= Server.CreateObject(“ADODB.Connection”)
Conn.ConnectionString=”Provider=Microsoft.Jet.OLEDB.4.0;Data Source=” & Server.MapPath(AccessFile)&”;Persist Security Info=False;Jet OLEDB:Database Password=数据库密码”


Access2003以前的版本就有设置数据库密码的功能,同时在Access2003中提供了一个加密数据库的功能(这个功能 不知在Access2003之前是否存在),是给数据库进行编码,此种加密是防止用于以Access之外的工具打开数据库,使用Access或者其他ms提供的数据驱动程序打开数据库和加密之前没有区别,建议即给数据库设定密码,同时对数据库进行编码加密。


在AOD中使用设定了密码的,应这样设置:




Dim Conn
Set Conn= Server.CreateObject(“ADODB.Connection”)
Conn.ConnectionString=”Provider=Microsoft.Jet.OLEDB.4.0;Data Source=” & Server.MapPath(AccessFile)&”;Persist Security Info=False;Jet OLEDB:Database Password=数据库密码”


1、provider是代表数据库提供者是什么样的数据库!如SQL Server是SQLOLEDB.1 Oracle 是ORAOLEDB。1等等!


2、Persist Security Info



Persist Security Info property specifies whether the data source can persist sensitive authentication information such as a password.

True:The data source object can persist sensitive authentication information such as a password along with other authentication information.

False:The data source object cannot persist sensitive authentication information.

您可以参考下面的网站:
http://msdn.microsoft.com/library/default.asp?url=/library/en-us/adosql/adoprg04_97g9.asp


另:SQLServer数据库中连接字符串有一个参数Initial Catalog,是表示数据库名是什么!

Response.Redirect和Server.Transfer的区别

2006年2月21日

Response.Redirect和Server.Transfer两者都是服务器端转向的工具,但是实现细节有所不同。


前者页面信息流的方向是:服务器->客户端->服务器->客户端


后者页面信息流的方向是:服务器->服务器->客户端


很显然,Server.Transfer少去了第一个从服务器传输数据到浏览器的过程,而是直接从服务器端转向到另一个页面,所以,一般情况下推荐使用这个转向。需要注意的是,Server.Transfer是服务器端转向,所以,浏览器地址栏将显示Server.Transfer语句所在的页面,而不是转向到的页面。使用Response.Redirect则会在浏览器地址栏显示目的页面的URL。


另外,Response.Redirect在本站使用的时候出现了乱码,服务器默认的数据传输编码是GB2312,而本站采用了UTF-8。在使用Response.Redirect的时候,页面并没有按照head标签里边设定的编码格式,而是按照服务器默认方式。使用Server.Transfer就没有这个问题了。

PHP 与 ASP.NET 正面交锋

2005年11月20日

  如今当提到 Web 开发时,您有许多选择。这些方法中许多都涉及到预处理 — 即,利用特定的标记将代码嵌入到 HTML 页面中,这些标记告诉预处理器,它们包含代码,并且应对它们作出处理。与 CGI 非常相似,这些代码在服务器上运行,并返回一些内容,这些内容表现为发回给浏览器的结果 HTML 页面的部分外观。开放源代码脚本语言 PHP 和 Microsoft 的 ASP.NET 框架中的语言都属于这种类型;JavaServer Pages (JSP) 和 Perl/Mason 也以这种方式运行。

  在本文中,我将重点讲述 PHP — Oracle 已选择合并到其产品中的技术 — 和 ASP.NET。我将概述两者的各种优势和弱点,并着重讨论那些将帮助您决定开发项目应选择哪种技术的因素。有很多因素需要考虑,不同的项目可能诉诸于不同的技术。总之,您将看到在价格、速度和效率、安全性、跨平台支持等方面的逐项比较,以及开放源代码解决方案的优势。


什么是 ASP.NET?
  如今当提到 Web 开发时,您有许多选择。这些方法中许多都涉及到预处理 — 即,利用特定的标记将代码嵌入到 HTML 页面中,这些标记告诉预处理器,它们包含代码,并且应对它们作出处理。与 CGI 非常相似,这些代码在服务器上运行,并返回一些内容,这些内容表现为发回给浏览器的结果 HTML 页面的部分外观。开放源代码脚本语言 PHP 和 Microsoft 的 ASP.NET 框架中的语言都属于这种类型;JavaServer Pages (JSP) 和 Perl/Mason 也以这种方式运行。

  在本文中,我将重点讲述 PHP — Oracle 已选择合并到其产品中的技术 — 和 ASP.NET。我将概述两者的各种优势和弱点,并着重讨论那些将帮助您决定开发项目应选择哪种技术的因素。有很多因素需要考虑,不同的项目可能诉诸于不同的技术。总之,您将看到在价格、速度和效率、安全性、跨平台支持等方面的逐项比较,以及开放源代码解决方案的优势。


什么是 ASP.NET?

  ASP 最新的版本 ASP.NET 并不完全与 ASP 早期的版本后向兼容,因为该软件进行了完全重写。早期的 ASP 技术实际上与 PHP 的共同之处比与 ASP.NET 的共同之处多得多,ASP.NET 是用于构建 Web 应用程序的一个完整的框架。这个模型的主要特性之一是选择编程语言的灵活性。ASP.NET 可以使用脚本语言(如 VBScript、JScript、Perlscript 和 Python)以及编译语言(如 VB、C#、C、Cobol、Smalltalk 和 Lisp)。新框架使用通用语言运行环境 (CLR);先将您语言的源代码编译成 Microsoft 中间语言代码,然后 CLR 执行这些代码。

  这个框架还提供真正的面向对象编程 (OOP),并支持真正的继承、多态和封装。.NET 类库根据特定的任务(例如,使用 XML 或图像处理)组织成可继承的类。

  除了编程语言和方法之外,数据库访问也是要着重关心的一个因素。当您用 ASP.NET 编程时,可以用 ODBC 来集成数据库;ODBC 提供了一组一致的调用函数来访问您的目标数据库。

优势和弱点

  ASP.NET 的优势很明显在于它简洁的设计和实施。这是面向对象的编程人员的梦想:语言灵活,并支持复杂的面向对象特性。在这种意义下,它真正能够与编程人员现有的技能进行互操作。

  ASP.NET 的另一个优势是其开发环境。例如,开发人员可以使用 WebMatrix(一个社区支持的工具)、Visual Studio .NET 或各种 Borland 工具(如 Delphi 和 C++ Builder)。例如,Visual Studio 允许设置断点、跟踪代码段和查看调用堆栈。总而言之,它是一个复杂的调试环境。许多其他第三方的 ASP.NET IDE 解决方案也将必然出现。

  但您得到了强健性,就将以损失效率为代价。ASP.NET 在内存使用和执行时间方面耗费非常大,这大部分归因于较长的代码路径。对基于 Web 的应用程序,这些局限可能是一个严重的问题,因为在 Web 上,您的应用程序可能扩展为每秒成千上万的用户。内存使用率还可能成为 Web 服务器上的一个问题。

什么是 PHP?

  PHP 是基于预处理 HTML 页面模型的一种脚本语言。当 Web 服务器中的 PHP 预处理器发现像下面这样的 PHP 语言标记时,将调用 PHP 引擎来执行该代码:






some code here
?>




  任何使用过命令式编程语言的编程人员都会对 PHP 非常熟悉;您会发现它与 Perl、C 和 Java 在语法上的相似处。严格来讲,Java 是一种命令式编程语言,但它也利用了面向对象的结构和概念。PHP 在适宜的时候借鉴了这种结构,但它不是一种纯粹的 OOP 语言。

  在上面对 ASP.NET 的讨论中,我提到了 ODBC 驱动程序,以及在考虑数据库抽象的情况下如何构建应用程序。在 PHP 中,您也可以使用 ODBC 与数据库对话,因此您已经有了一系列支持的数据库可供选择。也有 MySQL、Oracle 和 Postgres 的原生驱动程序。此外,如果要使用 Oracle,则有一个特殊的 OCI8 库将提供对 Oracle 更多访问功能,从而允许您使用诸如 LOB、BLOB、CLOB 和 BFILE 之类的特性。

  这时您可能会问“为什么与数据库相关的资料库被称为 PHP 的特性?”数据库抽象或独立性是您在设法构建使用多种数据库或需要在数据库间移植(例如,从开发到投产时的移植)的应用程序时所要考虑的一个因素。而这些确实是应该关心和考虑的。

  但正如 Tom Kyte 在他的新书 — Effective Oracle by Design (Oracle 出版社) — 中指出的那样,数据库相关性将成为您真正的目标,因为这将使您在该技术中的投资最大化。如果您对 Oracle 进行一般的访问 — 无论通过 ODBC 或 Perl 的 DBI 库,那么您将无法使用其他数据库所不具有的特性。此外,优化查询在各个数据库中各不相同。

  Zend Technologies 是一家商业软件公司,它对 PHP 作出了重大贡献。它创建了一个称为 Zend Studio 的商业开发环境,这个环境包含了一个复杂的调试器、一个监测器和其他特性。该公司还构建了免费的 Zend Optimizer,它与 Zend Encoder 结合使用来编译 PHP 代码,以提升性能。还有其他商业产品,如 Zend Performance Suite,它可以缓存预编译的 PHP 页面,从而进一步显著提升总体性能。

优势和弱点

  到测试版 4 为止,PHP 5 仍然有一些缺点,包括缺少异常和基于事件的错误处理例程 — 它们能够中断正常的程序流,并将代码跳转到一个特殊的错误处理部分。Java 也提供了用于错误处理的异常,而 C++ 通过 try、catch 和 throw 语法提供异常处理。当然,您仍然可以在 PHP 中管理错误,但其结构不是标准化的,因而使得编程人员必须使用自己的工具来决定执行错误处理的方式,这将导致更低的一致性并往往导致推倒重来。

  另一个弱点是 PHP 的函数名是不区分大小写的。虽然这不是一个严重的缺点,但有些程序员可能会觉得这一特性很讨厌。

  不过,我的确对 PHP 的对象模型心存疑虑。PHP 不是专门设计为一种面向对象的语言。这些特性中的一部分是后来添加的 — 尽管注意了保持与 PHP 3 的后向兼容性,因此两种模型的特性都留有一些。实际上,在 PHP 5 中这些弱点中的许多都得到了解决。请留心关注。

  PHP 在一些领域中缺少什么,它就会在擅长的领域中迅速地弥补什么。价格合理,因此您不需要担心许可问题。它也是开放源代码的,因此整个社区将密切关注开发过程:发现错误并将其修复。如果有一个特性您不喜欢,那么您可以修改代码。此外,PHP 可与 Apache 自然结合:它可以作为一个模块编译,或直接编译成 Apache 二进制文件。

  但在 Apache 上运行意味着,利用 PHP,您能够利用您已经在任意服务器投资,这是因为 Apache 能够运行在 Windows、Linux、Solaris 和各种其他 Unix 平台上。此外,使用拥有 Apache 的跟踪记录的 web 服务器意味着安全性能够保持在最高的优先级上。最后,PHP 拥有更小的代码路径,这意味着更少的分析和执行 PHP 页面服务器端代码,这将带来更高效的内存和使用率以及更快的运行。

在 PHP 5 中新增了哪些特性?

  PHP 5 的第 4 个测试版在 2003 年 12 月底推出,更改日志清楚地显示已发现许多错误并已将其消除。虽然它仍在进行测试,但它的所有新特性和进步都确实值得关注。

  PHP 5 的主要新成就在于它的异常处理和一个新对象,这个新对象引入的特性赋予了 PHP 真正的 OOP。异常处理无疑是 PHP 4 中最显著的缺憾之一,而 PHP 5 对异常处理的引入无疑是一个成熟的标志。异常处理意味着您在您的软件中拥有了根据语言定义的和标准化的错误处理方法。只需使用 try、catch 和 throw 方法,您的 PHP 代码就变得更加强健和简洁。


class blue
{

    function
openFile ($inFile
) {
        if (
file_exists ($inFile
)) {
            
# code to open the file here
        
} else {
            throw new
Exception
            
(“Cannot open file:$inFile”
);
        }
    }
}

$blueObj = new blue
();

try {
    
$blueObj->openFile (‘/home/shull/file.txt’
);

} catch (
Exception $myException
) {
    echo
$myException->getMessage
();

    
# rest of exception handling code here
}

# rest of blue methods here

?>


  新的对象模型为用 PHP 编写的程序带来许多积极的影响。在 PHP 4 中,当把将一个对象传递给函数或方法时,是通过值来传递的 — 除非您另行显式告知 PHP。这个过程意味着必须复制对象的一个副本(内存中的所有数据结构)。这个步骤使用内存,使得访问变慢和拥塞。而在 PHP 5 中,通常通过引用传递对象。

  PHP 5 中新的面向对象的特性(包括构造器和析构器)值得关注。与 C++ 和 Java 相同,它们提供了一种标准方法,即通过一个构造器方法来创建对象、分配内存和执行任何必要的设置,并通过一个析构器方法来执行清理。

  PHP 5 还引入了对类中方法和变量的更精细控制。在 PHP 4,一切都是公有的:您可以从类外或在继承类中访问您类中的变量。在 PHP 5 中,您仍然可以使变量或方法为公有,但您也可以使它们为私有,以使得只能在类本身中使用它们。还可以将它们保护起来,也就是说,可以在类内或在子类中查看方法和变量。

  此外,PHP 5 引进了类型提示或更好的类型检查。当您将一个对象传递给一个例程时,PHP 能够检查它是否是正确的类型,并在检查失败时产生类型不匹配错误。

  由于还有其他特性(如静态方法和变量以及抽象类),因此请务必查看文档,以获取详细信息。


安全性比较

  ASP.NET 官方要求您使用 IIS。不幸的是,IIS 易受攻击的历史由来已久,这使得许多管理员不愿意部署它来处理 Web 站点。这些弱点是因为 Microsoft 的缺陷或是因为 IIS 是黑客们攻击的目标无关紧要:这些系统有被黑或被攻击的历史。PHP 也运行在 Apache 上,Apache 快速并且是开放源代码的,并拥有很好的安全性记录。此外,正如我提到过的,Apache 可以在许多平台上运行。

  如果您在考虑 ASP.NET,但您想用 Apache 作为 Internet 门户,那么很幸运您可以有一些选择。首先,您可以使用 Apache 将请求转发给在另一台计算机上内部运行的 IIS。然后 Apache 处理静态内容,并将 aspx 内容传送给 IIS 服务器(不暴露给 Internet)。

  不过,如果您想利用 Apache 承载 ASP.NET,那么提供了一些选项,这些选项 Microsoft 可能支持也可能不支持。作为最后一种选择,有 Ximian 的 Project Mono,它致力于构建一个开放源代码模块。请浏览 www.go-mono.com,以获取更多信息。


数据库编码示例

  您在选用 PHP 或 ASP.NET 时首先要考虑因素之一就是与数据库的连接。不过,利用 ASP.NET 更复杂,因为您可以从许多备选语言中选择任意一种语言。当然,这些代码示例将必须嵌入到 HTML 页面、实例化的类等之中。不过,以下信息将使您对二者的编码风格有所了解。

PHP 5 与 Oracle 连接

  下面是一个 PHP 5 类,它提供了一个 Oracle 连接和断开例程,用以演示利用 PHP 5(也可以使用其他驱动程序(如 ODBC 驱动程序)和通用的数据库接口)与 Oracle 连接的一种方式:

class oracle_object {
    protected
$theDB
;
    protected
$user
;
    protected
$pass
;
    protected
$db
;

    function
__construct($u, $p, $d
) {
        
$this->user = $u
;
        
$this->pass = $p
;
        
$this->db = $d
;
    }

    function
db_open
() {
        
$theDB  =  @OCILogon($this->user,  $this->pass,  $this->db
);
        
db_check_errors($php_errormsg
);
    }

    function
db_close
() {
        @
OCILogoff($theDB
);
        
db_check_errors($php_errormsg
);
    }

    function
__destruct
() {
        print (
“so long…”
);
    }

}


ASP.NET 与 Oracle 连接

  如果您希望利用 VB.NET(Visual Basic 是 Microsoft 的默认 .NET 编程语言)与 Oracle 连接,那么请看一下这个来自 MSDN 的示例:

Imports System
Imports System.
Data
Imports System
.Data.
OracleClient
Imports Microsoft
.
VisualBasic

Class
Sample

Public Shared Sub Main
()

Dim oraConn As OracleConnection = New
OracleConnection
(“Data Source=MyOracleServer;Integrated Security=yes;”
)

Dim oraCMD As OracleCommand = New
OracleCommand
(“SELECT CUSTOMER_ID, NAME FROM DEMO.CUSTOMER”, oraConn
)

oraConn.Open
()

Dim myReader As OracleDataReader = oraCMD.ExecuteReader
()

Do While (
myReader.Read
())
Console.WriteLine(vbTab & “{0}” & vbTab & “{1}”
,
myReader.GetInt32(0), myReader.GetString(1
))
Loop

myReader
.Close
()
oraConn.Close
()
End Sub
End
Class

作出选择

  假定您还没有决定选用 PHP,我可以断言 PHP 的优势远远超过它的弱点。(请参见表 1 中的概要。)这些优势归结为价格、速度和效率、安全性、跨平台适用性和开放源代码机遇。它惟一的弱点是缺少一种纯粹和完美的 OOP 实现,不过,这是一个很小的缺点。虽然语言结构的确有帮助,但好的编码最终是由实践、执行、好的习惯和规范带来的。

表 1


























































PHP 4 PHP 5 ASP.NET
软件价格 免费 免费 免费
平台价格 免费 免费 $$
速度
效率
安全性
平台 弱(仅用于 IIS)
平台 任意 任意 win32(仅用于 IIS)
是否提供源代码
异常
OOP


  价格。在此,我们不应简单地考虑初始投资 — 对 PHP 而言,这很明显是免费的 — 还应考虑实施、维护和调试的成本。对 PHP 而言,您可能需要购买 Zend 优化引擎。然而,使用 ASP,您将从一开始就进行投资,而且您还要为附加的技术 — 例如,执行图形处理的库 — 支付费用。但从长远来看,PHP 将不会逼迫您升级并向您收取更多的许可费用。接触过复杂许可的每一个人都知道,很多公司花费大量的时间和金钱仅是为了确保其顺应性。此外,当在获得错误修复的时间上,您得到的反应也有所不同。这当然将转化成时间,时间又将转化成总体开发的成本。

  速度和效率。正如我先前提到的,ASP.NET 是一个允许您使用各种编程语言的框架。此外,它据称拥有一个极好的面向对象模型。尽管所有这些都是真的,但在考虑到速度时,它却是不利的。基于以上原因,在 ASP.NET 中运行 ASP 页面与在 PHP 引擎中运行等效的 PHP 页面相比需要执行更多代码。PHP 是一种“快速粗糙”的解决方案,是为完成工作而设计的解决方案。虽然自 2.0 和 3.0 版本以来为其增强了许多强健性,但它仍然保留着核心的优化的高速方法。

  速度不是惟一要考虑的因素。内存使用率也很重要。

  安全性。ASP.NET 运行在 IIS 上,而 IIS 已被攻击了无数次 — 正如每隔一周的 IT 新闻报告所证实的。它已经成为这样一种负担,实际上,尽管其耗巨资进行销售宣传,但许多 IT 专业人员仍然拒绝用 IIS Web 服务器来开放他们的网络。而 PHP 使用 Apache。Apache 拥有已被证明的速度、可靠性和稳固的安全性的记录。请浏览 www.securityfocus.com,以获取更多信息。

  跨平台适用性。ASP.NET 在 IIS 上运行,并且开始在 Apache 上运行(Apache 能够在许多平台上运行)。PHP 从一开始就设计用于和 Apache 一起工作,因此您拥有许多可供选择的已被证明和可靠的服务器平台。

  开放源代码机遇。开放源代码并不只靠一些异想天开的编程人员或想要节省一些许可费用的公司来开展。当您处理软件本身的错误时,开放源代码可能成为真正的天赐之物。

  在使用 PHP 或 ASP.NET 的情况下,您都会有一个大的用户群,他们使用软件并且可能遇到错误。使用 ASP.NET,这些错误必须通过一个官方程序来通知、修复、测试,并在一个新的补丁或版本中消除。然而,PHP 补丁可以快速地得到修补并发布。目睹开放源代码发展的任何人都知道,新的版本和补丁通常在几天内而不是像商业软件一样在几个星期或几个月内推出。如果这不够快,那么您通常可以自己来修补问题(如果必要的话)。

关于作者

  Sean Hull 是 iHeavy Inc.公司的高级顾问,该公司位于纽约市。他专注于开放源代码技术与商业技术(如 Oracle)的集成,并已服务于多家成功的纽约公司。







some code here
?>



  任何使用过命令式编程语言的编程人员都会对 PHP 非常熟悉;您会发现它与 Perl、C 和 Java 在语法上的相似处。严格来讲,Java 是一种命令式编程语言,但它也利用了面向对象的结构和概念。PHP 在适宜的时候借鉴了这种结构,但它不是一种纯粹的 OOP 语言。

  在上面对 ASP.NET 的讨论中,我提到了 ODBC 驱动程序,以及在考虑数据库抽象的情况下如何构建应用程序。在 PHP 中,您也可以使用 ODBC 与数据库对话,因此您已经有了一系列支持的数据库可供选择。也有 MySQL、Oracle 和 Postgres 的原生驱动程序。此外,如果要使用 Oracle,则有一个特殊的 OCI8 库将提供对 Oracle 更多访问功能,从而允许您使用诸如 LOB、BLOB、CLOB 和 BFILE 之类的特性。

  这时您可能会问“为什么与数据库相关的资料库被称为 PHP 的特性?”数据库抽象或独立性是您在设法构建使用多种数据库或需要在数据库间移植(例如,从开发到投产时的移植)的应用程序时所要考虑的一个因素。而这些确实是应该关心和考虑的。

  但正如 Tom Kyte 在他的新书 — Effective Oracle by Design (Oracle 出版社) — 中指出的那样,数据库相关性将成为您真正的目标,因为这将使您在该技术中的投资最大化。如果您对 Oracle 进行一般的访问 — 无论通过 ODBC 或 Perl 的 DBI 库,那么您将无法使用其他数据库所不具有的特性。此外,优化查询在各个数据库中各不相同。

  Zend Technologies 是一家商业软件公司,它对 PHP 作出了重大贡献。它创建了一个称为 Zend Studio 的商业开发环境,这个环境包含了一个复杂的调试器、一个监测器和其他特性。该公司还构建了免费的 Zend Optimizer,它与 Zend Encoder 结合使用来编译 PHP 代码,以提升性能。还有其他商业产品,如 Zend Performance Suite,它可以缓存预编译的 PHP 页面,从而进一步显著提升总体性能。

优势和弱点

  到测试版 4 为止,PHP 5 仍然有一些缺点,包括缺少异常和基于事件的错误处理例程 — 它们能够中断正常的程序流,并将代码跳转到一个特殊的错误处理部分。Java 也提供了用于错误处理的异常,而 C++ 通过 try、catch 和 throw 语法提供异常处理。当然,您仍然可以在 PHP 中管理错误,但其结构不是标准化的,因而使得编程人员必须使用自己的工具来决定执行错误处理的方式,这将导致更低的一致性并往往导致推倒重来。

  另一个弱点是 PHP 的函数名是不区分大小写的。虽然这不是一个严重的缺点,但有些程序员可能会觉得这一特性很讨厌。

  不过,我的确对 PHP 的对象模型心存疑虑。PHP 不是专门设计为一种面向对象的语言。这些特性中的一部分是后来添加的 — 尽管注意了保持与 PHP 3 的后向兼容性,因此两种模型的特性都留有一些。实际上,在 PHP 5 中这些弱点中的许多都得到了解决。请留心关注。

  PHP 在一些领域中缺少什么,它就会在擅长的领域中迅速地弥补什么。价格合理,因此您不需要担心许可问题。它也是开放源代码的,因此整个社区将密切关注开发过程:发现错误并将其修复。如果有一个特性您不喜欢,那么您可以修改代码。此外,PHP 可与 Apache 自然结合:它可以作为一个模块编译,或直接编译成 Apache 二进制文件。

  但在 Apache 上运行意味着,利用 PHP,您能够利用您已经在任意服务器投资,这是因为 Apache 能够运行在 Windows、Linux、Solaris 和各种其他 Unix 平台上。此外,使用拥有 Apache 的跟踪记录的 web 服务器意味着安全性能够保持在最高的优先级上。最后,PHP 拥有更小的代码路径,这意味着更少的分析和执行 PHP 页面服务器端代码,这将带来更高效的内存和使用率以及更快的运行。

在 PHP 5 中新增了哪些特性?

  PHP 5 的第 4 个测试版在 2003 年 12 月底推出,更改日志清楚地显示已发现许多错误并已将其消除。虽然它仍在进行测试,但它的所有新特性和进步都确实值得关注。

  PHP 5 的主要新成就在于它的异常处理和一个新对象,这个新对象引入的特性赋予了 PHP 真正的 OOP。异常处理无疑是 PHP 4 中最显著的缺憾之一,而 PHP 5 对异常处理的引入无疑是一个成熟的标志。异常处理意味着您在您的软件中拥有了根据语言定义的和标准化的错误处理方法。只需使用 try、catch 和 throw 方法,您的 PHP 代码就变得更加强健和简洁。


class blue
{

    function
openFile ($inFile
) {
        if (
file_exists ($inFile
)) {
            
# code to open the file here
        
} else {
            throw new
Exception
            
(“Cannot open file:$inFile”
);
        }
    }
}

$blueObj = new blue
();

try {
    
$blueObj->openFile (‘/home/shull/file.txt’
);

} catch (
Exception $myException
) {
    echo
$myException->getMessage
();

    
# rest of exception handling code here
}

# rest of blue methods here

?>


  新的对象模型为用 PHP 编写的程序带来许多积极的影响。在 PHP 4 中,当把将一个对象传递给函数或方法时,是通过值来传递的 — 除非您另行显式告知 PHP。这个过程意味着必须复制对象的一个副本(内存中的所有数据结构)。这个步骤使用内存,使得访问变慢和拥塞。而在 PHP 5 中,通常通过引用传递对象。

  PHP 5 中新的面向对象的特性(包括构造器和析构器)值得关注。与 C++ 和 Java 相同,它们提供了一种标准方法,即通过一个构造器方法来创建对象、分配内存和执行任何必要的设置,并通过一个析构器方法来执行清理。

  PHP 5 还引入了对类中方法和变量的更精细控制。在 PHP 4,一切都是公有的:您可以从类外或在继承类中访问您类中的变量。在 PHP 5 中,您仍然可以使变量或方法为公有,但您也可以使它们为私有,以使得只能在类本身中使用它们。还可以将它们保护起来,也就是说,可以在类内或在子类中查看方法和变量。

  此外,PHP 5 引进了类型提示或更好的类型检查。当您将一个对象传递给一个例程时,PHP 能够检查它是否是正确的类型,并在检查失败时产生类型不匹配错误。

  由于还有其他特性(如静态方法和变量以及抽象类),因此请务必查看文档,以获取详细信息。


安全性比较

  ASP.NET 官方要求您使用 IIS。不幸的是,IIS 易受攻击的历史由来已久,这使得许多管理员不愿意部署它来处理 Web 站点。这些弱点是因为 Microsoft 的缺陷或是因为 IIS 是黑客们攻击的目标无关紧要:这些系统有被黑或被攻击的历史。PHP 也运行在 Apache 上,Apache 快速并且是开放源代码的,并拥有很好的安全性记录。此外,正如我提到过的,Apache 可以在许多平台上运行。

  如果您在考虑 ASP.NET,但您想用 Apache 作为 Internet 门户,那么很幸运您可以有一些选择。首先,您可以使用 Apache 将请求转发给在另一台计算机上内部运行的 IIS。然后 Apache 处理静态内容,并将 aspx 内容传送给 IIS 服务器(不暴露给 Internet)。

  不过,如果您想利用 Apache 承载 ASP.NET,那么提供了一些选项,这些选项 Microsoft 可能支持也可能不支持。作为最后一种选择,有 Ximian 的 Project Mono,它致力于构建一个开放源代码模块。请浏览 www.go-mono.com,以获取更多信息。


数据库编码示例

  您在选用 PHP 或 ASP.NET 时首先要考虑因素之一就是与数据库的连接。不过,利用 ASP.NET 更复杂,因为您可以从许多备选语言中选择任意一种语言。当然,这些代码示例将必须嵌入到 HTML 页面、实例化的类等之中。不过,以下信息将使您对二者的编码风格有所了解。

PHP 5 与 Oracle 连接

  下面是一个 PHP 5 类,它提供了一个 Oracle 连接和断开例程,用以演示利用 PHP 5(也可以使用其他驱动程序(如 ODBC 驱动程序)和通用的数据库接口)与 Oracle 连接的一种方式:

class oracle_object {
    protected
$theDB
;
    protected
$user
;
    protected
$pass
;
    protected
$db
;

    function
__construct($u, $p, $d
) {
        
$this->user = $u
;
        
$this->pass = $p
;
        
$this->db = $d
;
    }

    function
db_open
() {
        
$theDB  =  @OCILogon($this->user,  $this->pass,  $this->db
);
        
db_check_errors($php_errormsg
);
    }

    function
db_close
() {
        @
OCILogoff($theDB
);
        
db_check_errors($php_errormsg
);
    }

    function
__destruct
() {
        print (
“so long…”
);
    }

}


ASP.NET 与 Oracle 连接

  如果您希望利用 VB.NET(Visual Basic 是 Microsoft 的默认 .NET 编程语言)与 Oracle 连接,那么请看一下这个来自 MSDN 的示例:

Imports System
Imports System.
Data
Imports System
.Data.
OracleClient
Imports Microsoft
.
VisualBasic

Class
Sample

Public Shared Sub Main
()

Dim oraConn As OracleConnection = New
OracleConnection
(“Data Source=MyOracleServer;Integrated Security=yes;”
)

Dim oraCMD As OracleCommand = New
OracleCommand
(“SELECT CUSTOMER_ID, NAME FROM DEMO.CUSTOMER”, oraConn
)

oraConn.Open
()

Dim myReader As OracleDataReader = oraCMD.ExecuteReader
()

Do While (
myReader.Read
())
Console.WriteLine(vbTab & “{0}” & vbTab & “{1}”
,
myReader.GetInt32(0), myReader.GetString(1
))
Loop

myReader
.Close
()
oraConn.Close
()
End Sub
End
Class

作出选择

  假定您还没有决定选用 PHP,我可以断言 PHP 的优势远远超过它的弱点。(请参见表 1 中的概要。)这些优势归结为价格、速度和效率、安全性、跨平台适用性和开放源代码机遇。它惟一的弱点是缺少一种纯粹和完美的 OOP 实现,不过,这是一个很小的缺点。虽然语言结构的确有帮助,但好的编码最终是由实践、执行、好的习惯和规范带来的。

表 1


























































PHP 4 PHP 5 ASP.NET
软件价格 免费 免费 免费
平台价格 免费 免费 $$
速度
效率
安全性
平台 弱(仅用于 IIS)
平台 任意 任意 win32(仅用于 IIS)
是否提供源代码
异常
OOP


  价格。在此,我们不应简单地考虑初始投资 — 对 PHP 而言,这很明显是免费的 — 还应考虑实施、维护和调试的成本。对 PHP 而言,您可能需要购买 Zend 优化引擎。然而,使用 ASP,您将从一开始就进行投资,而且您还要为附加的技术 — 例如,执行图形处理的库 — 支付费用。但从长远来看,PHP 将不会逼迫您升级并向您收取更多的许可费用。接触过复杂许可的每一个人都知道,很多公司花费大量的时间和金钱仅是为了确保其顺应性。此外,当在获得错误修复的时间上,您得到的反应也有所不同。这当然将转化成时间,时间又将转化成总体开发的成本。

  速度和效率。正如我先前提到的,ASP.NET 是一个允许您使用各种编程语言的框架。此外,它据称拥有一个极好的面向对象模型。尽管所有这些都是真的,但在考虑到速度时,它却是不利的。基于以上原因,在 ASP.NET 中运行 ASP 页面与在 PHP 引擎中运行等效的 PHP 页面相比需要执行更多代码。PHP 是一种“快速粗糙”的解决方案,是为完成工作而设计的解决方案。虽然自 2.0 和 3.0 版本以来为其增强了许多强健性,但它仍然保留着核心的优化的高速方法。

  速度不是惟一要考虑的因素。内存使用率也很重要。

  安全性。ASP.NET 运行在 IIS 上,而 IIS 已被攻击了无数次 — 正如每隔一周的 IT 新闻报告所证实的。它已经成为这样一种负担,实际上,尽管其耗巨资进行销售宣传,但许多 IT 专业人员仍然拒绝用 IIS Web 服务器来开放他们的网络。而 PHP 使用 Apache。Apache 拥有已被证明的速度、可靠性和稳固的安全性的记录。请浏览 www.securityfocus.com,以获取更多信息。

  跨平台适用性。ASP.NET 在 IIS 上运行,并且开始在 Apache 上运行(Apache 能够在许多平台上运行)。PHP 从一开始就设计用于和 Apache 一起工作,因此您拥有许多可供选择的已被证明和可靠的服务器平台。

  开放源代码机遇。开放源代码并不只靠一些异想天开的编程人员或想要节省一些许可费用的公司来开展。当您处理软件本身的错误时,开放源代码可能成为真正的天赐之物。

  在使用 PHP 或 ASP.NET 的情况下,您都会有一个大的用户群,他们使用软件并且可能遇到错误。使用 ASP.NET,这些错误必须通过一个官方程序来通知、修复、测试,并在一个新的补丁或版本中消除。然而,PHP 补丁可以快速地得到修补并发布。目睹开放源代码发展的任何人都知道,新的版本和补丁通常在几天内而不是像商业软件一样在几个星期或几个月内推出。如果这不够快,那么您通常可以自己来修补问题(如果必要的话)。

关于作者

  Sean Hull 是 iHeavy Inc.公司的高级顾问,该公司位于纽约市。他专注于开放源代码技术与商业技术(如 Oracle)的集成,并已服务于多家成功的纽约公司。