by lostfire
这两天准备做一些网站编程的工作,于是对HtmlParse小研究了一下,目的是快速入手,而不是深入研究,做了一下整理,和大家共同讨论一下。
一,数据组织分析:
HtmlParser主要靠Node、AbstractNode和<wbr></wbr>Tag来表达Html,因为Remark和Text相对简单<wbr></wbr>,此处就将其忽略了。
- Node是形成树结构表示HTML的基础,所有的数据表示都是接口<wbr></wbr>Node的实现,Node定义了与页面树结构所表达的页面Page<wbr></wbr>对象,定义了获取父、子、兄弟节点的方法,定义了节点到对应htm<wbr></wbr>l文本的方法,定义了该节点对应的起止位置,定义了过滤方法<wbr></wbr>,定义了Visitor访问机制。
- AbstractNode是Node的一种具体的类实现<wbr></wbr>,起到构成树形结构的作用,除了同具体Node相关的accetp<wbr></wbr>方法,toString,toHtml,toPlainTextS<wbr></wbr>tring方法以外,AbstractNode实现了大多基本的方<wbr></wbr>法,使得它的子类,不用理会具体的树操作。
- Tag是具体分析的主要内容。Tag分成composite的Ta<wbr></wbr>g和不能包含其他Tag的简单Tag两类,其中前者的基类是Compo<wbr></wbr>siteTag,其子类包含BodyTag,Div<wbr></wbr>,FrameSetTag,OptionTag,等27个子类<wbr></wbr>;而简单Tag有BaseHrefTag<wbr></wbr>、DoctypeTag,FrameTag,ImageTag<wbr></wbr>,InputTag,JspTag,MetaTag<wbr></wbr>,ProcessingInstructionTag这八类。
Node分成三类:
- RemarkNode:代表Html中的注释
- TagNode:标签节点,是种类最多的节点类型,上述Tag的具体节点类都是TagNode的实现。
- TextNode:文本节点
二,Visitor方式访问Html:
1,整体解析过程
- 用一个URL或页面String做一个Parser
- 用这个Parser做一个Visitor
- 使用Parser.visitAllNodeWith(Visitor)来遍历节点
- 获取Visitor遍历后得到的数据
2,Visit过程
- 做解析之前做的事情:visitor.beginParsing<wbr></wbr>();
- 每次取到一个节点Node,让该Node接受accept该Vis<wbr></wbr>itor
- 做解析后做的事情:visitor.finishedParsin<wbr></wbr>g();
3,获取节点的过程:逐步遍历Html,分析出Node<wbr></wbr>。此部分较为复杂,且对于我们应用来说无需很多了解,暂跳过。
4,节点访问
节点访问采用Visitor模式,Node的accept方法和具<wbr></wbr>体Visitor的visit方法是关键。
首先三类Node来accept的方式各不相同:
- 对于所有TagNode都使用一个accept方法<wbr></wbr>,即TagNode的accept方法。首先判断是否是标签结尾<wbr></wbr>,如果是就visitor.visitEndTag (this);否则visitor.visitTag (this);
- 如果是TextNode,那就visitor.visitStri<wbr></wbr>ngNode (this);就可以了。
- 如果是RemarkNode,那就visitor<wbr></wbr>.visitRemarkNode (this);就可以了。
实际上NodeVisitor里边这四种visit方法都是空的<wbr></wbr>,因为在不同的Visitor中对于这三类节点的处理是不同的<wbr></wbr>;对于需要处理的节点,只要重载对应的visit方法就行了<wbr></wbr>,如果不处理那就不理会就可以了;另外,如果用户用自己的Visi<wbr></wbr>tor,那么还可以灵活的处理不同类型的节点了。
系统为我们实现了下面我要介绍的8种Visitor<wbr></wbr>,实际上可以看作是系统给我们演示了如何做各种各样的Visito<wbr></wbr>r来访问Html,因为实际上我们要真正来用HtmlParser<wbr></wbr>的话,还需要特定的Visitor,而通过简单的这些系统提供的V<wbr></wbr>isitor组合是难以做成什么事情的。
三,系统Visitor功能简介:
四,Filter
如果说visitor是遍历提取信息,当然这个信息可以包括某些节点或者从节点分析出来的更有效的信息,这都取决于我们的Visitor做成什么样子,那么Filter则目标很明确,就是用来提取节点的。所以说要想用HtmlParser,首先要熟悉上面讲到的数据组织。
系统定义了17种具体的Filter,包括依据节点父子关系的Filter,连接Filter组合的Filter,依据网页内容匹配情况的filter,等等。我们也可以implement Filter来做自己的Filter来提取节点。
Filter的调用是同Visitor独立的,因为也无需先filter出一些NodeList,再用Visitor来访问。调用Filter的方法是:
NodeList nodeList = myParser.parse(someFilter);
解析之后,我们可以采用:
Node[] nodes = nodeList.toNodeArray();
来获取节点数组,也可以直接访问:
Node node = nodeList.elementAt(i)来获取Node。
另外,在Filter后得到NodeList以后,我们仍然可以使用NodeList的extractAllNodesThatMatch(someFilter)来进一步过滤,同时又可以用NodeList的isitAllNodesWith(someVisitor)来做进一步的访问。
这样,我们可以看到HtmlParser为我们提供了非常方便的Html解析方式,针对不同的应用可以采用visitor来遍历Html节点提取数据,也可以用Filter来过滤节点,提取出我们所关注的节点,再对节点进行处理。通过这样的组合,一定能够找出我们所需要的信息。
参考: