屋顶上的木帷幕

海鸥之所以追着渔船飞,是因为它们认为会有沙丁鱼抛向大海 - Eric Cantona, 1995
随笔 - 146, 评论 - 3169, 引用 - 56

导航

关于


标签

每月存档

最新留言

广告

 

有人想把Web Page拉下来并抽取其中的内容。这其实是搜索引擎的一项最最基本的工作:下载,抽取,再下载。我早年做过一个Search Engine项目,不过代码都已经不见了。这次有人又问到我这个事情,我给攒了两个方法。

方法A,在一个WinForm里面用一个隐藏的Browser控件下载Web Page,并用IHTMLDocument来分析内容。这个方法比较简单,但如果对于大量文件的分析速度很慢。

这个方法中用到的主要代码如下:
private void button1_Click(object sender, System.EventArgs e) {
 object url="
http://www.google.com";
 object nothing=null;
 this.axWebBrowser1.Navigate2(ref url,ref nothing,ref nothing,ref nothing,ref nothing);
 this.axWebBrowser1.DownloadComplete+=new System.EventHandler(this.button2_Click);
}

private void button2_Click(object sender, System.EventArgs e) {
 this.textBox1.Text="";
 mshtml.IHTMLDocument2 doc=(mshtml.IHTMLDocument2)this.axWebBrowser1.Document;
 mshtml.IHTMLElementCollection all=doc.all;
 System.Collections.IEnumerator enumerator=all.GetEnumerator();
 while(enumerator.MoveNext() && enumerator.Current!=null)
 {
  mshtml.IHTMLElement element=(mshtml.IHTMLElement)(enumerator.Current);
  if(this.checkBox1.Checked==true)
  {
   this.textBox1.Text+="\r\n\r\n"+element.innerHTML;
  }
  else
  {
   this.textBox1.Text+="\r\n\r\n"+element.outerHTML;
  }
 }
}

方法B,用System.Net.WebClient下载Web Page存到本地文件或者String中用正则表达式来分析。这个方法可以用在Web Crawler等需要分析很多Web Page的应用中。

下面是一个例子,能够把http://www.google.com首页里的所有的Hyperlink都抽取出来:

using System;
using System.Net;
using System.Text;
using System.Text.RegularExpressions;

namespace HttpGet{
 class Class1{
  [STAThread]
  static void Main(string[] args){
   System.Net.WebClient client=new WebClient();
   byte[] page=client.DownloadData("
http://www.google.com");
   string content=System.Text.Encoding.UTF8.GetString(page);
   string regex="href=[\\\"
\\\'](http:\\/\\/|\\.\\/|\\/)?\\w+(\\.\\w+)*(\\/\\w+(\\.\\w+)?)*(\\/|\\?\\w*=\\w*(&\\w*=\\w*)*)?[\\\"\\\']";
   Regex re=new Regex(regex);
   MatchCollection matches=re.Matches(content);
   
   System.Collections.IEnumerator enu=matches.GetEnumerator();
   while(enu.MoveNext() && enu.Current!=null)
   {
    Match match=(Match)(enu.Current);
    Console.Write(match.Value+"\r\n");
   }
  }
 }
}

真正做爬虫的,都是用正则表达式来做抽取的,可以找些开源的爬虫,代码都差不多。只是有些更高,可以把Flash或者JavaScript里面的URL都抽取出来。

再补充一个,有人问我如果一个element是用document.write画出来的,还能不能在DOM里面取到。答案是肯定的。具体取的方法也和平常的一样。下面这个HTML就演示了从DOM里面取到用document.write动态生成的HTML Tag:

<FORM>
<SCRIPT>
   document.write("<input type=button id='btn1' value='button 1'>");
</SCRIPT>
<INPUT onclick=show() type=button value="click me">
</FORM>
<TEXTAREA id=allnode rows=29 cols=53></TEXTAREA>
<SCRIPT>
function show()
{
   document.all.item("allnode").innerText="";
   var i=0;
   for(i=0;i<document.forms[0].childNodes.length;i++)
   {
      document.all.item("allnode").innerText=document.all.item("allnode").innerText+"\r\n"+document.forms[0].childNodes[i].tagName+" "+document.forms[0].childNodes[i].value;
   }
}
</SCRIPT>

点了“Click Me”以后,打印出来的forms[0]的子元素列表里面,“button 1”赫然在列。

打印 | 张贴于 2004-04-27 15:54:00 | Tag:Dot NET

留言反馈

#ifqkutim - Google Search 编辑
ifqkutim - Google Search
2008-09-30 19:19:45 | [匿名用户:]
#ophcsqxl - Google Search 编辑
ophcsqxl - Google Search
2008-09-30 15:09:11 | [匿名用户:]
#回复: 两个分析HTML网页的方法 编辑
我也觉得正则的速度快,而且也操作相对也简单一些.
2007-06-23 10:36:00 | [匿名用户:relocation]
#回复: 两个分析HTML网页的方法 编辑
真正做爬虫的也得用正则来分析才快吧
2007-01-10 23:46:00 | [匿名用户:ohr]
#re: 两个分析HTML网页的方法 编辑
两个分析HTML网页的方法 ,看到这篇文章,认识你的,最近我也在看这方面的东西,希望以后能多交流,MSN coolpest
@hotmail.com , qq 63966617
2006-07-28 14:03:00 | [匿名用户:coolpest]
# 两个分析HTML网页的方法 编辑
希望能分享一下 NAVYSUN 的C# Spider代码 我的邮箱 majintao1982@gmail.com
2006-05-24 04:41:00 | [匿名用户:圣涛玄纯]
#re: 两个分析HTML网页的方法 编辑
我用WebRequest提交的URL收到的html不是真正的,少了些信息 不知道是不是cookie的问题
请教哪位高人指点一下
2006-04-14 23:04:00 | [匿名用户:aa]
#哪有开源的爬虫源码下载呢 编辑
哪有开源的爬虫源码下载呢
2006-03-19 10:56:00 | [匿名用户:weblead]
#re: 两个分析HTML网页的方法 编辑
那个方法(用 Web browser getelementsbytagname 得到table集合 对象后 遍历,再通过 cell row 等属性)是要用。net库2.0还是1.1阿?
2006-03-07 22:59:00 | [匿名用户:alexandercer@hotmail.com]
#re: 两个分析HTML网页的方法 编辑
用 Web browser getelementsbytagname 得到table集合 对象后 遍历,再通过 cell row 等属性
2005-11-15 15:14:00 | [匿名用户:ha]
#re: 两个分析HTML网页的方法 编辑
你好,我想请教再vb里解析html的某个table所包含的所有fields,应该怎么做?vb有现成的控件或者方法可以调用的吗?
2005-10-26 10:12:00 | [匿名用户:小秋]
#re: 两个分析HTML网页的方法 编辑
如果我想提取网页中的表格数据,用正则表达使该怎么写呢?
2005-05-11 13:20:00 | [匿名用户:哈哈]
#re: 两个分析HTML网页的方法 编辑
下载html前,怎么正确判断这个HTML的文件编码,有些HTML没有标出编码的情况下
2005-02-24 14:42:00 | [匿名用户:fangjing]
#re: 两个分析HTML网页的方法 编辑
这是一个非常笨的方法.
1、用“this.axWebBrowser1.Navigate2(ref url,ref nothing,ref nothing,ref nothing,ref nothing);”下载,会下载图片等。对我们没有用的页面元素(下载漫)。
2、IHTMLDocument分析非常漫。

不如用HTTP协议下载。然后在下载的字符串中查找"<A>"元素来提取链接。
2004-11-26 09:40:00 | [匿名用户:Zulin meng]
#to capthing 搞定了 编辑
request1.CookieContainer = new CookieContainer();
request1.CookieContainer.SetCookies(request1.RequestUri,strCookieHeader.Replace(';',','));
2004-11-19 15:08:00 | [匿名用户:playyuer]
#re: 两个分析HTML网页的方法 编辑
@mvm:
我能够理解您的立场。
我已经通过查资料慢慢的接近解决方法。

不过,还是要感谢您的帮助以及这篇帖子的帮助。

Thanks
2004-11-04 20:55:00 | [匿名用户:Jasper]
#re: 两个分析HTML网页的方法 编辑
@jasper
sorry,请理解我无法很深入的帮助您,我只能在我的空闲时间和精力许可范围内尽我所能。同时,我相信通过查阅公开的文档以及不断的调试代码,绝大部分的问题都是可以解决的。如果您解决了以后,不妨把问题与解决方案一起贴出来,这样后来者就更节省时间了。
2004-11-04 18:11:00 | [匿名用户:mvm]
#re: 两个分析HTML网页的方法 编辑
DocumentComplete(pDisp, uRL)事件中的pDisp怎么用,我看了msdn上的文档(pDisp:Object that specifies the top-level or frame WebBrowser object corresponding to the event. )

但是我从它直接转换成WebBrowser却不行。

还有两个小问题就是:
1. 找不到服务器之类的错误怎么判断?
2. 当html中有iframe的时候,DocumentComplete会被触发多次,那么怎么知道所有的内容(包含IFrame)都好了。

由于我初次接触这个方面的编程,所以碰到的问题比较多,请见谅。

Thanks
2004-11-04 17:24:00 | [匿名用户:Jasper]
#re: 两个分析HTML网页的方法 编辑
这份文档怎么取得?我用IHTMLIFrameElement取不到。

用IHTMLElement取到的是包含了IFrame的Document
2004-11-04 15:59:00 | [匿名用户:Jasper]
#re: 两个分析HTML网页的方法 编辑
不太清楚你的问题。我觉得IFrame里面就是新的一份HTML文档了
2004-11-04 14:28:00 | [匿名用户:mvm]
#re: 两个分析HTML网页的方法 编辑
MVM:
你好,我在做一个项目的时候碰到了一个问题:IFrame里的文档怎么分析?

Thanks!

2004-11-04 14:24:00 | [匿名用户:Jasper]
#re: 两个分析HTML网页的方法 编辑
用httpwebrequest 加上了cookie但是查看发送的包时并不包含,不知有什么办法
2004-08-19 14:06:00 | [匿名用户:capthing]
#re: 两个分析HTML网页的方法 编辑
如果一些需要session的(比如你得首先登陆才能进入的页面),我们怎么模拟web browser的session处理方式呢? 能不能给点具体的建议?
2004-08-19 14:04:00 | [匿名用户:capthing]
#re: 两个分析HTML网页的方法 编辑
方法B的正则表达式好像不能提取全部的 url;
2004-08-05 15:11:00 | [匿名用户:gallon]
#回复: 两个分析HTML网页的方法 编辑
如果一定要回答yes or no,
我的答案是yes。有方法的。

比如,redirect, http return code肯定就是300(或者302等等)。这样,你的代码自己做跳转。其实,netant什么的工具也都是这么做的。

session,其实就是cookie。cookieless session就是querysting。viewstate就是<input type=hidden>。等等,所有这些都可以最终反映成一些基本的HTML技术。所以说,总是有办法的。比如说,如果需要POST Data来登陆,那么就需要手工做一个body出来,然后post。

总之,都是可以的
2004-04-30 11:06:00 | [匿名用户:mvm]
#回复: 两个分析HTML网页的方法 编辑
我想知道,对于一些特殊的网页,是否有其他办法获取呢?
比如,有一些是经过多次redirect的,是不是要通过分析html头的return code呢?
又如,如果一些需要session的(比如你得首先登陆才能进入的页面),我们怎么模拟web browser的session处理方式呢?
2004-04-30 10:57:00 | [匿名用户:Bee]
#回复: 两个分析HTML网页的方法 编辑
隐藏的webbrowser, 个人觉得非常不好.因为要等所有图片和flash装载完毕才能用Dom来取非常慢吧?
2004-04-29 09:53:00 | [匿名用户:Roy]
#回复: 两个分析HTML网页的方法 编辑
我试过用Html Paser来分析html源代码但是效率不会很高.还有一些javascript的代码就不能分析, 比如:

function abc( in1, in2 )
{
document.write(in1 + in2);
}

<scriprt>abc('你好', 'MVM');</script>

这种用html paser根本不能取. 我是用httpwebrequest来取网页的
2004-04-29 09:51:00 | [匿名用户:Roy]
#回复: 两个分析HTML网页的方法 编辑
直接分析源代码哈~
2004-04-28 19:39:00 | [匿名用户:rivershan]
#回复: 两个分析HTML网页的方法 编辑
如果性能很重要的话,最好不要在Managed环境下遍历DOM,DOM的COM Interface太多,每次调用都需要Interop的Cost,性能出奇的慢。
一般应用只需要分析HTML,不需要DOM的话Tidy是个不错的选择。
2004-04-27 23:38:00 | [匿名用户:qqchen]
#回复: 两个分析HTML网页的方法 编辑
取Title的话,取长文件用隐藏的webbrowser最快,可以在浏览器获得Title之后马上Stop,不用下载完。短文件的话,就不太清楚了
2004-04-27 18:55:00 | [匿名用户:jiangsheng]
#回复: 两个分析HTML网页的方法 编辑
正则表达式速度怎么样?

如果我要取这段html里面的中文,可能网页还有其他很多代码:
....
<title>屋顶上的木帷幕</title>
....

用字符串匹配起始和结束字符快,还是用正则表达式快?
2004-04-27 16:50:00 | [匿名用户:Roy]
#回复: 两个分析HTML网页的方法 编辑
如果目标很明确而且是核心的处理,我觉得用正规表达式很吃亏。
2004-04-27 16:45:00 | [匿名用户:Lostinet]
对不起,目前本随笔不允许发表新评论.

Powered by: Joycode.MVC引擎 0.5.1.8