﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>BlogJava-走好脚下的路,让别人去说吧!-随笔分类-Java技术-基础语法</title><link>http://www.blogjava.net/human2008/category/28756.html</link><description /><language>zh-cn</language><lastBuildDate>Sat, 12 Jan 2008 12:53:00 GMT</lastBuildDate><pubDate>Sat, 12 Jan 2008 12:53:00 GMT</pubDate><ttl>60</ttl><item><title>使用Lucene进行全文检索(一)---处理索引 </title><link>http://www.blogjava.net/human2008/archive/2008/01/11/174507.html</link><dc:creator>灵!</dc:creator><author>灵!</author><pubDate>Fri, 11 Jan 2008 01:47:00 GMT</pubDate><guid>http://www.blogjava.net/human2008/archive/2008/01/11/174507.html</guid><wfw:comment>http://www.blogjava.net/human2008/comments/174507.html</wfw:comment><comments>http://www.blogjava.net/human2008/archive/2008/01/11/174507.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/human2008/comments/commentRss/174507.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/human2008/services/trackbacks/174507.html</trackback:ping><description><![CDATA[&nbsp;Lucene是一个全文检索的引擎,目前有Java和.Net 等几个版本.Java版本的网址是<a href="http://lucene.apache.org/">http://lucene.apache.org</a>.相关的一个项目是车东的WebLucene: <a href="http://sourceforge.net/projects/weblucene">http://sourceforge.net/projects/weblucene</a>.
<p>&nbsp;首先,基于一个简单的新闻系统,要想做全文检索.新闻系统的管理等在这里不在具体提出,下面列出新闻对象的类:<br />
&nbsp;<br />
&nbsp;注:程序用会到一些工具类,不在此列出,用户可以自己实现.<br />
&nbsp;<br />
&nbsp;</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;package com.jscud.website.newsinfo.bean;<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;import java.sql.Timestamp;<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;import com.jscud.util.DateTime;<br />
            &nbsp;&nbsp;import com.jscud.util.StringFunc;<br />
            &nbsp;&nbsp;import com.jscud.website.newsinfo.NewsConst;<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;/**<br />
            &nbsp;&nbsp; * 一个新闻.<br />
            &nbsp;&nbsp; * <br />
            &nbsp;&nbsp; * @author scud(飞云小侠) <a href="http://www.jscud.com/">http://www.jscud.com</a><br />
            &nbsp;&nbsp; *&nbsp; <br />
            &nbsp;&nbsp; */<br />
            &nbsp;&nbsp;public class NewsItem<br />
            &nbsp;&nbsp;{<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private int nid; //新闻编号<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private int cid; //类别编号<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private String title;//标题<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private int showtype; //内容类型:目前支持url和html<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private String content;//内容<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private String url;//对应网址,如果内容类型是url的话<br />
            &nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private Timestamp addtime; //增加时间<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; private int click; //点击数<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //对应的get,set函数,较多不在列出,可以使用工具生成<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //......<br />
            &nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * 按照类型格式化<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; public String getShowContent()<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; String sRes = content;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(showtype == NewsConst.ShowType_HTML)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }&nbsp;&nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return sRes;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; public String getTarget()<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(showtype == NewsConst.ShowType_URL)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return "_blank";<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return "";&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * 静态Html文件的路径及其名字<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; public String getHtmlFileName()<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; int nYear = DateTime.getYear_Date(getAddtime());<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; int nMonth =&nbsp; DateTime.getMonth_Date(getAddtime());<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; String sGeneFileName = <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;"/news/" + getCid() + "/" + nYear + "/" + nMonth +"/" + getNid() + ".htm";<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return sGeneFileName;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * 静态Html文件的路径<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; public String getHtmlFilePath()<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; int nYear = DateTime.getYear_Date(getAddtime());<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; int nMonth =&nbsp; DateTime.getMonth_Date(getAddtime());<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; String sGeneFilePath = <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;getCid() + "_" + nYear + "_" + nMonth;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return sGeneFilePath;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;}&nbsp;<br />
            </td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
&nbsp;<br />
&nbsp;可以看到,我们需要对标题和内容进行检索,为了这个目的,我们首先需要来研究一下lucene.<br />
&nbsp;<br />
&nbsp;在Lucene中,如果要进行全文检索,必须要先建立索引然后才能进行检索,当然实际工作中还会有删除索引和更新索引的工作.<br />
&nbsp;<br />
&nbsp;在此之前,介绍一个最基本的类(摘抄自<a href="http://www.blogjava.net/cap/archive/2005/07/17/7849.html">http://www.blogjava.net/cap/archive/2005/07/17/7849.html</a>):<br />
&nbsp;<br />
&nbsp;Analyzer 文件的分析器（听起来别扭，还是叫Analyzer好了)的抽象，这个类用来处理分词(对中文尤其重要，转换大小写(Computer-&gt;computer,实现查询大小写无关)，转换词根(computers-&gt;computer),消除stop words等,还负责把其他格式文档转换为纯文本等.<br />
&nbsp;<br />
&nbsp;在lucene中,一般会使用StandardAnalyzer来分析内容,它支持中文等多字节语言,当然可以自己实现特殊的解析器.StandardAnalyzer目前对中文的处理是按照单字来处理的,这是最简单的办法,但是也有缺点,会组合出一些没有意义的结果来.&nbsp;<br />
&nbsp;<br />
&nbsp;<br />
&nbsp;首先我们来了解建立索引,建立索引包含2种情况,一种是给一条新闻建立索引,另外的情况是在开始或者一定的时间给批量的新闻建立索引,所以为了通用,我们写一个通用的建立索引的函数:<br />
&nbsp;<br />
&nbsp;(一般一类的索引都放在一个目录下,这个配置可以在函数中定义,也可以写在配置文件中,通过参数传递给函数.)<br />
</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * 生成索引.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * <br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param doc 目标文档<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param indexDir 索引目录<br />
            &nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp; public static void makeIndex(Document doc, String indexDir)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List aList = new ArrayList();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; aList.add(doc);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; makeIndex(aList, indexDir);<br />
            &nbsp;&nbsp;&nbsp; }<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * 生成索引.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * <br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param doc 生成的document.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param indexDir 索引目录<br />
            &nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp; public static void makeIndex(List docs, String indexDir)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null == docs)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; boolean indexExist = indexExist(indexDir);
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; IndexWriter writer = null;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; try<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; StandardAnalyzer analyzer = new StandardAnalyzer();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //如果索引存在,就追加.如果不存在,就建立新的索引.lucene要是自动判决就好了.<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(indexExist) <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; writer = new IndexWriter(indexDir, analyzer, false);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; else<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; writer = new IndexWriter(indexDir, analyzer, true);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }</p>
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //添加一条文档<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; for (int i = 0; i &lt; docs.size(); i++)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Document doc = (Document) docs.get(i);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null != doc)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; writer.addDocument(doc);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }</p>
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //索引完成后的处理<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; writer.optimize();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; catch (IOException e)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; LogMan.warn("Error in Make Index", e);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; finally<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; try<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null != writer)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; writer.close();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; catch (IOException e)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; LogMan.warn("Close writer Error");<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp; }<br />
            </p>
            </td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
<br />
&nbsp;可以看到,建立索引用到类是IndexWrite,它可以新建索引或者追加索引,但是需要自己判断.判断是通过IndexReader这个类来实现的,函数如下:</p>
<p>&nbsp;</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * 检查索引是否存在.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param indexDir<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @return<br />
            &nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp; public static boolean indexExist(String indexDir)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return IndexReader.indexExists(indexDir);<br />
            &nbsp;&nbsp;&nbsp; }<br />
            &nbsp;</td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
&nbsp;如果每次都是新建索引的话,会把原来的记录删除,我在使用的时候一开始就没有注意到,后来观察了一下索引文件,才发现这个问题.<br />
&nbsp;<br />
&nbsp;<br />
&nbsp;还可以看到,建立索引是给用户的Document对象建立索引,Document表示索引中的一条文档记录.那么我们如何建立一个文档那?以新闻系统为例,代码如下:<br />
&nbsp;</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * 生成新闻的Document.<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * @param aNews 一条新闻.<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; * @return lucene的文档对象<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp;&nbsp; public static Document makeNewsSearchDocument(NewsItem aNews)<br />
            &nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Document doc = new Document();<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.Keyword("nid", String.valueOf(aNews.getNid())));<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.Text("title", aNews.getTitle()));<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //对Html进行解析,如果不是html,则不需要解析.或者根据格式调用自己的解析方法<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; String content = parseHtmlContent(aNews.getContent());<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.UnStored("content", content));<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.Keyword("addtime", aNews.getAddtime()));<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //可以加入其他的内容:例如新闻的评论等<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.UnStored("other", ""));<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; //访问url<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; String newsUrl = "/srun/news/viewhtml/" + aNews.getHtmlFilePath() + "/" + aNews.getNid()<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; + ".htm";<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; doc.add(Field.UnIndexed("visiturl", newsUrl));<br />
            &nbsp;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return doc;<br />
            &nbsp;&nbsp;&nbsp;&nbsp; }<br />
            </td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
&nbsp;<br />
&nbsp;通过上面的代码,我们把一条新闻转换为lucene的Document对象,从而进行索引工作.在上面的代码中,我们又引入了lucene中的Field(字段)类.Document文档就像数据库中的一条记录,它有很多字段,每个字段是一个Field对象.<br />
&nbsp;<br />
&nbsp;从别的文章摘抄一段关于Field的说明(摘抄自<a href="http://www.blogjava.net/cap/archive/2005/07/17/7849.html">http://www.blogjava.net/cap/archive/2005/07/17/7849.html</a>):<br />
&nbsp;[quote]<br />
&nbsp;&nbsp;&nbsp; 类型&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Analyzed Indexed Stored 说明 <br />
&nbsp;&nbsp;&nbsp; Field.Keyword(String,String/Date)&nbsp; N Y Y&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 这个Field用来储存会直接用来检索的比如(编号,姓名,日期等) <br />
&nbsp;&nbsp;&nbsp; Field.UnIndexed(String,String)&nbsp;&nbsp;&nbsp;&nbsp; N N Y&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 不会用来检索的信息,但是检索后需要显示的,比如,硬件序列号,文档的url地址 <br />
&nbsp;&nbsp;&nbsp; Field.UnStored(String,String)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Y Y N&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 大段文本内容,会用来检索,但是检索后不需要从index中取内容,可以根据url去load真实的内容 <br />
&nbsp;&nbsp;&nbsp; Field.Text(String,String)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Y Y Y&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 检索,获取都需要的内容,直接放index中,不过这样会增大index <br />
&nbsp;&nbsp;&nbsp; Field.Text(String,Reader)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Y Y N&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 如果是一个Reader, lucene猜测内容比较多,会采用Unstored的策略. <br />
&nbsp;[/quote]<br />
&nbsp;<br />
&nbsp;我们可以看到新闻的编号是直接用来检索的,所以是Keyword类型的字段,新闻的标题是需要检索和显示用的,所以是Text类型,而新闻的内容因为是Html格式的,所以在经过解析器的处理用,使用的UnStored的格式,而新闻的时间是直接用来检索的,所以是KeyWord类型.为了在新闻索引后用户可以访问到完整的新闻页面,还设置了一个UnIndexed类型的访问地址字段.<br />
&nbsp;<br />
&nbsp;(对Html进行解析的处理稍后在进行讲解)<br />
&nbsp;<br />
&nbsp;为一条新闻建立索引需要两个步骤:获取Document,传给makeIndex函数,代码如下:<br />
</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp; public static void makeNewsInfoIndex(NewsItem aNews)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null == aNews)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; makeIndex(makeNewsSearchDocument(aNews),indexDir);<br />
            &nbsp;&nbsp;&nbsp; }&nbsp;&nbsp;</td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
&nbsp;<br />
<br />
&nbsp;<br />
&nbsp;<br />
&nbsp;建立索引的工作就进行完了,只要在增加新闻后调用 makeNewsInfoIndex(newsitem); 就可以建立索引了.<br />
&nbsp;<br />
&nbsp;如果需要删除新闻,那么也要删除对应的索引,删除索引是通过IndexReader类来完成的:<br />
&nbsp;<br />
</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td><br />
            &nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * 删除索引.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param aTerm 索引删除条件<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param indexDir 索引目录<br />
            &nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp; public static void deleteIndex(Term aTerm, String indexDir)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List aList = new ArrayList();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; aList.add(aTerm);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; deleteIndex(aList, indexDir);<br />
            &nbsp;&nbsp;&nbsp; }
            <p>&nbsp;&nbsp;&nbsp; /**<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * 删除索引.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * <br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param aTerm 索引删除条件.<br />
            &nbsp;&nbsp;&nbsp;&nbsp; * @param indexDir 索引目录<br />
            &nbsp;&nbsp;&nbsp;&nbsp; *&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp; */<br />
            &nbsp;&nbsp;&nbsp; public static void deleteIndex(List terms, String indexDir)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null == terms)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if(!indexExist(indexDir)) { return; }</p>
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; IndexReader reader = null;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; try<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; reader = IndexReader.open(indexDir);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; for (int i = 0; i &lt; terms.size(); i++)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Term aTerm = (Term) terms.get(i);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null != aTerm)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; reader.delete(aTerm);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; catch (IOException e)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; LogMan.warn("Error in Delete Index", e);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; finally<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; try<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null != reader)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; reader.close();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; catch (IOException e)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; LogMan.warn("Close reader Error");<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp; }&nbsp;<br />
            </p>
            </td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
&nbsp;<br />
&nbsp;删除索引需要一个条件,类似数据库中的字段条件,例如删除一条新闻的代码如下:<br />
&nbsp;<br />
</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp;&nbsp; public static void deleteNewsInfoIndex(int nid)<br />
            &nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; Term aTerm = new Term("nid", String.valueOf(nid));<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; deleteIndex(aTerm,indexDir);<br />
            &nbsp;&nbsp;&nbsp;&nbsp; }&nbsp;&nbsp;&nbsp;</td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
<br />
<br />
&nbsp;通过新闻的ID,就可以删除一条新闻.<br />
&nbsp;<br />
&nbsp;如果需要更新新闻,如何更新索引哪? 更新索引需要先删除索引然后新建索引2个步骤,其实就是把上面的代码组合起来,例如更新一条新闻:</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp;&nbsp; public static void updateNewsInfoIndex(NewsItem aNews)<br />
            &nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null == aNews)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; return;<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; deleteNewsInfoIndex(aNews.getNid());<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; makeNewsInfoIndex(aNews);<br />
            &nbsp;&nbsp;&nbsp;&nbsp; }&nbsp;<br />
            &nbsp;</td>
        </tr>
    </tbody>
</table>
</p>
<p><br />
<br />
&nbsp;<br />
&nbsp;至此,索引的建立更新和删除就告一段落了.其中批量更新新闻的代码如下:<br />
&nbsp;(批量更新应该在访问人数少或者后台程序在夜间执行)</p>
<p>
<table class="code" cellspacing="1" cellpadding="1" width="80%" align="center" border="0">
    <tbody>
        <tr>
            <td>&nbsp;&nbsp;&nbsp; public static void makeAllNewsInfoIndex(List newsList)<br />
            &nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List terms = new ArrayList();<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; List docs = new ArrayList();
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; for (int i = 0; i &lt; newsList.size(); i++)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; NewsItem aitem = (NewsItem) newsList.get(i);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; if (null != aitem)<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; terms.add(new Term("nid", String.valueOf(aitem.getNid())));<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; docs.add(makeNewsSearchDocument(aitem));<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; }</p>
            <p>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; deleteIndex(terms,indexDir);<br />
            &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; makeIndex(docs,indexDir);<br />
            &nbsp;&nbsp;&nbsp; }&nbsp;</p>
            </td>
        </tr>
    </tbody>
</table>
</p>
<img src ="http://www.blogjava.net/human2008/aggbug/174507.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/human2008/" target="_blank">灵!</a> 2008-01-11 09:47 <a href="http://www.blogjava.net/human2008/archive/2008/01/11/174507.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>包装你的Session，使Session对象化 </title><link>http://www.blogjava.net/human2008/archive/2008/01/11/174498.html</link><dc:creator>灵!</dc:creator><author>灵!</author><pubDate>Fri, 11 Jan 2008 01:36:00 GMT</pubDate><guid>http://www.blogjava.net/human2008/archive/2008/01/11/174498.html</guid><wfw:comment>http://www.blogjava.net/human2008/comments/174498.html</wfw:comment><comments>http://www.blogjava.net/human2008/archive/2008/01/11/174498.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/human2008/comments/commentRss/174498.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/human2008/services/trackbacks/174498.html</trackback:ping><description><![CDATA[&nbsp;在Web开发中，经常需要使用Session来保存特定用户的信息，在我们的程序中很多地方散落着类似下面的语句：<br />
<div style="border-right: #cccccc 1px solid; padding-right: 5px; border-top: #cccccc 1px solid; padding-left: 4px; font-size: 13px; padding-bottom: 4px; border-left: #cccccc 1px solid; width: 98%; word-break: break-all; padding-top: 4px; border-bottom: #cccccc 1px solid; background-color: #eeeeee"><img src="http://www.cnblogs.com/Images/OutliningIndicators/None.gif" align="top"  alt="" /><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">int</span><span style="color: #000000">&nbsp;userAge&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;(</span><span style="color: #0000ff">int</span><span style="color: #000000">)</span><span style="color: #0000ff">this</span><span style="color: #000000">.Session[</span><span style="color: #000000">"</span><span style="color: #000000">UserAge</span><span style="color: #000000">"</span><span style="color: #000000">];</span></div>
&nbsp;&nbsp;&nbsp;&nbsp;<br />
&nbsp;&nbsp;&nbsp; 我们知道，Session中存放的是键值对，键是string类型的，如果我们一不小心把上面的语句写成这样：<br />
<div style="border-right: #cccccc 1px solid; padding-right: 5px; border-top: #cccccc 1px solid; padding-left: 4px; font-size: 13px; padding-bottom: 4px; border-left: #cccccc 1px solid; width: 98%; word-break: break-all; padding-top: 4px; border-bottom: #cccccc 1px solid; background-color: #eeeeee"><img src="http://www.cnblogs.com/Images/OutliningIndicators/None.gif" align="top"  alt="" /><span style="color: #000000">&nbsp;&nbsp;</span><span style="color: #0000ff">int</span><span style="color: #000000">&nbsp;userAge&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;(</span><span style="color: #0000ff">int</span><span style="color: #000000">)</span><span style="color: #0000ff">this</span><span style="color: #000000">.Session[</span><span style="color: #000000">"</span><span style="color: #000000"><span style="color: red"><span style="color: red">Use</span></span>Age</span><span style="color: #000000">"</span><span style="color: #000000">];</span></div>
&nbsp;&nbsp;&nbsp; 编译期不会发现这个错误，但运行时一定会抛出异常，这是在程序中直接操作Session可能引发的问题之一。另外，每次获取userAge的时候都要写代码进行强制转换，感觉很繁琐。我们需要一个解决方案来解决这些问题。我的做法是引入一个Session的包装，使之对象化、强类型化。就像接下来的例子一样：<br />
<div style="border-right: #cccccc 1px solid; padding-right: 5px; border-top: #cccccc 1px solid; padding-left: 4px; font-size: 13px; padding-bottom: 4px; border-left: #cccccc 1px solid; width: 98%; word-break: break-all; padding-top: 4px; border-bottom: #cccccc 1px solid; background-color: #eeeeee"><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">class</span><span style="color: #000000">&nbsp;SessionHelper<br />
{<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">private</span><span style="color: #000000">&nbsp;HttpSessionState&nbsp;curSession;<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;SessionHelper(HttpSessionState&nbsp;session)<br />
&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;session;<br />
&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">static</span><span style="color: #000000">&nbsp;SessionHelper&nbsp;CreateInstance(HttpSessionState&nbsp;session)<br />
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">return</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">new</span><span style="color: #000000">&nbsp;SessionHelper(session);<br />
&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">string</span><span style="color: #000000">&nbsp;UserID<br />
&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">get</span><span style="color: #000000"><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">return</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">UserID</span><span style="color: #000000">"</span><span style="color: #000000">].ToString();<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">set</span><span style="color: #000000"><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">UserID</span><span style="color: #000000">"</span><span style="color: #000000">]&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;value&nbsp;;<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">int</span><span style="color: #000000">&nbsp;UserAge<br />
&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">get</span><span style="color: #000000"><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">return</span><span style="color: #000000">&nbsp;(</span><span style="color: #0000ff">int</span><span style="color: #000000">)</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">UserAge</span><span style="color: #000000">"</span><span style="color: #000000">];<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">set</span><span style="color: #000000"><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">UserAge</span><span style="color: #000000">"</span><span style="color: #000000">]&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;value&nbsp;;<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000">//</span><span style="color: #008000">某用户上传的所有图片</span><span style="color: #008000"><br />
</span><span style="color: #000000">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;ArrayList&nbsp;PicList<br />
&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">get</span><span style="color: #000000"><br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">if</span><span style="color: #000000">&nbsp;(</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">PicList</span><span style="color: #000000">"</span><span style="color: #000000">]&nbsp;</span><span style="color: #000000">==</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">null</span><span style="color: #000000">)<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">PicList</span><span style="color: #000000">"</span><span style="color: #000000">]&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">new</span><span style="color: #000000">&nbsp;ArrayList();<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">return</span><span style="color: #000000">&nbsp;(ArraayList)</span><span style="color: #0000ff">this</span><span style="color: #000000">.curSession[</span><span style="color: #000000">"</span><span style="color: #000000">PicList</span><span style="color: #000000">"</span><span style="color: #000000">];<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<br />
&nbsp;&nbsp;&nbsp;&nbsp;}<br />
<br />
&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000">//</span><span style="color: #008000">清空图片列表</span><span style="color: #008000"><br />
</span><span style="color: #000000">&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">public</span><span style="color: #000000">&nbsp;</span><span style="color: #0000ff">void</span><span style="color: #000000">&nbsp;ClearAllPics()<br />
&nbsp;&nbsp;&nbsp;&nbsp;{<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #0000ff">this</span><span style="color: #000000">.PicList.Clear();<br />
&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;<br />
}</span></div>
<br />
&nbsp;&nbsp;&nbsp; 这样，我们用起来就非常方便了：<br />
<div style="border-right: #cccccc 1px solid; padding-right: 5px; border-top: #cccccc 1px solid; padding-left: 4px; font-size: 13px; padding-bottom: 4px; border-left: #cccccc 1px solid; width: 98%; word-break: break-all; padding-top: 4px; border-bottom: #cccccc 1px solid; background-color: #eeeeee"><span style="color: #000000">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; SessionHelper&nbsp;sessionHelper&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;SessionHelper.CreateInstance(</span><span style="color: #0000ff">this</span><span style="color: #000000">.Session);<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ArrayList&nbsp;picList&nbsp;</span><span style="color: #000000">=</span><span style="color: #000000">&nbsp;sessionHelper.PicList;<br />
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</span><span style="color: #008000">//</span><span style="color: #008000">&nbsp;<img src="http://www.cnblogs.com/Images/dot.gif"  alt="" />&nbsp;<img src="http://www.cnblogs.com/Images/dot.gif"  alt="" />&nbsp;&nbsp;处理picList中的图片</span><span style="color: #008000"><br />
</span><span style="color: #000000">&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;sessionHelper.ClearAllPics();&nbsp;&nbsp;</span></div>
<br />
&nbsp;&nbsp;&nbsp; 引入这一层包装，可以使我们的程序的可读性、可维护性更好，而且将原来的一些运行期的错误提前到了编译期，这也是强类型带来的好处。<br />
<img src ="http://www.blogjava.net/human2008/aggbug/174498.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/human2008/" target="_blank">灵!</a> 2008-01-11 09:36 <a href="http://www.blogjava.net/human2008/archive/2008/01/11/174498.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>迭代器 </title><link>http://www.blogjava.net/human2008/archive/2008/01/10/174228.html</link><dc:creator>灵!</dc:creator><author>灵!</author><pubDate>Thu, 10 Jan 2008 03:42:00 GMT</pubDate><guid>http://www.blogjava.net/human2008/archive/2008/01/10/174228.html</guid><wfw:comment>http://www.blogjava.net/human2008/comments/174228.html</wfw:comment><comments>http://www.blogjava.net/human2008/archive/2008/01/10/174228.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/human2008/comments/commentRss/174228.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/human2008/services/trackbacks/174228.html</trackback:ping><description><![CDATA[<div class="postText">迭代这个名词对于熟悉Java的人来说绝对不陌生。我们常常使用JDK提供的迭代接口进行java collection的遍历：<br />
<br />
<table bordercolor="#cccccc" width="90%" align="center" bgcolor="#e3e3e3" border="1">
    <tbody>
        <tr>
            <td>Iterator it = list.iterator();<br />
            while(it.hasNext()){<br />
            　//using &#8220;it.next();&#8221;do some businesss logic<br />
            }</td>
        </tr>
    </tbody>
</table>
<br />
　　而这就是关于迭代器模式<a class="bluekey" href="http://www.yesky.com/key/1668/146668.html" target="_blank"><u><font color="#0000ff">应用</font></u></a>很好的例子。<br />
<br />
　　<strong>二、 定义与<a class="bluekey" href="http://www.yesky.com/key/3031/148031.html" target="_blank"><u><font color="#0000ff">结构</font></u></a></strong><br />
<br />
　　迭代器（Iterator）模式，又叫做游标（Cursor）模式。GOF给出的定义为：提供一种方法访问一个容器（container）对象中各个元素，而又不需暴露该对象的内部细节。<br />
<br />
　　从定义可见，迭代器模式是为容器而生。很明显，对容器对象的访问必然涉及到遍历算法。你可以一股脑的将遍历方法塞到容器对象中去；或者根本不去提供什么遍历算法，让使用容器的人自己去实现去吧。这两种情况好像都能够解决问题。<br />
<br />
　　然而在前一种情况，容器承受了过多的功能，它不仅要负责自己&#8220;容器&#8221;内的元素维护（添加、删除等等），而且还要提供遍历自身的接口；而且由于遍历状态保存的问题，不能对同一个容器对象同时进行多个遍历。第二种方式倒是省事，却又将容器的内部细节暴露无遗。<br />
<br />
　　而迭代器模式的出现，很好的解决了上面两种情况的弊端。先来看下迭代器模式的真面目吧。 <br />
<br />
　　迭代器模式由以下角色组成：<br />
<br />
　　1) 迭代器角色（Iterator）：迭代器角色负责定义访问和遍历元素的接口。<br />
<br />
　　2) 具体迭代器角色（Concrete Iterator）：具体迭代器角色要实现迭代器接口，并要记录遍历中的当前位置。<br />
<br />
　　3) 容器角色（Container）：容器角色负责提供创建具体迭代器角色的接口。<br />
<br />
　　4) 具体容器角色（Concrete Container）：具体容器角色实现创建具体迭代器角色的接口——这个具体迭代器角色于该容器的结构相关。<br />
<br />
　　迭代器模式的类图如下：<br />
<br />
<table width="90%" align="center" border="0">
    <tbody>
        <tr>
            <td>
            <div align="center"><img style="width: 500px" alt="" src="http://www.j2medev.com/Article/UploadFiles/200511/20051121225925642.jpg" width="500" /></div>
            </td>
        </tr>
    </tbody>
</table>
<br />
　　从结构上可以看出，迭代器模式在客户与容器之间加入了迭代器角色。迭代器角色的加入，就可以很好的避免容器内部细节的暴露，而且也使得设计符号&#8220;单一职责原则&#8221;。<br />
<br />
　　注意，在迭代器模式中，具体迭代器角色和具体容器角色是耦合在一起的——遍历算法是与容器的内部细节紧密相关的。为了使客户程序从与具体迭代器角色耦合的困境中脱离出来，避免具体迭代器角色的更换给客户程序带来的修改，迭代器模式抽象了具体迭代器角色，使得客户程序更具一般性和重用性。这被称为多态迭代。<br />
<br />
　　<strong>三、 举例</strong><br />
<br />
　　由于迭代器模式本身的规定比较松散，所以具体实现也就五花八门。我们在此仅举一例，根本不能将实现方式一一呈现。因此在举例前，我们先来列举下迭代器模式的实现方式。 <br />
<br />
　　1．迭代器角色定义了遍历的接口，但是没有规定由谁来控制迭代。在Java collection的应用中，是由客户程序来控制遍历的进程，被称为外部迭代器；还有一种实现方式便是由迭代器自身来控制迭代，被称为内部迭代器。外部迭代器要比内部迭代器灵活、强大，而且内部迭代器在java语言环境中，可用性很弱。<br />
<br />
　　2．在迭代器模式中没有规定谁来实现遍历算法。好像理所当然的要在迭代器角色中实现。因为既便于一个容器上使用不同的遍历算法，也便于将一种遍历算法应用于不同的容器。但是这样就破坏掉了容器的封装——容器角色就要公开自己的私有属性，在java中便意味着向其他类公开了自己的私有属性。<br />
<br />
　　那我们把它放到容器角色里来实现好了。这样迭代器角色就被架空为仅仅存放一个遍历当前位置的功能。但是遍历算法便和特定的容器紧紧绑在一起了。<br />
<br />
　　而在Java Collection的应用中，提供的具体迭代器角色是定义在容器角色中的内部类。这样便保护了容器的封装。但是同时容器也提供了遍历算法接口，你可以扩展自己的迭代器。<br />
<br />
　　好了，我们来看下Java Collection中的迭代器是怎么实现的吧。<br />
<br />
<table bordercolor="#cccccc" width="90%" align="center" bgcolor="#e3e3e3" border="1">
    <tbody>
        <tr>
            <td>//迭代器角色，仅仅定义了遍历接口<br />
            <br />
            public interface Iterator {<br />
            　boolean hasNext();<br />
            　Object next();<br />
            　void remove();<br />
            }<br />
            <br />
            //容器角色，这里以List为例。它也仅仅是一个接口，就不罗列出来了<br />
            //具体容器角色，便是实现了List接口的<a class="bluekey" href="http://www.yesky.com/key/4119/129119.html" target="_blank"><u><font color="#0000ff">ArrayList</font></u></a>等类。为了突出重点这里指罗列和迭代器相关的内容<br />
            //具体迭代器角色，它是以内部类的形式出来的。AbstractList是为了将各个具体容器角色的<a class="bluekey" href="http://www.yesky.com/key/145/150145.html" target="_blank"><u><font color="#0000ff">公共</font></u></a>部分提取出来而存在的。<br />
            <br />
            public abstract class AbstractList extends AbstractCollection implements List {<br />
            &#8230;&#8230; <br />
            //这个便是负责创建具体迭代器角色的工厂方法<br />
            public Iterator iterator() {<br />
            　return new Itr();<br />
            }<br />
            <br />
            //作为内部类的具体迭代器角色<br />
            <br />
            private class Itr implements Iterator {<br />
            　int cursor = 0;<br />
            　int lastRet = -1;<br />
            　int expectedModCount = modCount;<br />
            <br />
            　public boolean hasNext() {<br />
            　　return cursor != size();<br />
            　}<br />
            <br />
            　public Object next() {<br />
            　　checkForComodification();<br />
            　　try {<br />
            　　　Object next = get(cursor);<br />
            　　　lastRet = cursor++;<br />
            　　　return next;<br />
            　　} catch(IndexOutOfBoundsException e) {<br />
            　　　checkForComodification();<br />
            　　　throw new NoSuchElementException();<br />
            　　}<br />
            　}<br />
            <br />
            　public void remove() {<br />
            　　<a class="bluekey" href="http://www.yesky.com/key/2274/147274.html" target="_blank"><u><font color="#0000ff">if</font></u></a> (lastRet == -1)<br />
            　　　throw new IllegalStateException();<br />
            　　　checkForComodification();<br />
            <br />
            　　try {<br />
            　　　AbstractList.this.remove(lastRet);<br />
            　　　if (lastRet &lt; cursor)<br />
            　　　　cursor--;<br />
            　　　lastRet = -1;<br />
            　　　expectedModCount = modCount;<br />
            　　} catch(IndexOutOfBoundsException e) {<br />
            　　　throw new ConcurrentModificationException();<br />
            　　}<br />
            　}<br />
            <br />
            　final void checkForComodification() {<br />
            　　if (modCount != expectedModCount)<br />
            　　　throw new ConcurrentModificationException();<br />
            　}<br />
            }</td>
        </tr>
    </tbody>
</table>
<br />
　　至于迭代器模式的使用。正如引言中所列那样，客户程序要先得到具体容器角色，然后再通过具体容器角色得到具体迭代器角色。这样便可以使用具体迭代器角色来遍历容器了&#8230;&#8230;<br />
<br />
　　四、 实现自己的迭代器<br />
<br />
　　在实现自己的迭代器的时候，一般要操作的容器有支持的接口才可以。而且我们还要注意以下问题： <br />
<br />
　　在迭代器遍历的过程中，通过该迭代器进行容器元素的增减操作是否安全呢？<br />
<br />
　　在容器中存在复合对象的情况，迭代器怎样才能支持深层遍历和多种遍历呢？<br />
<br />
　　以上两个问题对于不同结构的容器角色，各不相同，值得考虑。<br />
<br />
　　五、 适用情况 <br />
<br />
　　由上面的讲述，我们可以看出迭代器模式给容器的应用带来以下好处：<br />
<br />
　　1) 支持以不同的方式遍历一个容器角色。根据实现方式的不同，效果上会有差别。<br />
<br />
　　2) 简化了容器的接口。但是在java Collection中为了提高可扩展性，容器还是提供了遍历的接口。<br />
<br />
　　3) 对同一个容器对象，可以同时进行多个遍历。因为遍历状态是保存在每一个迭代器对象中的。<br />
<br />
　　由此也能得出迭代器模式的适用范围：<br />
<br />
　　1) 访问一个容器对象的内容而无需暴露它的内部表示。<br />
<br />
　　2) 支持对容器对象的多种遍历。<br />
<br />
　　3) 为遍历不同的容器结构提供一个统一的接口（多态迭代）。<br />
<br />
　　六、 总结<br />
<br />
　　迭代器模式在我们的应用中很广泛，希望本文能帮助你理解它。如有不对之处，还请不吝指正。 </div>
<img src ="http://www.blogjava.net/human2008/aggbug/174228.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/human2008/" target="_blank">灵!</a> 2008-01-10 11:42 <a href="http://www.blogjava.net/human2008/archive/2008/01/10/174228.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>JAVA数据类型转换 </title><link>http://www.blogjava.net/human2008/archive/2008/01/10/174224.html</link><dc:creator>灵!</dc:creator><author>灵!</author><pubDate>Thu, 10 Jan 2008 03:38:00 GMT</pubDate><guid>http://www.blogjava.net/human2008/archive/2008/01/10/174224.html</guid><wfw:comment>http://www.blogjava.net/human2008/comments/174224.html</wfw:comment><comments>http://www.blogjava.net/human2008/archive/2008/01/10/174224.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/human2008/comments/commentRss/174224.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/human2008/services/trackbacks/174224.html</trackback:ping><description><![CDATA[<p>基本类型有以下四种：<br />
int长度数据类型有：byte(8bits)、short(16bits)、int(32bits)、long(64bits)、<br />
float长度数据类型有：单精度（32bits float）、双精度（64bits double）<br />
boolean类型变量的取值有：ture、false<br />
char数据类型有：unicode字符,16位<br />
对应的类类型：<br />
Integer、Float、Boolean、Character、Double、Short、Byte、Long</p>
<p><br />
转换原则：<br />
从低精度向高精度转换<br />
byte 、short、int、long、float、double、char<br />
注：两个char型运算时，自动转换为int型；当char与别的类型运算时，也会先自动转换为int型的，再做其它类型的自动转换 </p>
<p>基本类型向类类型转换</p>
<p>正向转换：<br />
通过类包装器来new出一个新的类类型的变量<br />
Integer a= new Integer(2);</p>
<p>反向转换：<br />
通过类包装器来转换<br />
int b=a.intValue();<br />
类类型向字符串转换</p>
<p><br />
正向转换：<br />
因为每个类都是object类的子类，而所有的object类都有一个toString()函数，所以通过toString()函数来转换即可</p>
<p>反向转换：<br />
通过类包装器new出一个新的类类型的变量<br />
eg1: int i=Integer.valueOf(&#8220;123&#8221;).intValue()<br />
说明：上例是将一个字符串转化成一个Integer对象，然后再调用这个对象的intValue()方法返回其对应的int数值。<br />
eg2: float f=Float.valueOf(&#8220;123&#8221;).floatValue()<br />
说明：上例是将一个字符串转化成一个Float对象，然后再调用这个对象的floatValue()方法返回其对应的float数值。<br />
eg3: boolean b=Boolean.valueOf(&#8220;123&#8221;).booleanValue()<br />
说明：上例是将一个字符串转化成一个Boolean对象，然后再调用这个对象的booleanValue()方法返回其对应的boolean数值。<br />
eg4:Double d=Double.valueOf(&#8220;123&#8221;).doubleValue()<br />
说明：上例是将一个字符串转化成一个Double对象，然后再调用这个对象的doubleValue()方法返回其对应的double数值。<br />
eg5: long l=Long.valueOf(&#8220;123&#8221;).longValue()<br />
说明：上例是将一个字符串转化成一个Long对象，然后再调用这个对象的longValue()方法返回其对应的long数值。<br />
eg6: char=Character.valueOf(&#8220;123&#8221;).charValue()<br />
说明：上例是将一个字符串转化成一个Character对象，然后再调用这个对象的charValue()方法返回其对应的char数值。</p>
<p><br />
基本类型向字符串的转换</p>
<p>正向转换：<br />
如：int a=12;<br />
String b;<br />
b=a+&#8221;&#8221;;</p>
<p>反向转换：<br />
通过类包装器<br />
eg1:<br />
int i=Integer.parseInt(&#8220;123&#8221;)<br />
说明：此方法只能适用于字符串转化成整型变量<br />
eg2: float f=Float.valueOf(&#8220;123&#8221;).floatValue()<br />
说明：上例是将一个字符串转化成一个Float对象，然后再调用这个对象的floatValue()方法返回其对应的float数值。<br />
eg3: boolean b=Boolean.valueOf(&#8220;123&#8221;).booleanValue()<br />
说明：上例是将一个字符串转化成一个Boolean对象，然后再调用这个对象的booleanValue()方法返回其对应的boolean数值。<br />
eg4:Double d=Double.valueOf(&#8220;123&#8221;).doubleValue()<br />
说明：上例是将一个字符串转化成一个Double对象，然后再调用这个对象的doubleValue()方法返回其对应的double数值。<br />
eg5: long l=Long.valueOf(&#8220;123&#8221;).longValue()<br />
说明：上例是将一个字符串转化成一个Long对象，然后再调用这个对象的longValue()方法返回其对应的long数值。<br />
eg6: char=Character.valueOf(&#8220;123&#8221;).charValue()<br />
说明：上例是将一个字符串转化成一个Character对象，然后再调用这个对象的charValue()方法返回其对应的char数值。</p>
<img src ="http://www.blogjava.net/human2008/aggbug/174224.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/human2008/" target="_blank">灵!</a> 2008-01-10 11:38 <a href="http://www.blogjava.net/human2008/archive/2008/01/10/174224.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>java 数据库基本操作 </title><link>http://www.blogjava.net/human2008/archive/2008/01/10/174222.html</link><dc:creator>灵!</dc:creator><author>灵!</author><pubDate>Thu, 10 Jan 2008 03:37:00 GMT</pubDate><guid>http://www.blogjava.net/human2008/archive/2008/01/10/174222.html</guid><wfw:comment>http://www.blogjava.net/human2008/comments/174222.html</wfw:comment><comments>http://www.blogjava.net/human2008/archive/2008/01/10/174222.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/human2008/comments/commentRss/174222.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/human2008/services/trackbacks/174222.html</trackback:ping><description><![CDATA[java <a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>基本操作
<p>　　1、java<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>操作基本流程</p>
<p>　　2、几个常用的重要技巧：</p>
<p>　　&#183;可滚动、更新的记录集</p>
<p>　　&#183;批量更新</p>
<p>　　&#183;事务处理</p>
<p>　　java<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>操作基本流程：取得<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>连接 - 执行sql语句 - 处理执行结果 - 释放<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>连接</p>
<p>　　<strong>1、取得<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>连接</strong></p>
<p>　　1)用DriverManager取<a href="http://www.zhujiangroad.com/price/soft_sql.html" target="_blank"><font color="#0000ff">数据库</font></a>连接</p>
<p>　　例子</p>
<p>　　String className,url,uid,pwd;</p>
<p>　　className = "oracle.jdbc.driver.<a href="http://www.zhujiangroad.com/program/Oracle.html" target="_blank"><font color="#0000ff">Oracle</font></a>Driver";</p>
<p>　　url = "jdbc:oracle:thin:@127.0.0.1:1521:orasvr;</p>
<p>　　uid = "system";</p>
<p>　　pwd = "manager";</p>
<p>　　Class.forName(className);</p>
<p>　　Connection cn = DriverManager.getConnection(url,uid,pwd);</p>
<p>　　2)用jndi(java的命名和目录服务)方式</p>
<p>　　例子</p>
<p>　　String jndi = "jdbc/db";</p>
<p>　　Context ctx = (Context) new InitialContext().lookup("java:comp/env");</p>
<p>　　DataSource ds = (DataSource) ctx.lookup(jndi);</p>
<p>　　Connection cn = ds.getConnection();</p>
<p>　　多用于jsp中</p>
<p>　　<strong>2、执行sql语句</strong></p>
<p>　　1)用Statement来执行sql语句</p>
<p>　　String sql;</p>
<p>　　Statement sm = cn.createStatement();</p>
<p>　　sm.executeQuery(sql); // 执行数据查询语句(select)</p>
<p>　　sm.executeUpdate(sql); // 执行数据更新语句(delete、update、insert、drop等)statement.close();</p>
<p>　　2)用PreparedStatement来执行sql语句</p>
<p>　　String sql;</p>
<p>　　sql = "insert into user (id,name) values (?,?)";</p>
<p>　　PreparedStatement ps = cn.prepareStatement(sql);</p>
<p>　　ps.setInt(1,xxx);</p>
<p>　　ps.setString(2,xxx);</p>
<p>　　...</p>
<p>　　ResultSet rs = ps.executeQuery(); // 查询</p>
<p>　　int c = ps.executeUpdate(); // 更新</p>
<p>　　<strong>3、处理执行结果</strong></p>
<p>　　查询语句，返回记录集ResultSet</p>
<p>　　更新语句，返回数字，表示该更新影响的记录数</p>
<p>　　ResultSet的方法</p>
<p>　　1、next()，将游标往后移动一行，如果成功返回true;否则返回false</p>
<p>　　2、getInt("id")或getSting("name")，返回当前游标下某个字段的值</p>
<p>　　<strong>4、释放连接</strong></p>
<p>　　cn.close();</p>
<p>　　一般，先关闭ResultSet，然后关闭Statement(或者PreparedStatement);最后关闭Connection</p>
<p>　　可滚动、更新的记录集</p>
<p>　　1、创建可滚动、更新的Statement</p>
<p>　　Statement sm = cn.createStatement(ResultSet.TYPE_SCROLL_ENSITIVE,ResultSet.CONCUR_READ_ONLY);</p>
<p>　　该Statement取得的ResultSet就是可滚动的</p>
<p>　　2、创建PreparedStatement时指定参数</p>
<p>　　PreparedStatemet ps = cn.prepareStatement(sql,ResultSet.TYPE_SCROLL_INSENSITIVE,ResultSet.CONCUR_READ_ONLY);</p>
<p>　　ResultSet.absolute(9000);</p>
<p>　　&#183;批量更新</p>
<p>　　1、Statement</p>
<p>　　Statement sm = cn.createStatement();</p>
<p>　　sm.addBatch(sql1);</p>
<p>　　sm.addBatch(sql2);</p>
<p>　　...</p>
<p>　　sm.executeBatch()</p>
<p>　　一个Statement对象，可以执行多个sql语句以后，批量更新。这多个语句可以是delete、update、insert等或兼有</p>
<p>　　2、PreparedStatement</p>
<p>　　PreparedStatement ps = cn.preparedStatement(sql);</p>
<p>　　{</p>
<p>　　ps.setXXX(1,xxx);</p>
<p>　　...</p>
<p>　　ps.addBatch();</p>
<p>　　}</p>
<p>　　ps.executeBatch();</p>
<p>　　一个PreparedStatement，可以把一个sql语句，变换参数多次执行，一次更新。</p>
<p>　　&#183;事务的处理</p>
<p>　　1、关闭Connection的自动提交</p>
<p>　　cn.setAutoCommit(false);</p>
<p>　　2、执行一系列sql语句</p>
<p>　　要点：执行每一个新的sql语句前，上一次执行sql语句的Statement(或者PreparedStatemet)必须先close</p>
<p>　　Statement sm ;</p>
<p>　　sm = cn.createStatement(insert into user...);</p>
<p>　　sm.executeUpdate();</p>
<p>　　sm.close();</p>
<p>　　sm = cn.createStatement("insert into corp...);</p>
<p>　　sm.executeUpdate();</p>
<p>　　sm.close();</p>
<p>　　3、提交</p>
<p>　　cn.commit();</p>
<p>　　4、如果发生异常，那么回滚</p>
<p>　　cn.rollback();</p>
<img src ="http://www.blogjava.net/human2008/aggbug/174222.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/human2008/" target="_blank">灵!</a> 2008-01-10 11:37 <a href="http://www.blogjava.net/human2008/archive/2008/01/10/174222.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>