欢迎光临
我们一直在努力

如何用PHP把RDF内容插入Web站点之中(四)-PHP教程,PHP应用

建站超值云服务器,限时71元/月

返回到类(back to class)

既然你有这么大的权力,那么究竟为什么要把自己限制在仅仅是单个的rdf来源呢?就象我早先说过的一样,大多数主要的站点都经常为他们所提供的内容做快照。其实将所有这些不同的来源插入到你的站点当中是相当简单的。让我们看看是如何做的。

首先,我们把前面例子中的代码模块化。这样一来,你就无须为每一个单个的来源都一遍又一遍的重写相同的代码了。简化的方法就是将之打包成类,再把这个类包含到我的php脚本当中。

类代码如下:

<?

class rdfparser

{

//

// variables

//

// set up local variables for this class

var $currenttag = "";

var $flag = "";

var $count = 0;

// this is an associative array of channel data with keys

("title", "link", "description")

var $channel = array();

// this is an array of arrays, with each array element

representing an <item>

// each outer array element is itself an associative array

// with keys ("title", "link", "description")

var $items = array();

//

// methods

//

// set the name of the rdf file to parse

// this is usually a local file

// you may set it to a remote file if your php build supports

url fopen()

function setresource($file)

{

$this->file = $file;

}

// parse the rdf file set with setresource()

// this populates the $channel and $items arrays

function parseresource()

{

// create parser

$this->xp = xml_parser_create();

// set object reference

xml_set_object($this->xp, $this);

// set handlers and parser options

xml_set_element_handler($this->xp, "elementbegin",

"elementend");

xml_set_character_data_handler($this->xp,

"characterdata");

xml_parser_set_option($this->xp,

xml_option_case_folding, true);

xml_parser_set_option($this->xp, xml_option_skip_white,

true);

// read xml file

if (!($fp = fopen($this->file, "r")))

{

die("could not read $this->file");

}

// parse data

while ($xml = fread($fp, 4096))

{

if (!xml_parse($this->xp, $xml, feof($fp)))

{

die("xml parser error: " .

xml_error_string(xml_get_error_code($this->xp)));

}

}

// destroy parser

xml_parser_free($this->xp);

}

// opening tag handler

function elementbegin($parser, $name, $attributes)

{

$this->currenttag = $name;

// set flag if entering <channel> or <item> block

if ($name == "item")

{

$this->flag = 1;

}

else if ($name == "channel")

{

$this->flag = 2;

}

}

// closing tag handler

function elementend($parser, $name)

{

$this->currenttag = "";

// set flag if exiting <channel> or <item> block

if ($name == "item")

{

$this->count++;

$this->flag = 0;

}

else if ($name == "channel")

{

$this->flag = 0;

}

}

// character data handler

function characterdata($parser, $data)

{

$data = trim(htmlspecialchars($data));

if ($this->currenttag == "title" || $this->currenttag ==

"link" || $this->currenttag == "description")

{

// add data to $channels[] or $items[] array

if ($this->flag == 1)

{

$this->items[$this->count][strtolower($this->currenttag)] .= $data;

}

else if ($this->flag == 2)

{

$this->channel[strtolower($this->currenttag)] .= $data;

}

}

}

// return an associative array containing channel information

// (the $channel[] array)

function getchannelinfo()

{

return $this->channel;

}

// return an associative array of arrays containing item

information

// (the $items[] array)

function getitems()

{

return $this->items;

}

}

?>

如果你对php类较为熟悉的话,那么理解这段代码是相当容易的。如果不太懂的话,那么请直接跳到文章末尾的链接部分,看一篇关于类工作原理的好文章。然后在回来继续阅读上面的代码。

在使用这个类之前,我要特别花几分钟指出其中的一行代码——即上面对xml_set_object()函数调用的那一行。

现在的问题是如何使用这个类实际生成具有多个内容来源的web页。

<?

include("class.rdfparser.php");

// how many items to display in each channel

$maxitems = 5;

?>

<html>

<head>

<basefont face="verdana">

<body>

<table width="100%" border="0" cellspacing="5" cellpadding="5"> <tr>

<!– first cell –>

<td valign=top align=left>

<font size="-1">

<?

// get and parse freshmeat.net channel

$f = new rdfparser();

$f->setresource("http://www.freshmeat.net/backend/fm-releases.rdf");

$f->parseresource();

$f_channel = $f->getchannelinfo();

$f_items = $f->getitems();

// now format and print it…

?>

the latest from <a href=<? echo $f_channel["link"]; ?>><? echo

$f_channel["title"]; ?></a> <br> <ul> <? // iterate through items array

for ($x=0; $x<$maxitems; $x++) {

if (is_array($f_items[$x]))

{

// print data

$item = $f_items[$x];

echo "<li><a href=" . $item["link"] . ">" .

$item["title"] . "</a>";

}

}

?>

</ul>

</font>

</td>

<!– second cell –>

<td align=center width=50%>

<i>primary page content here</i>

</td>

<!– third cell –>

<td valign=top align=left>

<font size="-1">

<?

// get and parse slashdot.org channel

$s = new rdfparser();

$s->setresource("http://slashdot.org/slashdot.rdf");

$s->parseresource();

$s_channel = $s->getchannelinfo();

$s_items = $s->getitems();

// now format and print it…

?>

the latest from <a href=<? echo $s_channel["link"]; ?>><? echo

$s_channel["title"]; ?></a> <br> <ul> <? // iterate through items array

for ($x=0; $x<$maxitems; $x++) {

if (is_array($s_items[$x]))

{

// print data

$item = $s_items[$x];

echo "<li><a href=" . $item["link"] . ">" .

$item["title"] . "</a>";

}

}

?>

</ul>

</font>

</td>

</tr>

</table>

</body>

</head>

</html>

这段代码相当简单。一旦你用“new”关键字生成一个类的实例,

$f = new rdfparser();

那么就可以用类方法来设置要分析的rdf文件的位置,

$f->setresource("http://www.freshmeat.net/backend/fm-releases.rdf");

$f->parseresource();

并且获取$channel和$items数组,以用于后面的处理。

<?

$f_channel = $f->getchannelinfo();

$f_items = $f->getitems();

?>

the latest from <a href=<? echo $f_channel["link"]; ?>><? echo

$f_channel["title"]; ?></a> <br> <ul> <? // iterate through items array

for ($x=0; $x<$maxitems; $x++) {

if (is_array($f_items[$x]))

{

// print data

$item = $f_items[$x];

echo "<li><a href=" . $item["link"] . ">" .

$item["title"] . "</a>";

}

}

?>

</ul>

每次你重新装入上面的脚本,相应的rdf文件就会被从特定的位置上取来,经过分析之后,按要求的格式显示出来。

如果你站点具有高的访问量,你就可能觉得我们的辛苦无意义之极,尤其是当所用的rdf数据更新地没有那么快时,情况更糟。 在这种情况下,或许探究一下在本地缓存rdf数据才是较明智的做法:要么扩展上面的例子程序,在其中加入缓存功能;要么每阁几个小时都花很长的时间下载一个最新rdf文件的本地副本到你的web服务器上,然后使用这个本地副本,而不是那个“活”的(the “live” one)。

赞(0)
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com 特别注意:本站所有转载文章言论不代表本站观点! 本站所提供的图片等素材,版权归原作者所有,如需使用,请与原作者联系。未经允许不得转载:IDC资讯中心 » 如何用PHP把RDF内容插入Web站点之中(四)-PHP教程,PHP应用
分享到: 更多 (0)