只需编辑这一个模板文件就可以改变网站所有页面左边的导航条。
避免页面元素重复
“这确实不错”,你也许会想,“我的网站主要就是由大量的静态页面构成。现在我可以从所有页
面中删除它们的公共部分,要更新这些公共部分实在太麻烦了。以后我就可以用模板制作出很容易维护
的统一页面布局。”但事情并非这么简单,“大量的静态页面”道出了问题的所在。
请考虑上面的例子。这个例子实际上只有一个example.php页面,它之所以能够生成整个网站的所
有页面,是因为它利用了URL中的查询字符串从数据库之类的信息源动态地构造出页面。
我们之中的大多数人所运行的网站并不一定都有数据库支持。我们的网站大多数由静态页面构成,
然后用PHP在这里、那里加上一些动态功能,比如搜索引擎、反馈表单等。那么,如何在这种网站上应用
模板呢?
最简单的方法是为每一个页面复制一份PHP文件,然后在每一个页面中把PHP代码里代表内容的变量
设置成合适的页面内容。例如,假设有三个页面,它们分别是主页(home)、关于(about)和产品
(product),我们可以用三个文件分别生成它们。这三个文件的内容都类如:
<?php
// home.php
require('class.FastTemplate.php');
$tpl = new FastTemplate('.');
$tpl->define( array( 'main' => 'main.htm',
'header' => 'header.htm',
'leftnav' => 'leftnav.htm' ) );
$content = "<p>欢迎访问</p>
<img src=/"demo.jpg/">
<p>希望你能够喜欢本网站</p>";
$tpl->assign('CONTENT', $content);
$tpl->parse('HEADER', 'header');
$tpl->parse('LEFTNAV', 'leftnav');
$tpl->parse('MAIN', 'main');
$tpl->FastPrint('MAIN');
?>
显然,这种方法有三个问题:我们必须为每一个页面复制这些复杂的、牵涉到模板的PHP代码,这
与重复公共页面元素一样使得页面难以维护;现在文件又混合了HTML和PHP代码;为内容变量赋值将变得
非常困难,因为我们必须处理好大量的特殊字符。
解决这个问题的关键就在于分离PHP代码和HTML内容,虽然我们不能从文件中删除所有的HTML内容
,但可以移出绝大多数PHP代码。
静态网站的模板框架
首先,我们象前面一样为所有的页面公用元素以及页面整体布局编写模板文件;然后从所有的页面
删除公共部分,只留下页面内容;接下来再在每个页面中加上三行PHP代码,如下所示:
<?php
<!-- home.php -->
<?php require('prepend.php'); ?>
<?php pageStart('Home'); ?>
<h1>你好</h1>
<p>欢迎访问</p>
<img src="demo.jpg">
<p>希望你能够喜欢本网站</p>
<?php pageFinish(); ?>
?>
这种方法基本上解决了前面提到的各种问题。现在文件里只有三行PHP代码,而且没有任何一行代码
直接涉及到模板,因此要改动这些代码的可能性极小。此外,由于HTML内容位于PHP标记之外,所以也不
存在特殊字符的处理问题。我们可以很容易地将这三行PHP代码加入到所有静态HTML页面中。
require函数引入了一个PHP文件,这个文件包含了所有必需的与模板相关的PHP代码。其中
pageStart函数设置模板对象以及页面标题,pageFinish函数解析模板然后生成结果发送给浏览器。
这是如何实现的呢?为什么在调用pageFinish函数之前文件中的HTML不会发送给浏览器?答案就在
于PHP 4的一个新功能,这个功能允许把输出到浏览器的内容截获到缓冲区之中。让我们来看看
prepend.php的具体代码:
<?php
require('class.FastTemplate.php');
function pageStart($title = '') {
GLOBAL $tpl;
$tpl = new FastTemplate('.');
$tpl->define( array( 'main' => 'main.htm',
'header' => 'header.htm',
'leftnav'=> 'leftnav.htm' ) );
$tpl->assign('TITLE', $title);
ob_start();
}
function pageFinish() {
GLOBAL $tpl;
$content = ob_get_contents();
ob_end_clean();
$tpl->assign('CONTENT', $content);
$tpl->parse('HEADER', 'header');
$tpl->parse('LEFTNAV', 'leftnav');
$tpl->parse('MAIN', 'main');
$tpl->FastPrint('MAIN');
}
?>
pageStart函数首先创建并设置了一个模板实例,然后启用输出缓存。此后,所有来自页面本身的
HTML内容都将进入缓存。pageFinish函数取出缓存中的内容,然后在模板对象中指定这些内容,最后解
析模板并输出完成后的页面。
这就是整个模板框架全部的工作过程了。首先编写包含了网站各个页面公共元素的模板,然后从所
有页面中删除全部公共的页面布局代码,代之以三行永远无需改动的PHP代码;再把FastTemplate类文件
和prepend.php加入到包含路径,这样你就得到了一个页面布局可以集中控制的网站,它有着更好的可靠
性和可维护性,而且网站级的大范围修改也变得相当容易。
本文下载包包含了一个可运行的示例网站,它的代码注释要比前面的代码注释更详细一些。
FastTemplate类可以在http://www.thewebmasters.net/找到,最新的版本号是1.1.0,那里还有一个用
于保证该类在PHP 4中正确运行的小补丁。本文下载代码中的类已经经过该补丁的修正。
PHP简易生成静态页面
<?php
/*
* 文件名:index.php
*/
require "conn.php";
$query = "select * from news order by datetime desc";
$result = mysql_query($query);
?>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=??????">
<title>NEWS</title>
</head>
<body>
<table width="500" border="1" align="center">
<tr>
<td>标题</td>
<td width="200">发布时间</td>
</tr>
<?
while($re = mysql_fetch_array($result)){
?>
<tr>
<td><a href="<?= $re["newsid"].".html"?>"><?= $re["title"]?></a></td>
<td><?= $re["datetime"]?></td>
</tr>
<?
}
?>
<tr>
<td> </td>
<td><a href="addnews.php">添加新闻</a></td>
</tr>
</table>
</body>
</html>
--------------------------------------------------------------------------
<?php
/*
文件名:AddNews.php
简易动态添加生成静态新闻页面
#
# 表的结构 `news`
#
CREATE TABLE `news` (
`newsid` int(11) NOT NULL auto_increment,
`title` varchar(100) NOT NULL default '',
`content` text NOT NULL,
`datetime` datetime NOT NULL default '0000-00-00 00:00:00',
KEY `newsid` (`newsid`)
) TYPE=MyISAM AUTO_INCREMENT=11 ;
*/
if(isset($_POST["title"])){
$title = $_POST["title"];
$content = $_POST["content"];
//定义模版文件的内容,可用其它方法
$filecontent = '<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=??????">
<title>{title}</title>
</head>
<body>
<table width="500" border="1" align="center">
<tr>
<td align="center"><strong>{title}</strong></td>
</tr>
<tr>
<td>{content}</td>
</tr>
<tr>
<td align="right">{datetime}</td>
</tr>
</table>
</body>
</html>
';
$datetime = date("Y-m-d H:i:s");
$query = "insert into news values('','".$title."','".$content."','".$datetime."')";
require "conn.php";//连接数据库
$result = mysql_query($query) or die(mysql_error());
if($result){
$id = mysql_insert_id();
$filename = $id.".html";
if($fp = fopen($filename, "w")){//新建文件,成功后添加内容
$filecontent = str_replace("{title}", $title, $filecontent);
$filecontent = str_replace("{content}", $content, $filecontent);
$filecontent = str_replace("{datetime}", $datetime, $filecontent);
if(!fwrite($fp, $filecontent)){//把内容写入文件
$query = "delete from news where newsid=".$id;
$result = mysql_query($query) or die(mysql_error());
fclose($fp);
unlink($filename);
echo "<script>alert('Add news
failed!');location.href='index.php';</script>";
exit;
}else {
echo "<script>alert('Add news
successed!');location.href='index.php';</script>";
}
}
}
}
?>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=??????">
<title>添加</title>
</head>
<body>
<form name="form1" method="post" action="">
<table width="500" border="1" align="center">
<tr>
<td>标题</td>
<td><input name="title" type="text" id="title" size="40"></td>
</tr>
<tr>
<td>内容</td>
<td><textarea name="content" cols="45" rows="6" id="content"></textarea></td>
</tr>
<tr>
<td> </td>
<td><input type="submit" name="Submit" value="提交"></td>
</tr>
</table>
</form>
</body>
</html>
PHP生成静态页面详解
应用实例:
http://xjtusky.net/down
说明:有一部分功能还没有完全做好,修改中,有兴趣的朋友可以看看下面这篇文章,看
完了一定要说好!生成页面的速度快的~~~呵呵。下面文章中一些地方请自行修改!
PHP生成静态页面详解
http://xjtusky.net/article/htmld ... /article_594_1.html
看到很多朋友在各个地方发帖问PHP生成静态文章系统的方法,以前曾做过这样一个系
统,遂谈些看法,以供各位参考。好了,我们先回顾一些基本的概念。
一,PHP脚本与动态页面。
PHP脚本是一种服务器端脚本程序,可通过嵌入等方法与HTML文件混合,也可以类,函
数封装等形式,以模板的方式对用户请求进行处理。无论以何种方式,它的基本原理是这
样的。由客户端提出请求,请求某一页面 -----> WEB服务器引入指定相应脚本进行处理
-----> 脚本被载入服务器 -----> 由服务器指定的PHP解析器对脚本进行解析形成H
TML语言形式 ----> 将解析后的HTML语句以包的方式传回给浏览器。由此不难看出,在
页面发送到浏览器后,PHP就不存在了,已被转化解析为HTML语句。客户请求为一动态文件
,事实上并没有真正的文件存在在那里,是PHP解析而成相对应的页面,然后发送回浏览器
。这种页面处理方式被称为“动态页面”。
二,静态页面。
静态页面是指在服务器端确实存在的仅含HTML以及JS,CSS等客户端运行脚本的页面。
它的处理方式是。由客户端提出请求,请求某一页面 ----> WEB服务器确认并载入某一
页面 ----> WEB服务器将该页面以包的形式传递回浏览器。由这一过程,我们对比一下
动态页面,即可方现。动态页面需由WEB服务器的PHP解析器进行解析,而且通常还需连接
数据库,进行数据库存取操作,然后才能形成HTML语言信息包;而静态页面,无须解析,
无须连接数据库,直接发送,可大大减轻服务器压力,提高服务器负载能力,大幅提供页
面打开速度和网站整体打开速度。但其缺点是,不能动态地对请求进行处理,服务器上必
须确实存在该文件。
三,模板及模板解析。
模板即尚未填充内容html文件。例如:
Code: [点击这里把以下内容复制到剪贴板]
temp.html
<HTML>
<TITLE>{title}</TITLE>
<BODY>
this is a {file} file's templets
</BODY>
</HTML>
PHP处理:
templetest.php
<?php
$title = "拓迈国际测试模板";
$file = "TwoMax Inter test templet,<br>author:Matrix@Two_Max";
$fp = fopen ("temp.html","r");
$content = fread ($fp,filesize ("temp.html"));
$content .= str_replace ("{file}",$file,$content);
$content .= str_replace ("{title}",$title,$content);
echo $content;
?>
模板解析处理,即将经PHP脚本解析处理后得出的结果填充(content)进模板的处理过
程。通常借助于模板类。目前较流行的模板解析类有phplib,smarty,fastsmarty等等。
模板解析处理的原理通常为替换。也有些程序员习惯将判断,循环等处理放进模板文件中
,用解析类处理,典型应用为block概念,简单来说即为一个循环处理。由PHP脚本指定循
环次数,如何循环代入等,再由模板解析类具体实施这些操作。
好了,对比过静态页面与动态页面各自的优劣,现在我们就来说说,如何用PHP生成静
态文件。
PHP生成静态页面并不是指PHP的动态解析,输出HTML页面,而是指用PHP创建HTML页面
。同时因为HTML的不可写性,我们创建的HTML若有修改,则需删掉重新生成即可。(当然你
也可以选择用正则进行修改,但个人认为那样做倒不如删掉重新生成来得快捷,有些得不
偿失。)
言归正传。用过PHP文件操作函数的PHP FANS知道,PHP中有一个文件操作函数fopen,
即打开文件。若文件不存在,则尝试创建。这即是PHP可以用来创建HTML文件的理论基础。
只要用来存放HTML文件的文件夹有写权限(即权限定义0777),即可创建文件。(针对UNIX
系统而言,Win系统无须考虑。)仍以上例为例,若我们修改最后一句,并指定在test目录
下生成一个名为test.html的静态文件:
Code: [点击这里把以下内容复制到剪贴板]
<?php
$title = "拓迈国际测试模板";
$file = "TwoMax Inter test templet,<br>author:Matrix@Two_Max";
$fp = fopen ("temp.html","r");
$content = fread ($fp,filesize ("temp.html"));
$content .= str_replace ("{file}",$file,$content);
$content .= str_replace ("{title}",$title,$content);
// echo $content;
$filename = "test/test.html";
$handle = fopen ($filename,"w"); //打开文件指针,创建文件
/*
检查文件是否被创建且可写
*/
if (!is_writable ($filename)){
die ("文件:".$filename."不可写,请检查其属性后重试!");
}
if (!fwrite ($handle,$content)){ //将信息写入文件
die ("生成文件".$filename."失败!");
}
fclose ($handle); //关闭指针
die ("创建文件".$filename."成功!");
?>
实际应用中常见问题解决方案参考:
一,文章列表问题:
在数据库中创建字段,记录文件名,每生成一个文件,将自动生成的文件名存入数据
库,对于推荐文章,只需指向存放静态文件的指定文件夹中的该页面即可。利用PHP操作处
理文章列表,存为字符串,生成页面时替换此字符串即可。如,在页面中放置文章列表的
表格加入标记{articletable},而在PHP处理文件中:
Code: [点击这里把以下内容复制到剪贴板]
<?php
$title = "拓迈国际测试模板";
$file = "TwoMax Inter test templet,<br>author:Matrix@Two_Max";
$fp = fopen ("temp.html","r");
$content = fread ($fp,filesize ("temp.html"));
$content .= str_replace ("{file}",$file,$content);
$content .= str_replace ("{title}",$title,$content);
// 生成列表开始
$list = '';
$sql = "select id,title,filename from article";
$query = mysql_query ($sql);
while ($result = mysql_fetch_array ($query)){
$list .= '<a href='.$root.$result['filename'].' target=_blank>'.$result[
'title'].'</a><br>';
}
$content .= str_replace ("{articletable}",$list,$content);
//生成列表结束
// echo $content;
$filename = "test/test.html";
$handle = fopen ($filename,"w"); //打开文件指针,创建文件
/*
检查文件是否被创建且可写
*/
if (!is_writable ($filename)){
die ("文件:".$filename."不可写,请检查其属性后重试!");
}
if (!fwrite ($handle,$content)){ //将信息写入文件
die ("生成文件".$filename."失败!");
}
fclose ($handle); //关闭指针
die ("创建文件".$filename."成功!");
?>
二,分页问题。
如我们指定分页时,每页20篇。某子频道列表内文章经数据库查询为45条,则,首先
我们通过查询得到如下参数:1,总页数;2,每页篇数。第二步,for ($i = 0; $i < al
lpages; $i++),页面元素获取,分析,文章生成,都在此循环中执行。不同的是,die (
"创建文件".$filename."成功!";这句去掉,放到循环后的显示,因为该语句将中止程序
执行。例:
Code: [点击这里把以下内容复制到剪贴板]
<?php
$fp = fopen ("temp.html","r");
$content = fread ($fp,filesize ("temp.html"));
$onepage = '20';
$sql = "select id from article where channel='$channelid'";
$query = mysql_query ($sql);
$num = mysql_num_rows ($query);
$allpages = ceil ($num / $onepage);
for ($i = 0;$i<$allpages; $i++){
if ($i == 0){
$indexpath = "index.html";
} else {
$indexpath = "index_".$i."html";
}
$start = $i * $onepage;
$list = '';
$sql_for_page = "select name,filename,title from article where channel='
$channelid' limit $start,$onepage";
$query_for_page = mysql_query ($sql_for_page);
while ($result = $query_for_page){
$list .= '<a href='.$root.$result['filename'].' target=_blank>'.$titl
e.'</a><br>';
}
$content = str_replace ("{articletable}",$list,$content);
if (is_file ($indexpath)){
@unlink ($indexpath); //若文件已存在,则删除
}
$handle = fopen ($indexpath,"w"); //打开文件指针,创建文件
/*
检查文件是否被创建且可写
*/
if (!is_writable ($indexpath)){
echo "文件:".$indexpath."不可写,请检查其属性后重试!"; //修改为echo
}
if (!fwrite ($handle,$content)){ //将信息写入文件
echo "生成文件".$indexpath."失败!"; //修改为echo
}
fclose ($handle); //关闭指针
}
fclose ($fp);
die ("生成分页文件完成,如生成不完全,请检查文件权限系统后重新生成!");
?>
大致思路如此,其中如其它数据生成,数据输入输出检查,分页内容指向等可酌情在
页面中加入。
在实际文章系统处理过程当中,还有许多问题有待考虑,与动态页面不同之处,需注
意的地方还有很多。但大致思路即是如此,其它方面可举一反三而得。
目前网络上好多网站的新闻发布系统都采用了 动态服务器技术生成静态HTML的做法,这样做的好处是:一是能减轻其服务器的负担,二是因为生成了HTML静态页面,所以其网站被搜索引擎搜索到的机率 更大一些。笔者的网站曾经使用PHP这一动态技术来构建新闻发布系统,其原理也就是应用了PHP生成HTML静态页面的技术,相关平台是 Windows XP Sp2+php4.32+mysql,因此,在这里,想简单地谈一下这种做法的思路。这篇文章适合于对PHP+MYSQL数据库操作,SQL语句以及网页 设计有点基础的朋友,如果您是一个从头开始学的朋友,那么请先打好基础吧!到这里就不用往下看了。如果您都符合上述条件的话,那么恭喜您,请接着往下看。 但是,在具体动手构建之前,您还要做好以下几点准备工作。
一、 具备本地调试PHP的功能
在WINDOWS XP操作系统下,笔者建议你可以到网上下载一个PHP+MYSQL+APHCHE的服务器套装,如华军软件园,去那里搜索一下就可以下载到。下载完后默认 安装即可,这样您就具备了在本地测试PHP的功能了,省去了很多手动配置的麻烦,怎么样,简单吧,OK,这仅仅只是第一步。
二、 构思新闻发布系统所具备的功能
首页的新闻发布往往是通过后台更新的,后台的更新无非是由添加、编辑、删除数据等基本的功能所实现的。在这里,你可以使用网页设计软件来构建你的想 要的后台界面,其功能的实现当然是要用到PHP喽。这一步建议您先想好新闻发布系统要有的功能。在这里,如何用PHP添加、编辑、删除数据就不再重复,因 为重点是如何在这基础上生成静态技术。
三、 PHP生成HTML的技术原理。
哈哈。费话说了那么多,终于到了该讲的地方了。其实,这一原理并不复杂,综合地来讲的话,应该是PHP中一个替换数据语法的应用。OK,讲一个简单 范例吧,一步一步地分析!相信聪明的你能看得明白的咯,仔细看好每一步即可,在这里,只是引导大家如何做而已,具体的可以实践一下!
(1)在MYSQL里新建一数据库,命名为 database (可自定义),新建一表,命名为 news(因为是新闻发布嘛,取个好记的名字即可,可自定义),然后建立这几个字段名:
id (自动递增,这是关键,类型:INT)
title (顾名思义,新闻标题,类型可取 TEXT)
content (新闻内容,类型可取 TEXT)
path (HTML文件路径,类型可取 TEXT)
(2)建立 conn.php
这是连接数据库的PHP文件,你可以把连接数据的语句单独放在这一文件里,以后多个需要连接数据库的文件直接引用这个文件即可。
(3)设计添加新闻的表格 add.form 简单的源代码如下:
<form method=”post” action=”add.php”> //提交至 add.php
新闻标题:<input type=”text” name=”title” size=”20”><br>
新闻内容:<textarea name=”content” cols=”10” rows=”25”></textarea><br>
<input type=”submit” name=”提交”>
</form>
(4)建立一个 HTML 的模板,另存为model.htm,和 add.php可以在同一目录下。
示例源代码:
<html>
<body>
此新闻的标题:{title}
此新闻的内容:{content}
</body>
</html>
{ }大括号内的内容即是要被替换的内容,整个静态模板的设计可以根据自己的思路,但{ }内被替换的内容必须包含在内,如上面的{title},{content};咔咔~简单地说,设计好一个很好看的新闻模板后,把要被替换的如 {title},{content}等标记放到需要的地方就可以了撒。
(5)详解 add.php 源码
<?php
require_once(“conn.php”); //引用conn.php,连接数据库
$title=$_POST[“title”];
$content=$_POST[“content”]; //获得表单变量
//以下建立一文本文档,其值自动计数
$countfile="count.txt";
if(!file_exists($countfile))
{
fopen($countfile,"w"); //如果此文件不存在,则自动建立一个
}
$fp=fopen($countfile,"r");
$num=fgets($fp,20);
$num=$num+1; //每次其值自动加一
fclose($fp);
$fp=fopen($countfile,"w");
fwrite($fp,$num); //更新其值
fclose($fp);
//利用上面自动计数的值获得HTML的路径$path
$houzui=”.html”;
$path=$num.$houzui;
//这样形成的路径是自动增长的,如1.html,2.html,3.html……….添加一条新闻便自动加上1
//以下用SQL语句添加数据至表 news
$sql=”insert into news (title,content,path) values (‘”.$title.”’,’”.$content.”’,’”.$path.”’)”;
$query=mysql_query($sql);
//以下为关键之处,把从表单获得的数据替换模板中的{title},{content}标记 $fp=fopen(“model.htm”,”r”) //只读打开模板
$str=fread($fp,filesize(“mode.htm”));//读取模板中内容
$str=str_replace(“{title}”,$title,$str);
$str=str_replace(“{content}”,$content,$str);//替换内容
fclose($fp);
$handle=fopen($path,”w”); //写入方式打开新闻路径
fwrite($handle,$str); //把刚才替换的内容写进生成的HTML文件
fclose($handle);
//收尾工作:
echo “<a href=$path target=_blank>查看刚才添加的新闻</a>”;
OK,整个生成HTML的示例源码就到这里,其关键是用了替换的方法。
$str=str_replace(“{被替换的内容}”,$替换的内容,$str);
因此,总结一下以上的做法:先设计好新闻模板,把需要被替换的内容用{ }放到模板中相应的位置,然后设计表单,再是最后的表单处理程序,把从表单中获取的变量替换模板中相应的内容即可,这样每次都会生成不同的HTML;如果 需要修改HTML的内容也是一样的,获得修改后的表单内容后,先用 update 语句更新数据库,再重新替换一下模板中的内容即可;删除的话,先delete表中要删除的内容,再用unlink($path) 来删除HTML的物理文件即可。