有人想把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

留言反馈
@hotmail.com , qq 63966617
请教哪位高人指点一下
1、用“this.axWebBrowser1.Navigate2(ref url,ref nothing,ref nothing,ref nothing,ref nothing);”下载,会下载图片等。对我们没有用的页面元素(下载漫)。
2、IHTMLDocument分析非常漫。
不如用HTTP协议下载。然后在下载的字符串中查找"<A>"元素来提取链接。
request1.CookieContainer.SetCookies(request1.RequestUri,strCookieHeader.Replace(';',','));
我能够理解您的立场。
我已经通过查资料慢慢的接近解决方法。
不过,还是要感谢您的帮助以及这篇帖子的帮助。
Thanks
sorry,请理解我无法很深入的帮助您,我只能在我的空闲时间和精力许可范围内尽我所能。同时,我相信通过查阅公开的文档以及不断的调试代码,绝大部分的问题都是可以解决的。如果您解决了以后,不妨把问题与解决方案一起贴出来,这样后来者就更节省时间了。
但是我从它直接转换成WebBrowser却不行。
还有两个小问题就是:
1. 找不到服务器之类的错误怎么判断?
2. 当html中有iframe的时候,DocumentComplete会被触发多次,那么怎么知道所有的内容(包含IFrame)都好了。
由于我初次接触这个方面的编程,所以碰到的问题比较多,请见谅。
Thanks
用IHTMLElement取到的是包含了IFrame的Document
你好,我在做一个项目的时候碰到了一个问题:IFrame里的文档怎么分析?
Thanks!
我的答案是yes。有方法的。
比如,redirect, http return code肯定就是300(或者302等等)。这样,你的代码自己做跳转。其实,netant什么的工具也都是这么做的。
session,其实就是cookie。cookieless session就是querysting。viewstate就是<input type=hidden>。等等,所有这些都可以最终反映成一些基本的HTML技术。所以说,总是有办法的。比如说,如果需要POST Data来登陆,那么就需要手工做一个body出来,然后post。
总之,都是可以的
比如,有一些是经过多次redirect的,是不是要通过分析html头的return code呢?
又如,如果一些需要session的(比如你得首先登陆才能进入的页面),我们怎么模拟web browser的session处理方式呢?
function abc( in1, in2 )
{
document.write(in1 + in2);
}
<scriprt>abc('你好', 'MVM');</script>
这种用html paser根本不能取. 我是用httpwebrequest来取网页的
一般应用只需要分析HTML,不需要DOM的话Tidy是个不错的选择。
如果我要取这段html里面的中文,可能网页还有其他很多代码:
....
<title>屋顶上的木帷幕</title>
....
用字符串匹配起始和结束字符快,还是用正则表达式快?