背景
php在爬取网页信息的时候,有一些函数可以使用。
这里介绍两个
- DOMDocument
- DOMXPath
代码解析
<?php
// 示例HTML
$html = '<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Example</title>
</head>
<body>
<div>
<p>Hello World!</p>
<p xx="test_custom_key" k="test_multi_key">this is a test 中文</p>
<a href="#xxx_link">Link</a>
<p id="xxx_p">Link</p>
<a href="#xxx_link------2">Link</a>
<a href="#xxx_link_%E4%B8%AD%E6%96%87%E9%93%BE%E6%8E%A5">Link</a>
</div>
<div id="custom_div_id">
<p id="xxx">Another paragraph</p>
<a href="#">Another link</a>
<div>second empty div</div>
<div id="custom_div_id">
<p>second div paragraph</p>
</div>
</div>
</body>
</html>';
function printDomNode($paragraph){
echo "----\nNode: ".$paragraph->nodeValue . "\n";
echo "all attr: \n";
for ($i = 0; $i < $paragraph->attributes->length; $i++) {
$attr = $paragraph->attributes->item($i);
echo "\t".$attr->nodeName . ': ' . $attr->nodeValue . "\n";
}
}
// 创建DOMDocument实例并加载HTML
$dom = new DOMDocument();
@$dom->loadHTML($html);
// 创建DOMXPath实例
$xpath = new DOMXPath($dom);
echo "----------------------------\n";
// 示例1:查找所有<p>元素
$paragraphs = $xpath->query('//p');
foreach ($paragraphs as $paragraph) {
echo "----\nNode----------: ".$paragraph->nodeValue . "\n";
echo "id attr: ".$paragraph->getAttribute('id') . "\n";
echo "all attr: \n";
for ($i = 0; $i < $paragraph->attributes->length; $i++) {
$attr = $paragraph->attributes->item($i);
echo "\t".$attr->nodeName . ': ' . $attr->nodeValue . "\n";
}
echo "=foreach=\n";
foreach ($paragraph->attributes as $attr) {
echo "\t".$attr->name . ': ' . $attr->value . "\n";
echo "\t".$attr->nodeName . ': ' . $attr->nodeValue . "\n";
}
}
echo "----------------------------\n";
$paragraphs = $xpath->query('//p[@xx="test_custom_key"]');
foreach ($paragraphs as $paragraph) {
printDomNode($paragraph);
}
echo "----------------------------\n";
// 示例2:查找包含特定文本的<a>元素
$links = $xpath->query('//a[text()="Link"]');
//$links = $xpath->query('//*[text()="Link"]'); //不限制a标签,会找到所有值是Link的节点
foreach ($links as $link) {
$herf = $link->getAttribute('href');
echo "origin: ".$herf . "\n";
echo "decode: ".urldecode($herf) . "\n";
}
//如果找到指定路径下面的节点
echo "----------------------------指定路径下的p节点\n";
$paragraphs = $xpath->query('//div[@id="custom_div_id"]//p[@id="xxx"]');
foreach ($paragraphs as $paragraph) {
printDomNode($paragraph);
}
echo "----------------------------\n";
// 示例3:查找<div>元素内的所有节点(这个会找到所有子节点,包括节点里面的节点)
$divChildren = $xpath->query('//div/*');
foreach ($divChildren as $child) {
// echo $child->nodeName . ": " . $child->nodeValue . "\n";
printDomNode($paragraph);
}
?>
示例:比如获取一个html文档中的p标签
- 步骤
- 获取网页html
- 这里省略了请求url。如果需要从url获取html:
$html = file_get_contents($url);
- 这里省略了请求url。如果需要从url获取html:
- 将html文件构建成DOM树结构
- 使用DOMXPath类来查找指定的元素节点
- 构建DOMXPath类实例
- 使用query函数查询
- 如果找所有p标签,那么会遍历整个树,把所有的p标签找出来
- 如何找指定属性的p标签:
p[@xx="test_custom_key"]
- 这里的xx是自定义的属性
- 这里的@表示选择节点的属性
- 如何找到指定路径下的节点
//div[@id="custom_div_id"]//p[@id="xxx"]
- 获取网页html