数据采集实践学习二(C) 数据采集实验
yuyutoo 2024-10-29 17:29 3 浏览 0 评论
前一篇文章写到我获取数据的方式不是通过分析HTML获得,而是通过分析请求链接,然后模拟请求方法获取数据,这只是一种方法。而且是在我通过分析HTML获取不到的情况下,曲线救国,参考别人文章实现的。很高兴,我实现了自己获取数据的目标。我以为这样就算结束了。可是,今天又发现了另外一种方法,而且是通过分析HTML实现的,看到它,我感觉太不可思议了,我花了那么多的时间都没有实现,怎么现在又可以了。现在兴趣正浓,赶紧操刀实践一番。于是有了这篇,算是意外之喜吧!
先说明一下实现思路,原来它是通过调用WebBrowser控件来实现的。怪不得它可以获取HTML,然后分析获取数据。管你什么动态解析,ajax,现在我是浏览器行为了,所有的都逃不过我的法眼。真的是不错的选择方式。
说明一下,包含三个地方。
一个解析获取解析HTML类,一个事件类,一个调用的地方。上次我是拿那个情趣网站实验,结果大家都说我好污,好污,其实我是一个好人,一个让大家都有动力兴趣的好人,代码写累了,看看图片,又鸡血了,我不信大家对美图不感兴趣。学习与欢乐同行,自娱自乐。好了,这次避免大家的想法,我拿我们的博客园实验,我只是获取前面三个页面,太多了也是一样的效果,没有必要,说明方法可行就可以了。
开始代码吧 一个解析类
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading; using System.Threading.Tasks; using System.Windows.Forms; namespace WebBrowserCrawlerdemo { //这种感觉只适合单个页面数据抓取//可以抓取多个页面如博客园的数据 //http://www.cnblogs.com/rookey/p/5019090.html /// <summary> /// 通过WebBrowser抓取网页数据 /// WebBrowserCrawler webBrowserCrawler=new WebBrowserCrawler; /// 示例:File.WriteAllText(Server.MapPath("sample.txt"),webBrowserCrawler.GetReult(http://www.in2.cc/sample/waterfalllab.htm)); /// </summary> public class WebBrowserCrawler { // WebBrowser private WebBrowser _WebBrowder; //最後結果 private string _Result { get; set; } //網址 private string _Path { get; set; } //当一直在抓取资料,允许等待的的最大秒数,超时时间(秒) private int _MaxWaitSeconds { get; set; } public delegate bool MyDelegate(object sender, TestEventArgs e); /// <summary> /// 是否达到停止加载条件 /// </summary> public event MyDelegate IsStopEvent; /// <summary> /// 對外公開的Method /// </summary> /// <param name="url">URL Path</param> /// <param name="maxWaitSeconds">最大等待秒数</param> /// <returns></returns> public string GetReult(string url, int maxWaitSeconds = 60) { _Path = url; _MaxWaitSeconds = maxWaitSeconds <= 0 ? 60 : maxWaitSeconds; var mThread = new Thread(FatchDataToResult); //Apartment 是處理序當中讓物件共享相同執行緒存取需求的邏輯容器。 同一 Apartment 內的所有物件都能收到 Apartment 內任何執行緒所發出的 //.NET Framework 並不使用 Apartment;Managed 物件必須自行以安全執行緒 (Thread-Safe) 的方式運用一切共 //因為 COM 類別使用 Apartment,所以 Common Language Runtime 在 COM Interop 的狀況下呼叫出 COM 物件時必須建立 Apartment 並且加以初 //Managed 執行緒可以建立並且輸入只容許一個執行緒的單一執行緒 Apartment (STA),或者含有一個以上執行緒的多執行緒 Apartment (MT //只要把執行緒的 ApartmentState 屬性設定為其中一個 ApartmentState 列舉型別 (Enumeration),即可控制所建立的 Apartment 屬於哪種 //因為特定執行緒一次只能初始化一個 COM Apartment,所以第一次呼叫 Unmanaged 程式碼之後就無法再變更 Apartment //From : http://msdn.microsoft.com/zh-tw/library/system.threading.apartmentstate. mThread.SetApartmentState(ApartmentState.STA); mThread.Start; mThread.Join; return _Result; } /// <summary> /// Call _WebBrowder 抓取資料 /// For thread Call /// </summary> private void FatchDataToResult { _WebBrowder = new WebBrowser; _WebBrowder.ScriptErrorsSuppressed = true; _WebBrowder.Navigate(_Path); DateTime firstTime = DateTime.Now; //處理目前在訊息佇列中的所有 Windows //如果在程式碼中呼叫 DoEvents,您的應用程式就可以處理其他事件。例如,如果您的表單將資料加入 ListBox 並將 DoEvents 加入程式碼中,則當另一個視窗拖到您的表單上時,該表單將重 //如果您從程式碼移除 DoEvents,您的表單將不會重新繪製,直到按鈕按一下的事件處理常式執 //通过不断循环把整个页面都加载完,然后从中获取自己想要的信息。可以结合这个JumonyParser一起用 while ((DateTime.Now - firstTime).TotalSeconds <= _MaxWaitSeconds) { if (_WebBrowder.Document != null && _WebBrowder.Document.Body != null && !string.IsNullOrEmpty(_WebBrowder.Document.Body.OuterHtml) && this.IsStopEvent != null) { string html = _WebBrowder.Document.Body.OuterHtml; bool rs = this.IsStopEvent(null, new TestEventArgs(html)); if (rs) { this._Result = html; break; } } Application.DoEvents; } _WebBrowder.Dispose; } } }
事件类
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace WebBrowserCrawlerdemo { public class TestEventArgs:EventArgs { public string Html { get; set; } public TestEventArgs(string html2) { this.Html = html2; } } }
调用端 先来一个界面吧
代码
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.IO; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; namespace WebBrowserCrawlerdemo { public partial class Form1 : Form { public Form1 { InitializeComponent; } public void test(int num) { WebBrowserCrawler obj = new WebBrowserCrawler; obj.IsStopEvent += new WebBrowserCrawler.MyDelegate((sender, e) => { //当前html中已经加载了我想要的数据,返回true// //return e.Html.Contains("<div id=\"post_list\">"); return e.Html.Contains("<div class=\"post_item\">"); }); string url = string.Format("http://www.cnblogs.com/#p{0}", num); string html = obj.GetReult(url); //获取采集的数据 if (!string.IsNullOrEmpty(html)) { //处理数据 Write(html); } } private void btntest_Click(object sender, EventArgs e) { for (int i = 1; i < 4; i++) { test(i); } } //http://www.cnblogs.com/akwwl/p/3240813.html public void Write( string html) { string path = @"D:\练习\MyPictureDownloader\WebBrowserCrawlerdemo\bin\Debug\test\test.txt"; FileStream fs = new FileStream(path, FileMode.Append); //获得字节数组 byte data = System.Text.Encoding.Default.GetBytes(html); //开始写入 fs.Write(data, 0, data.Length); //清空缓冲区、关闭流 fs.Flush; fs.Close; } } }
说明一下,我数据是保存到TXT文件里,没有去分析什么目标数据了,只要整个页面获取就可以了,我是通过追加的形式保存的。
e.Html.Contains("<div id=\"post_list\">"); 分析为啥不是这个,我用它结果获取不到数据。原来是这样的。
返回的是html元素格式,通过它,请求都还没有结束,没有获取到数据,肯定不行了。于是改成上面那个了。可以获取数据。
结果如图,我只有获取三页因此三个<body>标签,我也检验对比了,事实就是三页的数据。
如果你还想获取目标数据,可以借助一些HTML分析类如: Jumony,HtmlAgilityPack。
好了,已经下班了。内容也介绍完了。
参考:
http://www.cnblogs.com/rookey/p/5019090.html
http://www.cnblogs.com/akwwl/p/3240813.html
相关推荐
- 如何在HTML中使用JavaScript:从基础到高级的全面指南!
-
“这里是云端源想IT,帮你...
- 推荐9个Github上热门的CSS开源框架
-
大家好,我是Echa。...
- 硬核!知网首篇被引过万的论文讲了啥?作者什么来头?
-
整理|袁小华近日,知网首篇被引量破万的中文论文及其作者备受关注。知网中心网站数据显示,截至2021年7月23日,由华南师范大学教授温忠麟等人发表在《心理学报》2004年05期上的学术论文“中介效应检验...
- 为什么我推荐使用JSX开发Vue3_为什么用vue不用jquery
-
在很长的一段时间中,Vue官方都以简单上手作为其推广的重点。这确实给Vue带来了非常大的用户量,尤其是最追求需求开发效率,往往不那么在意工程代码质量的国内中小企业中,Vue占据的份额极速增长...
-
- 【干货】一文详解html和css,前端开发需要哪些技术?
-
网站开发简介...
-
2025-02-20 18:34 yuyutoo
- 分享几个css实用技巧_cssli
-
本篇将介绍几个css小技巧,目录如下:自定义引用标签的符号重置所有标签样式...
- 如何在浏览器中运行 .NET_怎么用浏览器运行代码
-
概述:...
- 前端-干货分享:更牛逼的CSS管理方法-层(CSS Layers)
-
使用CSS最困难的部分之一是处理CSS的权重值,它可以决定到底哪条规则会最终被应用,尤其是如果你想在Bootstrap这样的框架中覆盖其已有样式,更加显得麻烦。不过随着CSS层的引入,这一...
-
- HTML 基础标签库_html标签基本结构
-
HTML标题HTML标题(Heading)是通过-...
-
2025-02-20 18:34 yuyutoo
- 前端css面试20道常见考题_高级前端css面试题
-
1.请解释一下CSS3的flexbox(弹性盒布局模型),以及适用场景?display:flex;在父元素设置,子元素受弹性盒影响,默认排成一行,如果超出一行,按比例压缩flex:1;子元素设置...
- vue引入外部js文件并使用_vue3 引入外部js
-
要在Vue中引入外部的JavaScript文件,可以使用以下几种方法:1.使用``标签引入外部的JavaScript文件。在Vue的HTML模板中,可以直接使用``标签来引入外部的JavaScrip...
- 网页设计得懂css的规范_html+css网页设计
-
在初级的前端工作人员,刚入职的时候,可能在学习前端技术,写代码不是否那么的规范,而在工作中,命名的规范的尤为重要,它直接与你的代码质量挂钩。网上也受很多,但比较杂乱,在加上每年的命名都会发生一变化。...
- Google在Chrome中引入HTML 5.1标记
-
虽然负责制定Web标准的WorldWideWebConsortium(W3C)尚未宣布HTML5正式推荐规格,而Google已经迁移到了HTML5.1。即将发布的Chrome38将引入H...
- HTML DOM 引用( ) 对象_html中如何引用js
-
引用对象引用对象定义了一个同内联元素的HTML引用。标签定义短的引用。元素经常在引用的内容周围添加引号。HTML文档中的每一个标签,都会创建一个引用对象。...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)