最近在查wpf绘图资料时,偶然看到Python使用CSS选择器抓取网页的功能。觉得很强,这里用C#也实现一下。
先介绍一下CSS选择器
在 CSS 中,选择器是一种模式,用于选择需要添加样式的元素。
选择器 |
例子 |
例子描述 |
.intro |
选择 class="intro" 的所有元素。 |
|
#firstname |
选择 id="firstname" 的所有元素。 |
|
* |
选择所有元素。 |
|
p |
选择所有 <p> 元素。 |
|
div,p |
选择所有 <div> 元素和所有 <p> 元素。 |
|
div p |
选择 <div> 元素内部的所有 <p> 元素。 |
|
div>p |
选择父元素为 <div> 元素的所有 <p> 元素。 |
|
div+p |
选择紧接在 <div> 元素之后的所有 <p> 元素。 |
|
[target] |
选择带有 target 属性所有元素。 |
|
[target=_blank] |
选择 target="_blank" 的所有元素。 |
|
[title~=flower] |
选择 title 属性包含单词 "flower" 的所有元素。 |
|
[lang|=en] |
选择 lang 属性值以 "en" 开头的所有元素。 |
|
a:link |
选择所有未被访问的链接。 |
|
a:visited |
选择所有已被访问的链接。 |
|
a:active |
选择活动链接。 |
|
a:hover |
选择鼠标指针位于其上的链接。 |
|
input:focus |
选择获得焦点的 input 元素。 |
|
p:first-letter |
选择每个 <p> 元素的首字母。 |
|
p:first-line |
选择每个 <p> 元素的首行。 |
|
p:first-child |
选择属于父元素的第一个子元素的每个 <p> 元素。 |
|
p:before |
在每个 <p> 元素的内容之前插入内容。 |
|
p:after |
在每个 <p> 元素的内容之后插入内容。 |
|
p:lang(it) |
选择带有以 "it" 开头的 lang 属性值的每个 <p> 元素。 |
|
p~ul |
选择前面有 <p> 元素的每个 <ul> 元素。 |
|
a[src^="https"] |
选择其 src 属性值以 "https" 开头的每个 <a> 元素。 |
|
a[src$=".pdf"] |
选择其 src 属性以 ".pdf" 结尾的所有 <a> 元素。 |
|
a[src*="abc"] |
选择其 src 属性中包含 "abc" 子串的每个 <a> 元素。 |
|
p:first-of-type |
选择属于其父元素的首个 <p> 元素的每个 <p> 元素。 |
|
p:last-of-type |
选择属于其父元素的最后 <p> 元素的每个 <p> 元素。 |
|
p:only-of-type |
选择属于其父元素唯一的 <p> 元素的每个 <p> 元素。 |
|
p:only-child |
选择属于其父元素的唯一子元素的每个 <p> 元素。 |
|
p:nth-child(2) |
选择属于其父元素的第二个子元素的每个 <p> 元素。 |
|
p:nth-last-child(2) |
同上,从最后一个子元素开始计数。 |
|
p:nth-of-type(2) |
选择属于其父元素第二个 <p> 元素的每个 <p> 元素。 |
|
p:nth-last-of-type(2) |
同上,但是从最后一个子元素开始计数。 |
|
p:last-child |
选择属于其父元素最后一个子元素每个 <p> 元素。 |
|
:root |
选择文档的根元素。 |
|
p:empty |
选择没有子元素的每个 <p> 元素(包括文本节点)。 |
|
#news:target |
选择当前活动的 #news 元素。 |
|
input:enabled |
选择每个启用的 <input> 元素。 |
|
input:disabled |
选择每个禁用的 <input> 元素 |
|
input:checked |
选择每个被选中的 <input> 元素。 |
|
:not(p) |
选择非 <p> 元素的每个元素。 |
|
::selection |
选择被用户选取的元素部分。 |
C#自带的类库里不支持这个操作,所以需要用到三方库。这里用的是AngleSharp,使用Nuget搜索这个包就可以
这里以抓取https://technet-info.com/Main.aspx为例
页面源码如下:
<html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="Content-Type" content="text/html; charset=utf-8" /><meta name="description" content="Wandering the number of windows, stayed in the number of hotels, will feel that separation is not wronged, the feelings are used to browse or used to collect, so that the day had a memorable day" /><title>
Welcome To Technet-Info : Personal Gallery
</title><link rel="shortcut icon" type="image/x-icon" href="technet.ico" media="screen" /><link rel="stylesheet" href="Css/MainCss.css" /><link rel="stylesheet" href="Css/screen.css" />
<style>
#footer{
display: flex;
justify-content: center;
align-items: center;
position: fixed;
bottom: 0;
left: 0;
width: 100%;
}
</style>
<script type="text/javascript" src="js/jquery.js"></script>
<script type="text/javascript" src="js/easySlider1.7.js"></script>
<script type="text/javascript">
$(document).ready(function () {
$("#slider").easySlider({
auto: true,
pause:3000,
continuous: true,
numeric: true
});
});
</script>
</head>
<body>
<form method="post" action="./Main.aspx" id="form1">
<div class="aspNetHidden">
<input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwUKLTQyNjI2MTkwNmRkt331eyucv2SBluj0E2d+0haGV4exFHWtGQkZhNBnpHE=" />
</div> <div class="aspNetHidden"> <input type="hidden" name="__VIEWSTATEGENERATOR" id="__VIEWSTATEGENERATOR" value="202EA31B" />
</div>
<div id="main">
<div id="header">
<div class="musicarea"> <iframe frameborder="no" border="0" marginwidth="0" marginheight="0" width=150 height=52 src="http://music.163.com/outchain/player?type=0&id=516657278&auto=1&height=32"></iframe>
</div>
<div class="content"> <div class="logo"> <div class="logo_img">
<div class="logo_img"></div>
</div> <div class="logo_txt">
<div style="height: 50px;">
<p></p>
</div>
<div style="height: 50px;">
<p>我的freetime</p>
</div>
</div>
</div> <div class="menu"> </div>
</div> <div id="content"> </div> <div id="cards"> </div>
<div id="pin"> </div> </div> <div id="footer">
<div id="copyright">
<p style="margin: 3px">
<a href="http://www.miitbeian.gov.cn/">湘ICP备17816343号</a>
<span>|</span>
<span>Copyright © 2016, www.technet-info.com, All rights reserved.</span>
</p>
<p><a href="mailto:zhaotianff@163.com">Email:zhaotianff@163.com</a></p>
</div>
</div>
</div>
</form>
</body>
</html>
新建一个控制台工程,引用AngleSharp(由于在Main函数中使用了Async,所以需要Visual Studio 2017+,如果低于这个版本,可以把Main函数中的内容封装成一个函数执行,然后移除Main中的Async)
初始化
var config = Configuration.Default;
var context = BrowsingContext.New(config);
var source = Properties.Resources.HTML;
var document = await context.OpenAsync(req => req.Content(source));
建立CSS选择器并执行查询 ,查找class = content的节点
var cssSelector = ".content";
var cell = document.QuerySelector(cssSelector);
输出InnerHtml可以看到如下结果
基本上到这里就可以愉快的使用CSS选择器进行元素选取了。如果需要上面表格中全部CSS选择器的示例,可以下载示例代码
这里还有一个实用功能,就是浏览器开发者工具提供了复制CSS选择器的功能
示例代码待上传。。。。
最后:
如果需要进行XPath查询,可以参考
https://www.cnblogs.com/zhaotianff/p/11319871.html
这篇文章。虽然示例是基于XML的,但HTML基本上也一样。同样也可以使用上面的浏览器工具复制元素的XPath。
有兴趣了解C#爬虫相关知识的小伙伴,可以参考
https://github.com/zhaotianff/CSharpCrawler
再附上一个详细介绍CSS选择器的链接
https://developer.mozilla.org/zh-CN/docs/Web/CSS/Attribute_selectors
≧◔◡◔≦