??xml version="1.0" encoding="utf-8" standalone="yes"?>
樂觀鎖定QOptimistic lockingQ則樂觀的認料的存取很少發生同時存取的問,因而不作資料n層次上的鎖定Q為了維h的資料Q樂觀鎖定使用應用E式上的邏輯實現版本控制的解決?
在不實行悲觀鎖定{略的情況下Q資料不一致的情況一但發生,有幾個解決的ҎQ一E是先更新為主,一E是後更新的ZQ比較複雜的是檢查發生變動的資料來實現Q或是檢查所有屬性來實現樂觀鎖定?
Hibernate中透過版本號檢查來實現後更新為主,這也是Hibernate所推薦的方式,在資料n中加入一個versionƄ位a錄Q在讀取資料時連同版本號一同讀取,並在更新資料時比版本號與資料n中的版本號,如果{於資料庫中的版本號則予以更斎ͼ並遞增版本號Q如果小D料n中的版本號就丟出例外?br />
實際來透過例瞭解Hibernate的樂觀鎖定如何實現Q首先在資料庫中新增一個表|
這個user表格中的version用來a錄版本號,以供Hibernate實現樂觀鎖定Q接著設aUser別Q當中必須包括version屬性:
package onlyfun.caterpillar;
public class User {
private Integer id;
private Integer version; // 增加版本屬?
private String name;
private Integer age;
public User() {
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public Integer getVersion() {
return version;
}
public void setVersion(Integer version) {
this.version = version;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
}
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping
PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="onlyfun.caterpillar.User"
table="user"
optimistic-lock="version">
<id name="id" column="id" type="java.lang.Integer">
<generator class="native"/>
</id>
<version name="version"
column="version"
type="java.lang.Integer"/>
<property name="name" column="name" type="java.lang.String"/>
<property name="age" column="age" type="java.lang.Integer"/>
</class>
</hibernate-mapping>
注意<version>標籤必須出現?lt;id>標籤之後Q接著您可以試著在資料n中新增資料,例如Q?br />
您可以檢視資料n中的資料Q每一ơ對同一{資料進行更新QversionƄ位的內定w會自動更斎ͼ接著來作個實驗,直接以範例說明:
運行以下的程式片D,會出現以下的i果Q?br />
由於新的版本號是1Q而userV2的版本號還是0Q因此更新失敗丟出StableObjectStateExceptionQ您可以捕捉這個例外作善後處理Q例如在處理中重新讀取資料n中的資料Q同時將目前的資料與資料庫中的資料秀ZQ讓使用者有會比對不一致的資料Q以決定要變更的部䆾Q或者您可以a計E式自動讀取新的資料,並比真正要更新的資料,這一切可以在背景埯Q而不用讓您的使用者知道?
要注意的是,由於樂觀鎖定是用系i׃的程式來控制Q而不是用資料n中的鎖定制Q因而如果有人特意自行更新版本訊息來過檢查Q則鎖定制會無效Q例如在上例中自行更改userV2的version屬性,使之與資料n中的版本號相同的話就不會有錯誤,像這樣版本號被更改Q或是由D料是由外部系ip來Q因而版本資a不受控制時Q鎖定機制將會有問題Q設a時必須注意?br />
from : http://caterpillar.onlyfun.net/Gossip/HibernateGossip/HibernateGossip.html
?概况Q这个API扩展了java.net包,提供了模拟浏览器的功能?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>?
?何时适用Q当你要构造Web览器的功能Q当你的应用需要一U高效的办法q行HTTP/HTTPS通信时?
?CZ应用QHttpClientDemo.java。要求CLASSPATH中有commons-httpclient.jarQcommon-logging.jar。要求用JDK 1.4或更高版本?
?说明Q?
HttpClient扩展和增Z标准java.net包,是一个内容广泛的代码库,功能极其丰富Q能够构造出各种使用HTTP协议的分布式应用Q或者也可以嵌入到现有应用,为应用增加访问HTTP协议的能力。在CommonsE_版中QHttpClient的文档似乎要比其他包更完善一些,而且q带有几个实例。下面我们通过一个简单的例子来了解如何提取一个Web面QHttpClient文档中也有一个类似的例子Q我们将扩充那个例子使其支持SSL。注意本例需要JDK 1.4支持Q因为它要用到Java Secure Socket Connection库,而这个库只有JDK 1.4及更高的版本才提供?
?首先定一个可以通过HTTPS下蝲的页面,本例使用的是https://www.paypal.com/。同时确?JAVA_HOME%/jre/lib/security/java.security文g包含了下面这行代码:security.provider.2=com.sun.net.ssl.internal.ssl.Provider?
除了q些讄之外QHTTPSq接的处理方式没有其他特别的地方--臛_对于本例来说如此。不q,如果q程|站使用的根证书不被你用的Java认可Q则首先必须导入它的证书?
?创徏一个HttpClient的实例。HttpClientcd以看成是应用的主驱动E序Q所有针对网l的功能都依赖于它。HttpClientc需要一个Connection Manager来管理连接。HttpConnectionManager允许我们创徏自己的连接管理器Q或者,我们也可以直接用内建的SimpleHttpConnectionManager或MultiThreadedHttpConnectionManagercR如果在创徏HttpClient时没有指定连接管理器QHttpClient默认使用SimpleHttpConnectionManager?
以上只是非常单地介绍了一下HttpClient库,HttpClient实际的功能要比本文介l的丰富得多Q不仅健壮而且高效Q请参阅API文档了解详情?
3.3 Net
?概况Q一个用于操作Internet基础协议的底层API?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>?
?何时适用Q当你想要访问各UInternet底层协议之时QFingerQWhoisQTFTPQTelnetQPOP3QFTPQNNTPQ以及SMTPQ?
?CZ应用QNetDemo.java。要求CLASSPATH中包含commons-net-1.0.0.jar?
?说明Q?
Net包是一个强大、专业的cdQ类库里的类最初属于一个叫做NetComponents的商业品?
Net包不仅支持对各种低层ơ协议的讉KQ而且q提供了一个高层的抽象。大多数情况下,Net包提供的抽象已能满一般需要,它得开发者不再需要直接面对各U协议的SocketU的低层命o。用高层抽象ƈ不减Q何功能,Net API在这斚w做得很出Ԍ既提供了_的功能,又不至于在特色方面作q多的妥协?
SocketClient是支持所有协议的基础c,它是一个抽象类Q聚合了各种协议都需要的公用功能。各U不同协议的使用q程其实很相|首先利用connectҎ建立一个指向远E服务器的连接,执行必要的操作,最后终止与服务器的q接。下面通过实例介绍具体的用步骤?
其他客户端,例如FingerClient、POP3Client、TelnetClient{,用法也差不多?
l束语:有关Web相关cd其他cȝ介绍到此结束。在下一文章中Q我们将探讨XMLcd包装c,最后一文章则介绍工具cR?
希望读者有兴趣试试本文提供的程序实例。很多时候Jakarta Commonslh以q感觉Q希望本文你加׃对Jakarta Commons了解Q或者至引起了你对Commons子项目以及它提供的各U实用API和库的兴?
请从q里下蝲本文代码Q?a href="" target="_blank">JakartaCommons1_code.zip
以后我们可以按照下面的方式使用枚D变量QOnTV.getEnum("Idol")。该调用从前面创建的枚D数据cdq回Idol。这个例子比较简单,实际上Enumc还提供了许多有用的ҎQ请参见本文后面提供的完整实例?
2.4 Collections
?概况Q扩展了Java Collection框架Q增M新的数据l构、P代机制和比较操作W?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>?
?何时适用Q几乎所有需要操作数据结构的重要Java开发项目都可以使用Collections API。和Java的标准实现相比,Collections API有着诸多优势?
?CZ应用QCollectionsDemo.java。要求CLASSPATH中包含commons-collections.jar?
?说明Q?
要在有限的文章篇q之内详地介绍 Collections API实在是太困难了,不过q里仍将늛大多数最重要的类Q希望能够引起你的兴,认真了解一下其余的cRCollections本n的文档也提供了许多资料ƈ解释了每一个类的用法?
Bag接口扩展标准的Java CollectionQ允许生成计数器来跟tBag里面的所有元素。当你想要跟t进出某个集合的元素的LӞBag是非常有用的。由于Bag本n是一个接口,所以实际用的应该是实C该接口的c,例如HashBag或TreeBag--从这些类的名字也可以看出QHashBag实现的是一个HashMap的BagQ而TreeBag实现的是TreeMap的Bag。Bag接口中两个最重要的方法是QgetCount(Object o)Q用来返回Bag里面特定对象的出现次敎ͼuniqueSet()Q返回所有唯一元素?
Buffer接口允许按照预定义的ơ序删除集合中的对象Q删除次序可以是LIFOQLast In First OutQ后q先出)Q或FIFOQFirst In First OutQ先q先出)Q另外还可以是自定义的次序。下面来看看如何实现一个BufferQ按照自然次序删除元素?
BinaryHeapcdCBuffer接口Q能够按照自然次序删除元素。如果要颠倒次序,则必M入一个falseQ告诉Heap采用自然ơ序的逆序?/p>
FastArrayList、FastHashMap和FastTreeMapc能够按照两U模式操作,越了与它们对应的标准Collection。第一U模式是"慢模?Q类的修Ҏ作(d、删除元素)是同步的。与此相对,另一U模式是"快模?Q对q些cȝ讉K假定为只L作,因此不需要同步,速度较快。在快模式中Q结构性的改动通过下列方式完成Q首先克隆现有的c,修改克隆得到的类Q最后用克隆得到的类替换原有的类。FastArrayList、FastHashMap和FastTreeMapcȝ别适合于那U初始化之后大部分操作都是只L作的多线E环境?
iterators包ؓ各种集合和对象提供标准Java Collection包没有提供的q代器。本文的CZ应用C了ArrayIteratorQ通过q代方式讉KArray的内宏Viterators包里面各UP代器的用法基本上与标准Javaq代器一栗?
最后,comparators包提供了一些实用的比较W。所谓比较符其实也是一个类Q它定义的是如何比较两个属于同一cȝ对象Q决定它们的排序ơ序。例如,在前面提到的BuffercMQ我们可以定义自q比较W,用自定义的比较符来决定元素的排序ơ序Q而不是采用元素的自然排序ơ序。下面来看看具体的实现经q?
三、Webc?/b>
Webcȝlg用来执行与Web相关的Q务?
3.1 FileUpload
?概况Q一个可以直接用的文g上蝲lg?
?官方资源Q?a target="_blank">主页。由于这个组件尚未正式发布,今年二月发布的Beta版又有许多BUGQ所以徏议从nightly builds下蝲最新的版本?
?何时适用Q当你想要在Java服务器环境中加入一个易用、高性能的文件上载组件之时?
?CZ应用Qfileuploaddemo.jspQfileuploaddemo.htmQ和msg.jsp。要求服务器端应用目录的WEB-INF/lib下面有commons-fileupload-1.0-dev.jar?
?说明Q?
FileUploadlg解决了常见的文g上蝲问题。它提供了一个易用的接口来管理上载到服务器的文gQ可用于JSP和Servlet之中。FileUploadlg遵从RFC1867Q它分析输入hQ向应用E序提供一pd上蝲到服务器的文件。上载的文g可以保留在内存中Q也可以攑օ一个时位|(允许配置一个表C文件大的参数Q如果上载的文g过了该参数指定的大,则把文g写入一个时位|)。另外还有一些参数可供配|,包括可接受的最大文件、时文件的位置{?
下面介绍一下用FileUploadlg的步骤?
首先创徏一个HTML面。注意,凡是要上载文件的表单都必设|enctype属性,且属性的值必Lmultipart/form-dataQ同时请求方法必LPOST。下面的表单除了上蝲两个文gQ另外还有一个普通的文本输入框:
q有其他一些略有变化的使用形式Q正如前面所指出的,我们可以在上载的文g上打开一个输入流Q或者让它们ȝ在内存中直至I间占用辑ֈ一定的限制|或者在判断文gcd的基上,以String或Byte数组的Ş式获取其内容Q或者直接删除文件。这一切都只要使用FileItemcL供的Ҏ可以方便地做到QDefaultFileItem是FileItem的一个实玎ͼ?
一、概q?/b>
可重用性是Jakarta Commons目的灵所在。这些包在设计阶D就已经考虑了可重用性问题。其中一些包Q例如Commons里面用来记录日志的Logging包,最初是为其他项目设计的Q例如Jakarta Struts目Q当Z发现q些包对于其他项目也非常有用Q能够极大地帮助其他目的开发,他们军_些包构造一?公共"的存放位|,q就是Jakarta Commons目?
Z真正提高可重用性,每一个包都必M依赖于其他大型的框架或项目。因此,Commons目的包基本上都是独立的Q不仅是相对于其他项目的独立Q而且相对于Commons内部的大部分其他包独立。虽然存在一些例外的情况Q例如Betwixt包要用到XML APIQ但l大部分只用最基本的APIQ其主要目的是要能够通过单的接口方便地调用?
不过׃崇尚z,许多包的文档变得q于陋,~Zl护和支持,甚至有一部分q有错误的链接,文档也少得可怜。大部分的包需要我们自己去扑և其用法,甚至有时q需要我们自己去分析光用场合。本文将逐一介绍q些包,希望能够帮助你迅速掌握这一U篏了许多h心血的免费代码库?
说明QJakarta Commons?a target="_blank">Apache Commons是不同的Q后者是Apache Software Foundation的一个顶层项目,前者则是Jakarta目的一个子目Q同是也是本文要讨论的主角。本文后面凡是提到Commons的地斚w是指Jakarta的Commons?
Z便于说明Q本文把Commons目十八个成品的组Ӟ排除了EL、Latka和JexlQ分?c,如下表所C?
必须指出的是Q这U分cd是ؓ了方便文章说明,Commons目里面实际上ƈ不存在这U分c,同时q些分类的边界有时也存在一定的重叠?
本文首先介绍Web相关cd其他c里面的lgQ下一文章将涉及XML相关、包装这两类Q最后一文章专门介l属于工L的包?
二、其他类
CLI、Discovery、Lang和Collections包归入其他类Q这是因为它们都各自针对某个明确、实用的目标,可谓专而精?
2.1 CLI
?概况QCLI即Command Line InterfaceQ也是"命o行接?Q它为JavaE序讉K和解析命令行参数提供了一U统一的接口?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>
?何时适用Q当你需要以一U一致的、统一的方式访问命令行参数之时?
?CZ应用QCLIDemo.java。CLASSPATH中必d含commons-cli-1.0.jar?
?说明Q?
有多次你不得不Z个新的应用程序重新设计新的命令行参数处理方式Q如果能够只用某个单一的接口,l一完成诸如定义输入参数Q是否ؓ强制参数Q数D是字W串Q等{)、根据一pd规则分析参数、确定应用要采用的\径等dQ那该多好!{案在CLI?
在CLI中,每一个想要在命o中指定的参数都是一个Option对象。首先创Z个Options对象Q将各个Option对象加入Options对象Q然后利用CLI提供的方法来解析用户的输入参数。Option对象可以要求用户必须输入某个参数Q例如必d命o行提供文件名字。如果某个参数是必须的,创徏Option对象的时候就要显式地指定?
下面是用CLI的步骤?
2.2 Discovery
?概况QDiscoverylg是发现模式(Discovery PatternQ的一个实玎ͼ它的目标是按照一U统一的方式定位和实例化类以及其他资源?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>?
?何时适用Q当你想用最佳的法在JavaE序中查找Java接口的各U实C时?
?应用实例QDiscoveryDemo.javaQMyInterface.javaQMyImpl1.javaQMyImpl2.javaQMyInterface。要求CLASSPATH中必d含commons-discovery.jar和commons-logging.jar?
?说明Q?
Discovery的意思就?发现"Q它试图用最佳的法查找某个接口的所有已知的实现。在使用服务的场合,当我们想要查找某个服务的所有已知的提供者时QDiscoverylg其有用?
考虑一下这U情形:我们为某个特别复杂的d~写了一个接口,所有该接口的实现都用各不相同的方式来完成这个复杂Q务,最l用户可以根据需要来选择完成d的具体方式。那么,在这U情形下Q最l用户应该用什么办法来扑և接口的所有可用实玎ͼ卛_能的完成d的方式)呢?
上面描述的情形就是所谓的服务-服务提供者体pR服务的功能由接口描qͼ服务提供者则提供具体的实现。现在的问题是最l用戯用某U办法来Lpȝ中已l安装了哪些服务提供者。在q种情Ş下,Discoverylg很有用了,它不仅可以用来查N些实C特定接口的类Q而且q可以用来查找资源,例如囄或其他文件等。在执行q些操作ӞDiscovery遵从Sun的服务提供者体pL定义的规则?
׃q个原因Q用Discoverylg实带来许多方便。请读者参阅本文后面示例程序中的接口MyInterface.java和两个实现类MyImpl1.java、MyImple2.javaQ了解下面例子的l节。在使用Discovery的时候要提供MyInterface文gQ把它放入META-INF/services目录Q注意该文g的名字对应接口的完整限定名称QFully Qualified NameQ,如果接口属于某个包,该文件的名字也必ȝ应地改变?br />
当然Q实现类的注册办法ƈ非只有这么一U,否则的话Discovery的实用性就要大打折扣了Q实际上Q按照Discovery内部的类查找机制Q按照这U方法注册的cd是Discovery最后找到的cR另一U常用的注册Ҏ是通过pȝ属性或用户定义的属性来传递实现类的名字,例如Q放弃META-INF/services目录下的文gQ改为执行java -DMyInterface=MyImpl1 DiscoveryDemo命o来运行示例程序,q里的系l属性是接口的名字,值是该接口的提供者,q行的结果是完全一L?
Discoveryq可以用来创建服务提供者的(singleton)实例q调用其ҎQ语法如下:((MyInterface)discover.newInstance(MyInterface.class)).myMethod();。注意在q个例子中,我们q不知道到底哪一个服务提供者实CmyMethodQ甚x们根本不必关心这一炏V具体的情Ş与运行这D代码的方式以及q行环境中已l注册了什么服务提供者有养I在不同的环境下运行,实际得到的服务提供者可能不同?
2.3 Lang
?概况QLang是java.lang的一个扩展包Q增加了许多操作String的功能,另外q支持C风格的枚N?
?官方资源Q?a target="_blank">主页Q?a target="_blank">二进?/a>Q?a target="_blank">源代?/a>?
?何时适用Q当java.lang包提供的Ҏ未能满需要,惌更多的功能来处理String、数值和System属性时Q还有,当你惌使用C风格的枚N时?
?CZ应用QLangDemo.javaQMortgage.javaQOnTV.java。CLASSPATH中必d含commons-lang.jar?
?说明Q?
q个包提供了许多Z方便目的而提供的ҎQ它们中的大多数是静态的Q简化了日常~码工作。StringUtilscL其中的一个代表,它得开发者能够超标准的java.lang.String包来处理字符丌Ӏ用这些方法很单,通常只要在调用静态方法时提供适当的参数就可以了。例如,如果要将某个单词的首字符改ؓ大写Q只需调用QStringUtils.capitalise("name")Q调用的输出l果是Name。请览StringUtils API文档了解其他静态方法,也许你会扑ֈ一些可以直接拿来用的代码。本文提供的CZE序C了其中一些方法的使用?
另一个值得注意的类是RandomStringUtilsQ它提供了生成随机字W串的方法,用来创徏随机密码实在太方便了?
NumberUtilscL供了处理数值数据的ҎQ许多方法值得一用,例如L最大、最数的方法,String转换成数值的ҎQ等{。NumberRange和CharRangecd别提供了创徏和操作数D围、字W范围的Ҏ?
Builder包里的类提供了一些特D的ҎQ可用来构造类的toString、hashCode、compareTo和equalsҎQ其基本思\是构造出cȝ高质量的toString、hashCode、compareTo和equalsҎQ从而免M用户自己定义q些Ҏ之劳Q只要调用一下Builder包里面的Ҏ可以了。例如,我们可以用ToStringBuilder来构造出cȝtoString描述Q如下例所C: