燕之庐网站建设 - 优质网站设计公司

Archive for 10月 19th, 2009

php

2009/10/19

使用PHP的错误处理

Tags: , ,

在web 上所有常见的错误之一就是无效的链接。一旦从其它站点上出现了无效的链接,你会重新安排你
的站点。人们会将他们喜欢的站点存成书签,如果在三个月后再次访问时,仅仅发现的是’404 Not Fount ‘
时,将不会给他们任务帮助,告诉他们如何从你的站点去查找原始的信息。让我们解决这个问题,或者至少
给你的用户一个友好的帮助,一旦他们遇到’a 404′ 错误时,能够得到一些痕迹。你可以创建普通的页面来
报告在处理你的页面时所遇到的所有的错误。   PHP 与Apache一起可以很自由地让你创建自已的出错页面,但是需要重新进行配置,并且要编少量的代
码。先让我们学习配置部分。

Apache的ErrorDocument指示用来指出在出现错误时Apache应重定向到哪一个文档(或URI)。它允许你
为每一个你的用户可能遇到的错误代码指定一个资源。通过在你的服务器配置中增加一个ErrorDocument 404
/error.php指示。这个将在用户访问一个不存在的页面时,重定向到’error.php’中,下面我们就会写出
‘error.php’页面。不要忘了重新启动Apache以使改动生效。

接着,我们写出一个简单的error.php:

你所请求的文件 (<?=$REDIRECT_URL?>) 在这个服务器上不存在。
请查找你想要的文件从 <A HREF=”/”>前页</A>。

现在试着读取一个在你服务器上不存在的页面,怎么样,你可以看到error.php了,它有着一个良好和
友好的消息,并且还有一个到前页的链接。

让我们把它扩展一下。正如你所见,我在error.php中使用了REDIRECT_URL变量。这个变量是Apache在
执行了一个ErrorDocument指示时所设置的,并且给出了一种可能来找到原始的资源。在这种情况下,Apache
还设置了一些别的变量,所有的变量可以在这里找到。使用这些变量可能创建一个很好的出错页面,用于给
用户一个不错与友好的出错页面,而代替Apache给出的缺省页面。

从PHP页面中输出错误
从一个PHP页面输出错误与模拟Apache对ErrorDocument指示所做的很象,你只要简单地将用户重定向,
通过使用query-string变量,而Apache则通常是设置在环境变量里面。这样就可以使用同一个出错页面来处
理各种错误。下面是一个例子:

———————————————————————-
<?php

function throw_error($message) {
$error_page = “/err/error.php”;

$error_url = $error_page;
$error_url .= “?REDIRECT_ERROR_NOTES=$message”;
$error_url .= “&REDIRECT_URL=” . $GLOBALS["PHP_SELF"];
$error_url .= “&REDIRECT_REQUEST_METHOD=$REQUEST_METHOD”;
$error_url .= “&REDIRECT_STATUS=501″;
Header(”Status: 501″);
Header(”Location: $error_url”);
exit;
}

ob_start();
// 使用输出缓冲以便在这页中的任何地方输出错误

if(!condition) {
throw_error(”the condition failed”);
}

ob_end_flush();
// 页面处理完毕,刷新输出缓冲

?>——————————————————————
使用PHP4的输出缓冲特性对生成一般的出错报告功能也会有帮助。但是在你确认整个出错页面处理完毕
时,不要忘记刷新缓冲区,你可以在你的代码中的任可地方通过Header调用来进行重定向。

读者可以自行设计,实现自已的出错页面来适应他/她的站点。不要忘了,你可以包含一个带email的提
交表单在出错页面中,这样用户可以反馈给你一些意见。

php

PHP实现文件安全下载

Tags: ,

你一定会笑我”下载文件”如此简单都值得说?当然并不是想象那么简单。例如你希望客户要填完一份表格,才可以下载某一文件,你第一个想法一定是用”Redirect”的方法,先检查表格是否已经填写完毕和完整,然后就将网址指到该文件,这样客户才能下载,但如果你想做一个关于”网上购物”的电子商务网站,考虑安全问题,你不想用户直接复制网址下载该文件,笔者建议你使用PHP直接读取该实际文件然后下载的方法去做。程序如下:

$file_name = “info_check.exe”;

$file_dir = “/public/www/download/”;

if (!file_exists($file_dir . $file_name)) { //检查文件是否存在

echo “文件找不到”;

exit;

} else {

$file = fopen($file_dir . $file_name,”r”); //打开文件

//输入文件标签

Header(”Content-type: application/octet-stream”);

Header(”Accept-Ranges: bytes”);

Header(”Accept-Length: “.filesize($file_dir . $file_name));

Header(”Content-Disposition: attachment; filename=” . $file_name);

//输出文件内容

echo fread($file,filesize($file_dir . $file_name));

fclose($file);

exit;}

而如果文件路径是”http”或者”ftp”网址的话,则源代码会有少许改变,程序如下:

$file_name = “info_check.exe”;

$file_dir = “www.easycn.net/”;

$file = @ fopen($file_dir . $file_name,”r”);

if (!$file) {

echo “文件找不到”;

} else {

Header(”Content-type: application/octet-stream”);

Header(”Content-Disposition: attachment; filename=” . $file_name);

while (!feof ($file)) {

echo fread($file,50000);

}

fclose ($file);

}

这样就可以用PHP直接输出文件了。

php

Cookie及其使用(一)

Tags: ,

Cookie技术是一个非常有争议的技术,自经诞生它就成了广大网络用户和Web开发人员的一个争论焦点。有一些网络用户,甚至包括一些资深的Web专家 也对它的产生和推广感到不满,这倒不是因为Cookie技术的功能太弱或别的技术性能上的原因,而仅仅是因为他们觉得Cookie的使用,对网络用户的隐 私构成了危害。因为Cookie是由Web服务器保存在用户浏览器上的小文本文件,它包含有关用户的信息(如身份识别号码、密码、用户在Web站点上购物 的方式或用户访问该站点的次数)。

那么Cookie技术究竟怎样呢?是否真的给网络用户带来了个人隐私的危害呢?还是让我们看了下面的内容,再做回答吧。

(1)Cookie技术简介

在 WEB技术发展史上,Cookie技术的出现是一个重大的变革。最先是Netscape在它的Netscape Navigator 浏览器中引入了Cookie技术,从那时起,World Wide Web 协会就开始支持Cookie标准。以后又经过微软的大力推广(因为微软的IIS Web服务器所采用的ASP技术很大程度的使用了Cookier技术),即在微软的Internet Explorer浏览器中完全支持Cookie技术。到现在,绝大多数的浏览器都支持Cookie技术,或者至少兼容Cookie技术的使用。

1)什么是Cookie?

按 照Netscape官方文档中的定义,Cookie是在HTTP协议下,服务器或脚本可以维护客户工作站上信息的一种方式。Cookie是由Web服务器 保存在用户浏览器上的小广西文件,它可以包含有关用户的信息(如身份识别号码、密码、用户在Web站点购物的方式或用户访问该站点的次数)。无论何时用户 链接到服务器,Web站点都可以访问Cookie信息。

通俗地讲,浏览器用一个或多个限定的文件来支持Cookie。这些文件在使用 Windows操作系统的机器上叫做Cookie文件,在Macintosh机器上叫做magic Cookie 文件,这些文件被网站用来在上面存储Cookie数据。网站可以在这些Cookie文件中插入信息,这样对有些网络用户就有些副作用。有些用户认为这造成 了对个人隐私的侵犯,更糟的是,有些人认为Cookie是对个人空间的侵占,而且会对用户的计算机带来安全性的危害。

目前有些 Cookie是临时的,另一些则是持续的。临时的Cookie只在浏览器上保存一段规定的时间,一旦超过规定的时间该Cookie就会被系统清除。例如在 PHP中Cookie被用来跟踪用户进程直到用户离开网站。持续的Cookie则保存在用户的Cookie文件中,下一次用户返回时,仍然可以对它进行调 用。

在Cookie文件中保存Cookie,一些用户会过分地认为这将带来很大的问题。主要是有些用户担心Cookie会跟踪用户网上冲 浪的习惯,譬如用户喜爱到那些类型的站点、爱从事些什么活动等。害怕这种个人信息一旦落入一些别有用心的家伙手中,那么个人也就可能成为一大堆广告垃圾的 对象,甚至遭到意外的损害。不过,这种担心压根儿不会发生,因为网站以外的用户是无法跨过网站来获得Cookie信息的。所以想以这种目的来应用 Cookie是不可能的。不过,由于一些用户错误的理解以及“以讹传讹”,一些浏览器开发商别无选择,只好作出相识的响应(例如Netscape Navigator4.0和Internet Explorer3.0都提供了屏蔽Cookie的选项)。

对Cookie技术期待了这么 久的结果是,迫使许多浏览器开发商在它们的浏览器中提供了对Cookie的灵活性控制功能。例如,目前的两大主流浏览器Netscape Navigator 和 Internet Explorer是这样处理Cookie的:Netscape Navigator4.0不但可以接受Cookie进行警告,而且还可以屏蔽掉Cookie;InternetExplorer3.0也可以屏蔽 Cookie,但在Internet Explorer4.0中就只能进行接受警告而没有提供屏蔽选项,不过在Internet Explorer4.0之后的更新版本中又加入了屏蔽Cookie的功能选项。

此外,很多最新的技术甚至已经可以在不能屏蔽Cookie 的浏览器上进行Cookie的屏蔽了。例如,可以通过将Cookie文件设置成不同的类型来限制Cookie的使用。但是,非常不幸地是,要是你想完全屏 蔽Cookie的话,肯定会因此拒绝许多的站点页面。因为当今已经有许多Web站点开发人员爱上了Cookie技术的强大功能,例如Session对象的 使用就离不开Cookie的支持。

尽管今天仍有一些网络用户对于Cookie的争论乐此不倦,但是对于绝大多数的网络用户来说还是倾向于接受Cookie的。因此,我们尽可以放心地使用Cookie技术来开发我们的WEB页面。

2)Cookie是怎样工作的?

要 了解Cookie,必不可少地要知道它的工作原理。一般来说,Cookie通过HTTP Headers从服务器端返回到浏览器上。首先,服务器端在响应中利用Set-Cookie header来创建一个Cookie,然后,浏览器在它的请求中通过Cookie header包含这个已经创建的Cookie,并且反它返回至服务器,从而完成浏览器的论证。

例如,我们创建了一个名字为login的 Cookie来包含访问者的信息,创建Cookie时,服务器端的Header如下面所示,这里假设访问者的注册名是“Michael Jordan”,同时还对所创建的Cookie的属性如path、domain、expires等进行了指定。

Set-Cookie:login=Michael Jordan;path=/;domain=msn.com;
expires=Monday,01-Mar-99 00:00:01 GMT

上面这个Header会自动在浏览器端计算机的Cookie文件中添加一条记录。浏览器将变量名为“login”的Cookie赋值为“Michael Jordon”。注意,在实际传递过程中这个Cookie的值是经过了URLEncode方法的URL编码操作的。
这个含有Cookie值的HTTP Header被保存到浏览器的Cookie文件后,Header就通知浏览器将Cookie通过请求以忽略路径的方式返回到服务器,完成浏览器的认证操作。

此 外,我们使用了Cookie的一些属性来限定该Cookie的使用。例如Domain属性能够在浏览器端对Cookie发送进行限定,具体到上面的例子, 该Cookie只能传达室到指定的服务器上,而决不会跑到其他的如www.hp.com的Web站点上去。Expires属性则指定了该Cookie保存 的时间期限,例如上面的Cookie在浏览器上只保存到1999年3月1日1秒。当然,如果浏览器上Cookie太多,超过了系统所允许的范围,浏览器将 自动对它进行删除。至于属性Path,用来指定Cookie将被发送到服务器的哪一个目录路径下。

说明:浏览器创建了一个Cookie后,对于每一个针对该网站的请求,都会在Header中带着这个Cookie;不过,对于其他网站的请求Cookie是绝对不会跟着发送的。而且浏览器会这样一直发送,直到Cookie过期为止。

php

Cookie及其使用(二)

Tags: ,

上一部分讲了有关Cookie的技术背景,这部分来说说在PHP里如何设置、使用、删除Cookie,及Cookie的一些限制。PHP对Cookie支持是透明的,用起来非常方便。

1、设置Cookie

PHP用SetCookie函数来设置Cookie。必须注意的一点是:Cookie是HTTP协议头的一部分,用于浏览器和服务器之间传递信息,所以必须在任何属于HTML文件本身的内容输出之前调用Cookie函数。
SetCookie函数定义了一个Cookie,并且把它附加在HTTP头的后面,SetCookie函数的原型如下:
int SetCookie(string name, string value, int expire, string path, string domain, int secure);
除 了name之外所有的参数都是可选的。value,path,domain三个参数可以用空字符串代换,表示没有设置;expire 和 secure两个参数是数值型的,可以用0表示。expire参数是一个标准的Unix时间标记,可以用time()或mktime()函数取得,以秒为 单位。secure参数表示这个Cookie是否通过加密的HTTPS协议在网络上传输。
当前设置的Cookie不是立即生效的,而是要等到下一个页面时才能看到.这是由于在设置的这个页面里Cookie由服务器传递给客户浏览器,在下一个页面浏览器才能把Cookie从客户的机器里取出传回服务器的原因。
在同一个页面设置Cookie,实际是从后往前,所以如果要在插入一个新的Cookie之前删掉一个,你必须先写插入的语句,再写删除的语句,否则可能会出现不希望的结果。
来看几个例子:
简单的:
SetCookie(”MyCookie”, “Value of MyCookie”);
带失效时间的:
SetCookie(”WithExpire”, “Expire in 1 hour”, time() 3600);//3600秒=1小时
什么都有的:
SetCookie(”FullCookie”, “Full cookie value”, time() 3600, “/forum”, “.phpuser.com”, 1);

这 里还有一点要说明的,比如你的站点有几个不同的目录,那么如果只用不带路径的Cookie的话,在一个目录下的页面里设的Cookie在另一个目录的页面 里是看不到的,也就是说,Cookie是面向路径的。实际上,即使没有指定路径,WEB服务器会自动传递当前的路径给浏览器的,指定路径会强制服务器使用 设置的路径。解决这个问题的办法是在调用SetCookie时加上路径和域名,域名的格式可以是“www.phpuser.com”,也可是 “.phpuser.com”。
SetCookie函数里表示value的部分,在传递时会自动被encode,也就是说,如果value的值是 “test value”在传递时就变成了“test value”,跟URL的方法一样。当然,对于程序来说这是透明的,因为在PHP接收Cookie的值时会自动将其decode。
如果要设置同名的多个Cookie,要用数组,方法是:
SetCookie(”CookieArray[]“, “Value 1″);
SetCookie(”CookieArray[]“, “Value 2″);

SetCookie(”CookieArray[0]“, “Value 1″);
SetCookie(”CookieArray[1]“, “Value 2″);

2、接收和处理Cookie

PHP对Cookie的接收和处理的支持非常好,是完全自动的,跟FORM变量的原则一样,特别简单。
比 如设置一个名为MyCookier的Cookie,PHP会自动从WEB服务器接收的HTTP头里把它分析出来,并形成一个与普通变量一样的变量,名 为$myCookie,这个变量的值就是Cookie的值。数组同样适用。另外一个办法是引用PHP的全局变量$HTTP_COOKIE_VARS数组。
分别举例如下:(假设这些都在以前的页面里设置过了,并且仍然有效)
echo $MyCookie;
echo $CookieArray[0];
echo count($CookieArray);
echo $HTTP_COOKIE_VARS["MyCookie"];
就这么简单。

3、删除Cookie

要删除一个已经存在的Cookie,有两个办法:
一是调用只带有name参数的SetCookie,那么名为这个name的Cookie将被从关系户机上删掉;另一个办法是设置Cookie的失效时间为time()或time()-1,那么这个Cookie在这个页面的浏览完之后就被删除了(其实是失效了)。
要注意的是,当一个Cookie被删除时,它的值在当前页在仍然有效的。

4、使用Cookie的限制

首先是必须在HTML文件的内容输出之前设置;
其 次不同的浏览器对Cookie的处理不一致辞,且有时会出现错误的结果。比如:MS IE SERVICE PACK 1不能正确处理带域名和路径的Cookie,Netscape Communicator 4.05和MS IE 3.0不能正确处理不带路径和时间的Cookie。至于MS IE 5 好象不能处理带域名、路径和时间的Cookie。这是我在设计本站的页面时发现的。
第三个限制是在客户端的。一个浏览器能创建的Cookie数量最多为30个,并且每个不能超过4KB,每个WEB站点能设置的Cookie总数不能超过20个。

关于Cookie的话题,就说到这儿了。

(由于Cookie最初由Netscape定义的,所以附上Netscape公司关于Cookie的官方原始定义的网址:http://www.netscape.com/newsref/std/cookie_spec.html)

php

关于Zend Optimizer

Tags: ,

Zend Optimizer(以下简称ZO)用优化代码的方法来提高PHP 4.0应用程序的执行速度。实现的原理是对那些在被最终执行之前由运行编译器(Run-Time Compiler)产生的代码进行优化。 优化能提高你的盈利能力

一 般情况下,执行使用ZO的PHP程序比不使用的要快40%到100%。这意味着网站的访问者可以更快的浏览网页,从而完成更多的事务,创造更好的客户满意 度。更快的反应同时也意味着可以节省硬件投资,并增强网站所提供的服务。所以,使用ZO,就等于提高了电子商务的盈利能力。

ZO能给PHP用户带来很多益处,特别是那些运营网站的人。快速运行PHP程序可以显著降低服务器的CPU负载,并可以减少一半的反应时间,也就是从访问者点击链接到服务器开始读取页面之间的时间。

系统需求

当前版本的ZO(Beta 4)只能运行在PHP 4.0下。

对操作系统的要求如下:
- 基于glibc2.1的x86 Linux系统(Red Hat 6.1, Mandrake 7.0, Slackware 7.0及SuSE 6.1)
- 基于glibc2的x86 Linux系统(Red Hat 5.2, SuSE 6.1)
- 基于libc5的x86 Linux系统(Slackware 4.0, Debian 1.3.1r8)
- Sparc Solaris 2.6, 7和8
- FreeBSD 3.4和4.0
- Windows NT 4.0(不包括其它版本的Windows)

对PHP的要求如下:
- PHP 4.0.0:以外的版本均不被支持
- 同时支持CGI方式和Apache模块方式
- 在Windows下,PHP必须:1)是从http://www.php.net上下载的现成的WIN32执行版本;2)自己编译时带”Release_Ts”(Release Thread Safe)选项的。

安装过程

-UNIX
1 编译PHP,不要加调试选项-否则ZO不会工作:在配置是加上–disable-debug选项
2 复制ZendOptimizer.so文件到你的机器,通常放在:/usr/local/Zend/lib下
3 在php.ini文件中加入如下两行,不要包含任何空格:
zend_optimizer.optimization_level=7
zend_extension=”/usr/local/Zend/lib/ZendOptimizer.so”
4 重新启动Apache服务器

-WINDOWS
1 从http://www.php.net下载WINDOWS版的PHP 4.0.0,这个版本不包含调试特性。
2 复制ZendOptimizer.dll文件到你的机器,通常放在:C:\Program Files\Zend\lib下
3 在php.ini文件中加入如下两行,不要包含任何空格:
zend_optimizer.optimization_level=7
zend_extension_ts=”C:\Program Files\Zend\lib\ZendOptimizer.dll”
4 如果需要的话,重新启动WEB服务器

Zend Optmizer常见问题

问:什么是Zend Optimizer;有什么用?
答:ZO仔细检查有运行编译器产生的代码,分析并做优化,让它运行得更快。

问:为什么要用ZO,难道PHP4还不够快吗?
答:的确,用于PHP4的标准运行编译器已经够快了–相同情况下比PHP3要快2-10倍。但使用了ZO的PHP程序的执行速度还会加快40%到100%。

问:ZO在PHP4软件安装的什么位置?
答:ZO位于PHP4的ZEND引擎中,介于运行编译器和最终执行器之间。

问:ZO的最终发布版本要付费吗?
答:不要。ZO的最终版本将继续作为免费软件来发布。ZEND技术公司不打算对任何ZO版本收费。

问:开发人员如何获知ZO是否在运行?
答:只要用PHP的函数phpinfo()来检测便知,如下所示:
<?php
phpinfo();
?>
显示的结果中关于ZO的部分就象下面这样:
This program makes use of the Zend scripting language engine:
Zend Engine v1.0, Copyright © 1998-2000 Zend Technologies Ltd.
with Zend Optimizer v0.98, Copyright © 1998-2000 by Zend Technologies

php

phpbb mod for google 完全解决方案

Tags: , ,

前言

随着互联网上的内容以惊人速度的增长也越来越突出了搜索引擎的重要性,如果网站想更好地被搜索引擎收录,网站设计除了面向用户友 好(User Friendly)外,搜索引擎友好(Search Engine Friendly)的设计也是非常重要的。进入搜索引擎的页面内容越多,则被用户用不同的关键词找到的几率越大。不得不承认,将动态网页链接 rewriting成静态链接是最保险和稳定的面向搜索引擎优化方式。该方案就是针对phpBB论坛系统的URL重定向提出的。

解决方案

URL重定向从技术上将,目前可以通过两种方式实现,一种是基于URL rewrite,另一种是基于PATH_INFO。例如http://www.gbunix.com/bbs/ftopic102.html就是基于 rewrite实现的,而http://www.gbunix.com/article/article.php/515是基于PATH_INFO实现 的。

针对PHPBB论坛的改造,我们分别就这两种技术分别介绍。

一.使用rewrite技术实现:

修改phpBB代码:

打开/includes/page_header.php文件,

搜索代码:

//
// Generate logged in/logged out status
//

之前加:

ob_start();
function replace_for_mod_rewrite(&$s)
{
$urlin =
array(
“‘(?<!/)viewforum.php\?f=([0-9]*)&amp;topicdays=([0-9]*)&amp;start=([0-9]*)’”,
“‘(?<!/)viewforum.php\?f=([0-9]*)&amp;mark=topics’”,
“‘(?<!/)viewforum.php\?f=([0-9]*)’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)&amp;view=previous’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)&amp;view=next’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)&amp;postdays=([0-9]*)&amp;postorder=([a-zA-Z]*)&amp;start=([0-9]*)’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)&amp;start=([0-9]*)&amp;postdays=([0-9]*)&amp;postorder=([a-zA-Z]*)&amp;highlight=([a-zA-Z0-9]*)’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)start=([0-9]*)’”,
“‘(?<!/)viewtopic.php\?t=([0-9]*)’”,
“‘(?<!/)viewtopic.php&amp;p=([0-9]*)’”,
“‘(?<!/)viewtopic.php\?p=([0-9]*)’”,
);
$urlout = array(
“viewforum\\1-\\2-\\3.html”,
“forum\\1.html”,
“forum\\1.html”,
“ptopic\\1.html”,
“ntopic\\1.html”,
“ftopic\\1-\\2-\\3-\\4.html”,
“ftopic\\1.html”,
“ftopic\\1-\\2.html”,
“ftopic\\1.html”,
“sutra\\1.html”,
“sutra\\1.html”,
);
$s = preg_replace($urlin, $urlout, $s);
return $s;
}

打开/includes/page_tail.php文件,

搜索代码:

$db->sql_close();

之后加:

$contents = ob_get_contents();
ob_end_clean();
echo replace_for_mod_rewrite($contents);
global $dbg_starttime;

如果你的phpBB是2.06版本,打开includes/functions.php文件,

搜索代码:

if (!empty($db))
{
$db->sql_close();
}

之后加:

if (stristr($url, ‘http://’)) {
header(’Location: ‘ . $url);
exit;
}

最后在bbs目录下建立.htaccess 文件,文件内容为:

RewriteEngine On
RewriteRule ^forums.* index.php
RewriteRule ^forum([0-9]*).* viewforum.php?f=$1&mark=topic
RewriteRule ^viewforum([0-9]*)-([0-9]*)-([0-9]*).* viewforum.php?f=$1&topicdays=$2&start=$3
RewriteRule ^forum([0-9]*).* viewforum.php?f=$1
RewriteRule ^ptopic([0-9]*).* viewtopic.php?t=$1&view=previous
RewriteRule ^ntopic([0-9]*).* viewtopic.php?t=$1&view=next
RewriteRule ^ftopic([0-9]*)-([0-9]*)-([a-zA-Z]*)-([0-9]*).* viewtopic.php?t=$1&postdays=$2&postorder=$3&start=$4
RewriteRule ^ftopic([0-9]*)-([0-9]*).* viewtopic.php?t=$1&start=$2
RewriteRule ^ftopic([0-9]*).* viewtopic.php?t=$1
RewriteRule ^ftopic([0-9]*).html viewtopic.php?t=$1&start=$2&postdays=$3&postorder=$4&highlight=$5
RewriteRule ^sutra([0-9]*).* viewtopic.php?p=$1

如果你的服务器不支持.htaccess,请打开httpd.conf文件,编辑你的虚拟主机部分,如下:

<VirtualHost 1.2.3.4>
ServerAdmin webmaster@domain.com
DocumentRoot /home1/ftp/trotter/www
ServerName www.gbunix.com
RewriteEngine On
RewriteRule ^/bbs/forums.* /bbs/index.php
RewriteRule ^/bbs/forum([0-9]*).* /bbs/viewforum.php?f=$1&mark=topic
RewriteRule ^/bbs/viewforum([0-9]*)-([0-9]*)-([0-9]*).* /bbs/viewforum.php?f=$1&topicdays=$2&start=$3
RewriteRule ^/bbs/forum([0-9]*).* /bbs/viewforum.php?f=$1
RewriteRule ^/bbs/ptopic([0-9]*).* /bbs/viewtopic.php?t=$1&view=previous
RewriteRule ^/bbs/ntopic([0-9]*).* /bbs/viewtopic.php?t=$1&view=next
RewriteRule ^/bbs/ftopic([0-9]*)-([0-9]*)-([a-zA-Z]*)-([0-9]*).* /bbs/viewtopic.php?t=$1&postdays=$2&postorder=$3&start=$4
RewriteRule ^/bbs/ftopic([0-9]*)-([0-9]*).* /bbs/viewtopic.php?t=$1&start=$2
RewriteRule ^/bbs/ftopic([0-9]*).* /bbs/viewtopic.php?t=$1
RewriteRule ^/bbs/ftopic([0-9]*).html /bbs/viewtopic.php?t=$1&start=$2&postdays=$3&postorder=$4&highlight=$5
RewriteRule ^/bbs/sutra([0-9]*).* /bbs/viewtopic.php?p=$1
ErrorLog logs/gbunix.com-error_log
CustomLog logs/gbunix.com-access_log combined
</VirtualHost>

如果你用的不是虚拟主机,将RewriteRule部分代码放到httpd.conf文件最后就可以。

注意:非常重要的一点,为了系统的安全,请在bbs发布目录下建立robots.txt文件,文件内容如下:

Disallow: /your-forum-folder/sutra*.html$
Disallow: /your-forum-folder/ptopic*.html$
Disallow: /your-forum-folder/ntopic*.html$
Disallow: /your-forum-folder/ftopic*asc*.html$

给apache安装mod_rewrite模块

如果你的服务器apache还没有安装,那很简单,在编译apache时将mod_rewrite模块编译进去就可以,相关文档可以在 www.gbunix.com中找到。如果你的apache已经安装好了,现在只想编译出mod_rewrite.so模块,在apache中进行加载, 下面我们就介绍这个方法。

以Solaris操作系统进行举例:

# PATH=/usr/local/bin:/usr/sfw/bin:/usr/ccs/bin:$PATH
# export PATH
# which gcc
# which make

# find ./ -name mod_rewrite.c //在apache的安装目录中寻找mod_rewrite.c文件
# cd PATH/to/mod_rewrite.c //进入包含mod_rewrite.c文件的目录
# apxs -c mod_foo.c //apxs请指定绝对路径,在你当前正在使用apache的bin目录里
# apxs -i -a -n mod_rewrite mod_rewrite.la

如果没有什么错误的话,应该在你的apache的modules目录中编译出一个mod_rewrite.so文件。

编辑httpd.conf文件,确认httpd.conf中已经包含mod_rewrite.so的加载语句,如下:

LoadModule rewrite_module modules/mod_rewrite.so

这时,你的apache应该已经支持rewrite了。

二.基于PATH_INFO技术实现:

修改phpBB代码:

打开overall_header.tpl文件,在首行加如下代码:

<base href=”http://www.your-forum.com/forum-dir/”>

打开config.php文件,在?>前加入如下代码:

if ($REQUEST_METHOD == “GET”) {
if (strlen(getenv(’PATH_INFO’)) > 1) {
$GET_array = array();
$PHP_SELF = str_replace(getenv(’PATH_INFO’), ”, $PHP_SELF);
$vars = explode(’/', substr(getenv(’PATH_INFO’), 1));
for ($i=0, $n=sizeof($vars); $i<$n; $i++) {
if (strpos($vars[$i], ‘[]‘)) {
$GET_array[substr($vars[$i], 0, -2)][] = $vars[$i+1];
} else {
$HTTP_GET_VARS[$vars[$i]] = $vars[$i+1];
}
$i++;
}
if (sizeof($GET_array) > 0) {
while (list($key, $value) = each($GET_array)) {
$HTTP_GET_VARS[$key] = $value;
}
}
}

}

if ($REQUEST_METHOD == “POST”) {
if (strlen(getenv(’PATH_INFO’)) > 1) {
$POST_array = array();
$PHP_SELF = str_replace(getenv(’PATH_INFO’), ”, $PHP_SELF);
$vars = explode(’/', substr(getenv(’PATH_INFO’), 1));
for ($i=0, $n=sizeof($vars); $i<$n; $i++) {
if (strpos($vars[$i], ‘[]‘)) {
$POST_array[substr($vars[$i], 0, -2)][] = $vars[$i+1];
} else {
$HTTP_POST_VARS[$vars[$i]] = $vars[$i+1];
}
$i++;
}
if (sizeof($GET_array) > 0) {
while (list($key, $value) = each($POST_array)) {
$HTTP_POST_VARS[$key] = $value;
}
}
}

}

打开functions.php文件,在?>前加入如下代码:

function replace_for_mod_rewrite($s) {

$s = str_replace(”?”, “/”, $s);
$s = str_replace(”&amp;”, “/”, $s);
$s = str_replace(”&”, “/”, $s);
$s = str_replace(”=”, “/”, $s);
return $s;

}

打开sessions.php文件,用下面代码替换原来定义的append_sid()函数:

function append_sid($url, $non_html_amp = false)
{
global $SID;

if ( !empty($SID) && !preg_match(’#sid=#’, $url) && !preg_match(’#sid/#’, $url) && !stristr( $_SERVER["HTTP_USER_AGENT"] ,’bot’) && !stristr($_SERVER["HTTP_USER_AGENT"] ,’inktomi’))
{
$url .= ( ( strpos($url, ‘?’) != false ) ? ( ( $non_html_amp ) ? ‘&’ : ‘&amp;’ ) : ‘?’ ) . $SID ;
}
$url=replace_for_mod_rewrite($url);
return $url;
}

这时,你的论坛URL将会映射成(http://www.domain/bbs/viewtopic.php/t/4)这种方式。

php

用PHP调用Lucene包来实现全文检索

Tags: ,

由于工作需要,需要使用PHP实现对网站内大量数量进行全文检索,
而且目前最流行的全文检索的搜索引擎库就是Lucene了,
它是Apache Jakarta的一个子项目,并且提供了简单实用的API,
用这些API,就可以对任何基本文本的数据(包括数据库)进行全文检索。

因为PHP本身就支持调用外部Java类,所以先用Java写了一个类,
这个类通过调用Lucene的API,实现了两个方法:

  • public String createIndex(String indexDir_path,String dataDir_path)
  • public String searchword(String ss,String index_path)

其中createIndex是创建索引方法,
传入了两个参数分别是indexDir_path(索引文件的目录),dataDir_path(被索引的文件目录),返回被索引的文件列表字符串,
另一个是searchword,通过传入的关键字参数(ss)对索引进行检索,index_path就是索引文件的目录。返回所有检索到的文件。

这里是源代码,很简单,大家可以参考一下:TxtFileIndexer.java

而PHP程序就调用这两个方法,实现对Lucene的调用,从而达到全文检索的目的。
PHP的调用方法如下:
先创建一个我们写的TxtFileIndexer类的实例,

$tf = new Java(’TestLucene.TxtFileIndexer’);

然后就按正常PHP类的调用方法的方式进行调用,首先创建索引:

$data_path = “F:/test/php_lucene/htdocs/data/manual”;  //定义被索引内容的目录
$index_path = “F:/test/php_lucene/htdocs/data/search”;  //定义生成的索引文件存放目录
$s = $tf->createIndex($index_path,$data_path);  //调用Java类的方法
print $s;  //打印返回的结果

这次再试试检索:

$index_path = “F:/test/php_lucene/htdocs/data/search”;  //定义生成的索引文件存放目录
$s = $tf->searchword(”here is keyword for search”,$index_path);
print $s;

另外要注意Java类的路径,可以在PHP里设置

java_require(”F:/test/php_lucene/htdocs/lib/”);  //这是个例子,我的类和Lucene都放到这个目录下

这样就可以了,是不是很简单。

PHP源代码:test.php

接下来我把环境配置说一下,
首先需要有Java SDK,是必须的,我使用的是1.4.2版的,其它版本应该也没问题。
PHP5,试过PHP4,应该可以。

由于PHP5带的Java扩展没调通,并且以前用过调用Java效率很低,很慢,所以使用了Php/Java Bridge这个项目。

1.下载JavaBridge
URL:http://sourceforge.net/projects/php-java-bridge/
目前版本是
php-java-bridge_3.0.8_j2ee.zip

解包后把
JavaBridge\WEB-INF\cgi\java-x86-windows.dll
JavaBridge\WEB-INF\lib\JavaBridge.jar
复制到 c:\php\ext 目录下,并把
java-x86-windows.dll 改名为 php_java.dll

2.修改php.ini(例)
extension=php_java.dll

[Java]
java.class.path = “C:\php\ext\JavaBridge.jar;F:\test\php_lucene\htdocs”
java.java_home = “C:\j2sdk1.4.2_10″
java.library.path = “c:\php\ext;F:\test\php_lucene\htdocs”

3.重启Apache即可。

4.可以找一些文件进行索引
在test.php里可以修改索引文件和数据文件的路径。
TxtFileIndexer.java的37行限制了只索引html后缀的文件,有需要也可以修改。

根据目前的情况(JavaBridge支持Linux和Freebsd),完全可以在
linux或freebsd/apache2/php4/lucene/JavaBridge
环境下运行。

php

CSS反转字符加密邮箱地址

Tags: ,

为了防止爬虫软件搜索到你的Email地址,可以将Email倒着写,然后使用样式将其再正过来,代码如下:

效果如下:

我的邮箱moc.liamg@uoyroffeirg

<style type=”text/css”>
.backwards {unicode-bidi:bidi-override; direction: rtl; color:#c00;}
h2.email {color:#000; text-decoration:none; border:0; font-size:2em; font-weight:bold;}
</style>
<h2>我的邮箱 <span>moc.liamg@uoyroffeirg</span></h2>

php

php控制用户的浏览器

Tags: ,

用PHP控制用户的浏览器–ob*函数的使用
Output Control 函数可以让你自由控制脚本中数据的输出。它非常地有用,特别是对于:当你想在数据已经输出后,再输出文件头的情况。输出控制函数不对使用 header() 或 setcookie(), 发送的文件头信息产生影响,只对那些类似于 echo() 和 PHP 代码的数据块有作用。

我们先举一个简单的例子,让大家对Output Control有一个大致的印象:
Example 1.

<?php
ob_start(); //打开缓冲区
echo “Hellon”; //输出
header(”location:index.php”); //把浏览器重定向到index.php
ob_end_flush();//输出全部内容到浏览器
?>

所 有对header()函数有了解的人都知道,这个函数会发送一段文件头给浏览器,但是如果在使用这个函数之前已经有了任何输出(包括空输出,比如空格,回 车和换行)就会提示出错。如果我们去掉第一行的ob_start(),再执行此程序,我们会发现得到了一条错误提示:“Header had all ready send by”!但是加上ob_start,就不会提示出错,原因是当打开了缓冲区,echo后面的字符不会输出到浏览器,而是保留在服务器,直到你使用 flush或者ob_end_flush才会输出,所以并不会有任何文件头输出的错误!

一、 相关函数简介
1、Flush:刷新缓冲区的内容,输出。
函数格式:flush()
说明:这个函数经常使用,效率很高。

2、ob_start :打开输出缓冲区
函数格式:void ob_start(void)
说明:当缓冲区激活时,所有来自PHP程序的非文件头信息均不会发送,而是保存在内部缓冲区。为了输出缓冲区的内容,可以使用ob_end_flush()或flush()输出缓冲区的内容。

3 、ob_get_contents :返回内部缓冲区的内容。
使用方法:string ob_get_contents(void)
说明:这个函数会返回当前缓冲区中的内容,如果输出缓冲区没有激活,则返回 FALSE 。

4、ob_get_length:返回内部缓冲区的长度。
使用方法:int ob_get_length(void)
说明:这个函数会返回当前缓冲区中的长度;和ob_get_contents一样,如果输出缓冲区没有激活。则返回 FALSE。

5、ob_end_flush :发送内部缓冲区的内容到浏览器,并且关闭输出缓冲区。
使用方法:void ob_end_flush(void)
说明:这个函数发送输出缓冲区的内容(如果有的话)。

6、ob_end_clean:删除内部缓冲区的内容,并且关闭内部缓冲区
使用方法:void ob_end_clean(void)
说明:这个函数不会输出内部缓冲区的内容而是把它删除!

7、ob_implicit_flush:打开或关闭绝对刷新
使用方法:void ob_implicit_flush ([int flag])
说明:使用过Perl的人都知道$|=x的意义,这个字符串可以打开/关闭缓冲区,而ob_implicit_flush函数也和那个一样,默认为关闭缓冲区,打开绝对输出后,每个脚本输出都直接发送到浏览器,不再需要调用 flush()

二、深入了解

1. 关于Flush函数:
这个函数在PHP3中就出现了,是一个效率很高的函数,他有一个非常有用的功能就是刷新browser的cache.我们举一个运行效果非常明显的例子来说明flush.
Example 2.

<?php
for($i = 1; $i <= 300; $i ) print(” “);
// 这一句话非常关键,cache的结构使得它的内容只有达到一定的大小才能从浏览器里输出
// 换言之,如果cache的内容不达到一定的大小,它是不会在程序执行完毕前输出的。经
// 过测试,我发现这个大小的底限是256个字符长。这意味着cache以后接收的内容都会
// 源源不断的被发送出去。
for($j = 1; $j <= 20; $j ) {
echo $j . ”
“;
flush(); //这一部会使cache新增的内容被挤出去,显示到浏览器上
sleep(1); //让程序”睡”一秒钟,会让你把效果看得更清楚
}

?>

具体效果你可以到这里看看http://www.php2000.com/~uc…
PHP2000的最新的PHP聊天室就是用的这个技术,可惜的是源代码未公开 L
注:如果在程序的首部加入ob_implicit_flush()打开绝对刷新,就可以在程序中不再使用flush(),这样做的好处是:提高效率!

2. 关于ob系列函数:
我想先引用我的好朋友y10k的一个例子:
Example 3.
比如你用得到服务器和客户端的设置信息,但是这个信息会因为客户端的不同而不同,如果想要保存phpinfo()函数的输出怎么办呢?在没有缓冲区控制之前,可以说一点办法也没有,但是有了缓冲区的控制,我们可以轻松的解决:

<?php
ob_start(); //打开缓冲区
phpinfo(); //使用phpinfo函数
$info = ob_get_contents(); //得到缓冲区的内容并且赋值给$info
$file = fopen(’info.txt’, ‘w’); //打开文件info.txt
fwrite($file, $info); //写入信息到info.txt
fclose($file); //关闭文件info.txt

?>

用以上的方法,就可以把不同用户的phpinfo信息保存下来,这在以前恐怕没有办法办到!其实上面就是将一些“过程”转化为“函数”的方法!
或 许有人会问:“难道就这个样子吗?还有没有其他用途?”当然有了,比如笔者论坛的PHP 语法加亮显示就和这个有关(PHP默认的语法加亮显示函数会直接输出,不能保存结果,如果在每次调用都显示恐怕会很浪费CPU,笔者的论坛就把语法加亮函 数显示的结果用控制缓冲区的方法保留了),大家如果感兴趣的话可以来看看http://www.zphp.com/bbs/

可能现在大家对ob_start()的功能有了一定的了解,上面的一个例子看似简单,但实际上已经掌握了使用ob_start()的要点。
<1>.使用ob_start打开browser的cache,这样可以保证cache的内容在你调用flush(),ob_end_flush()(或程序执行完毕)之前不会被输出。
<2>. 现在的你应该知道你所拥有的优势:可以在任何输出内容后面使用header,setcookie以及session,这是 ob_start一个很大的特点;也可以使用ob_start的参数,在cache被写入后,然后自动运行命令,比如ob_start (”ob_gzhandler”);而我们最常用的做法是用ob_get_contents()得到cache中的内容,然后再进行处理……
<3>.当处理完毕后,我们可以使用各种方法输出,flush(),ob_end_flush(),以及等到程序执行完毕后的自动输出。当然,如果你用的是ob_get_contents(),那么就要你自己控制输出方式了。

来,让我们看看能用ob系列函数做些什么……

(一)、 静态模版技术

简 介:所谓静态模版技术就是通过某种方式,使得用户在client端得到的是由PHP产生的html页面。如果这个html页面不会再被更新,那么当另外的 用户再次浏览此页面时,程序将不会再调用PHP以及相关的数据库,对于某些信息量比较大的网站,例如sina,163,sohu。类似这种的技术带来的好 处是非常巨大的。

我所知道的实现静态输出的有两种办法:
<1>.通过y10k修改的phplib的一个叫template.inc.php类实现。
<2>.使用ob系列函数实现。
对于第一种方法,因为不是这篇文章所要研究的问题,所以不再赘述。
我们现在来看一看第二种方法的具体实现:
Example 4.

<?php
ob_start();//打开缓冲区
?>

php页面的全部输出

<?php
$content = ob_get_contents(); //取得php页面输出的全部内容
$fp = fopen(”output00001 . html”, “w”); //创建一个文件,并打开,准备写入
fwrite($fp, $content); //把php页面的内容全部写入output00001.html,然后……
fclose($fp);
?>

这样,所谓的静态模版就很容易的被实现了……

(二)、 捕捉输出

以上的Example 4.是一种最简单的情况,你还可以在写入前对$content进行操作……
你可以设法捕捉一些关键字,然后去对它进行再处理,比如Example 3.所述的PHP语法高亮显示。个人认为,这个功能是此函数最大的精华所在,它可以解决各种各样的问题,但需要你有足够的想象力……
Example 5.

<?php
function run_code($code) {
If ($code) {
ob_start();
eval($code);
$contents = ob_get_contents();
ob_end_clean();
} else {
echo “错误!没有输出”;
exit();
}
return $contents;
}

?>

以上这个例子的用途不是很大,不过很典型$code的本身就是一个含有变量的输出页面,而这个例子用eval把$code中的变量替换,然后对输出结果再进行输出捕捉,再一次的进行处理……

Example 6. 加快传输

<?php
/*
** Title………: PHP4 HTTP Compression Speeds up the Web
** Version…….: 1.20
** Author……..: catoc <catoc@163.net>
** Filename……: gzdoc.php
** Last changed..: 18/10/2000
** Requirments…: PHP4 >= 4.0.1
** PHP was configured with –with-zlib[=DIR]
** Notes………: Dynamic Content Acceleration compresses
** the data transmission data on the fly
** code by sun jin hu (catoc) <catoc@163.net>
** Most newer browsers since 1998/1999 have
** been equipped to support the HTTP 1.1
** standard known as “content-encoding.”
** Essentially the browser indicates to the
** server that it can accept “content encoding”
** and if the server is capable it will then
** compress the data and transmit it. The
** browser decompresses it and then renders
** the page.
**
** Modified by John Lim (jlim@natsoft.com.my)
** based on ideas by Sandy McArthur, Jr
** Usage……..:
** No space before the beginning of the first ‘<?’ tag.
** ————Start of file———-
** |<?
** | include(’gzdoc.php’);
** |? >
** |<HTML>
** |… the page …
** |</HTML>
** |<?
** | gzdocout();
** |? >
** ————-End of file———–
*/
ob_start();
ob_implicit_flush(0);
function CheckCanGzip() {
global $HTTP_ACCEPT_ENCODING;
if (headers_sent() || connection_timeout() || connection_aborted()) {
return 0;
}
if (strpos($HTTP_ACCEPT_ENCODING, ‘x-gzip’) !== false) return “x-gzip”;
if (strpos($HTTP_ACCEPT_ENCODING, ‘gzip’) !== false) return “gzip”;
return 0;
}
/* $level = compression level 0-9, 0=none, 9=max */
function GzDocOut($level = 1, $debug = 0) {
$ENCODING = CheckCanGzip();
if ($ENCODING) {
print “n<!– Use compress $ENCODING –>n”;
$Contents = ob_get_contents();
ob_end_clean();
if ($debug) {
$s = “<p>Not compress length: ” . strlen($Contents);
$s .= “Compressed length: ” . strlen(gzcompress($Contents, $level));
$Contents .= $s;
}
header(”Content-Encoding: $ENCODING”);
print “x1fx8bx08×00x00×00x00×00″;
$Size = strlen($Contents);
$Crc = crc32($Contents);
$Contents = gzcompress($Contents, $level);
$Contents = substr($Contents, 0, strlen($Contents) – 4);
print $Contents;
print pack(’V', $Crc);
print pack(’V', $Size);
exit;
} else {
ob_end_flush();
exit;
}
}

?>

这是catoc的一段很早以前的代码,是在weblogs.com看到的,他利用了zlib的函数,对传输的内容进行了压缩,测试表明,对于10k以上的页面,会产生效果,而且页面越大,效果越明显……

燕之庐 网站建设公司

php

缓存PHP页面的代码,可以减轻CPU和MYSQL负担

Tags: ,

使用前,先在根目录创建“cache”文件夹,然后运行1.php,第一次运行和第二次运行速度差异很大。
欢迎熟悉PHP的朋友使用和提意见。

使用方法:(请保存为1.php)

CODE:
<?
include(’arrcache.php’);
$cache = new ArrCache(’cache’,5,’txt’);

for ($i=0;$i<5;$i++)
{
echo $i;
sleep(1);
}

$cache->endCache();
?>

源代码:(请保存为’arrcache.php’)

CODE:
<?
/*
* 名称: 极光缓存
*
* 作者: DCD
* mail:haodan-_-b@163.com
*
* web开发群: 3574636
*http://www.525ok.com
*
* 完成日期: 2006-2-27
*
* 版本:V1.00
*
* 说明:
*
* 缓存网页内容,减轻服务器CPU负荷,使用简便、灵活。
* 本程序主要对MYSQL的select操作进行优化,还有部分小偷类程序。
* 比如在百度搜索一个不常见的关键字,第一次搜索速度非常慢,“找到相关网页181篇,用时0.426秒”
* 那么再搜索一次,就会变成“找到相关网页181篇,用时0.001秒”。
* 本程序就是使用了类似的方法缓存数据到服务器硬盘,减轻CPU和数据库负荷。
* 请根据您页面日访问量来设定缓存时间。
* 使用前,一定先创建好缓存目录。
*
* 注意:
*
* 不是所有页面都需要使用本程序。
* 缓存文件会占用一定硬盘空间。
* 例如需要对数据库进行inster、delete、update操作的页面不要使用,select可以。
* mail()、fwrite()等函数,将不产生任何作用。
* 请不要将缓存时间设置得太小(小于5秒),那样不会减轻负荷
* 如果你的页面访问量不太高,也不要使用本代码。
*
* 使用方法:
*
* include(’arrcache.php’);//必须在文件最开始处包含本页
* $cache = new ArrCache(’cache’);//设置缓存文件夹
* ……
* ……
* ……//代码
* $cache->endCache();//在文件最后写入
*
* 说明:
*
* class ArrCache(string path,[int time],[string type]);
* path: cache文件保存目录,从根目录算起,首尾不需要“/”
* time: 缓存时间,默认120秒
* type: 缓存文件后缀,默认“txt”
*
* void endCache();
* 在页面最后写上,否则这行后边的数据不会被缓存。
*
*/
ob_start();
class ArrCache
{
//构造函数
function ArrCache($path,$time = 120,$type = ‘txt’)
{
$this->path = $path;
$this->time = $time;
$this->fileType = $type;
$this->fileName = $_SERVER['DOCUMENT_ROOT'].’\\’.$this->path.’\\’.md5($_SERVER['URL'].’?’.$_SERVER['QUERY_STRING']).’.’.$this->fileType;
if (file_exists($this->fileName) && ((filemtime($this->fileName)+$this->time) > time()))
{
$fp = fopen($this->fileName,”r”);
echo fread($fp,filesize($this->fileName));
fclose($fp);
ob_end_flush();
exit;
}
}
//在文件最后加入这行,输出所有缓存内容,并且写入缓存文件。
function endCache()
{
$fp = fopen($this->fileName,”w”);
fwrite($fp,ob_get_contents());
fclose($fp);
ob_end_flush();
}
}//end class
?>
燕之庐 网站建设公司