??xml version="1.0" encoding="utf-8" standalone="yes"?>亚洲小说区图片区另类春色,久久夜色精品国产噜噜噜亚洲AV ,亚洲?V无码乱码国产精品http://www.tkk7.com/xixidabao/category/24795.htmlGROW WITH JAVAzh-cnFri, 31 Aug 2007 03:14:31 GMTFri, 31 Aug 2007 03:14:31 GMT60Hibernate实践 http://www.tkk7.com/xixidabao/archive/2007/08/29/140929.htmlJAVA之\JAVA之\Wed, 29 Aug 2007 07:18:00 GMThttp://www.tkk7.com/xixidabao/archive/2007/08/29/140929.html阅读全文

JAVA之\ 2007-08-29 15:18 发表评论
]]>
XXXX目~存Ҏȝhttp://www.tkk7.com/xixidabao/archive/2007/08/28/140497.htmlJAVA之\JAVA之\Tue, 28 Aug 2007 06:35:00 GMThttp://www.tkk7.com/xixidabao/archive/2007/08/28/140497.html?http://dev.csdn.net/author/bromon/8737868d27c74af7b4dbfc4d0497f4fa.html



XXXX目~存Ҏȝ
 
       XXXX目是目前在实际工作中正在做的事情,该项目是一个大型系l的内容理内核Q负责最核心的meta data的集中管理,性能有较高的要求Q设计初期就要求能够支持cluster。项目用hibernate 3.2Q针对开发过E中对于各种~存的不同看法,撰写了本文。重点在于澄清一些hibernate的缓存细节,U正一些错误的~存用法?/div>
 
一?/span>hibernate的二U缓?/div>
       如果开启了二~存Qhibernate在执行Q何一ơ查询的之后Q都会把得到的结果集攑ֈ~存中,~存l构可以看作是一个hash tableQkey是数据库记录的idQvalue是id对应的pojo对象。当用户Ҏid查询对象的时候(load、iteratorҎQ,会首先在~存中查找,如果没有扑ֈ再发h据库查询。但是如果用hql发v查询Qfind, queryҎQ则不会利用二~存Q而是直接从数据库获得数据Q但是它会把得到的数据放CU缓存备用。也是_Zhql的查询,对二U缓存是只写不读的?/div>
 
       针对二~存的工作原理,采用iterator取代list来提高二U缓存命中率的想法是不可行的。Iterator的工作方式是Ҏ索条件从数据库中选取所有目标数据的idQ然后用q些id一个一个的CU缓存里面做索,如果扑ֈq接加载,找不到就向数据库做查询。因此假如iterator?00条数据的话,最好情冉|100%全部命中Q最坏情冉|0%命中Q执?01条sql把所有数据选出来。而list虽然不利用缓存,但是它只会发?条sql取得所有数据。在合理利用分页查询的情况下Qlist整体效率高于iterator?/div>
 
       二~存的失效机制由hibernate控制Q当某条数据被修改之后,hibernate会根据它的idd~存失效操作。基于此机制Q如果数据表不是被hibernate独占Q比如同时用jdbc或者ado{)Q那么二U缓存无法得到有效控制?/div>
 
       ׃hibernate的缓存接口很灉|Qcache provider可以方便的切换,因此支持cluster环境不是大问题,通过使用swarmcache、jboss cache{支持分布式的缓存方案,可以实现。但是问题在?
1?nbsp;分布式缓存本w成本偏高(比如使用同步复制模式的jboss cacheQ?/div>
2?nbsp;分布式环境通常对事务控制有较高要求Q而目前的开源缓存方案对事务~存Qtransaction cacheQ支持得不够好。当jta事务发生会滚Q缓存的最后更新结果很N料。这一点会带来很大的部|成本,甚至得不偿失?/div>
 
l论QXXXX不应把hibernate二~存作ؓ优化的主要手D,一般情况下不要使用?/div>
 
原因如下Q?/div>
1?nbsp;XXXX的DAOcd部分是从1.0升q来Q由?.0采用的是hibernate 2.1Q所以在扚w删除数据的时候采用了native sql的方式。虽然XXXX2.0已经完全升到hibernate 3.2Q支持hibernate原生的批量删改,但是׃hibernate扚w操作的性能不如sqlQ而且Z兼容1.0的daoc,所以很多地方保留了sql操作。哪些数据表是单U被hibernate独占无法l计Q而且随着来业务的发展可能会有很大变数。因此不宜采用二U缓存?/div>
2?nbsp;针对pȝ业务来说Q基于id索的二~存命中率极为有限,hql被大量采用,二~存Ҏ能的提升很有限?/div>
3?nbsp;hibernate 3.0在做扚w修改、批量更新的时候,是不会同步更CU缓存的Q该问题在hibernate 3.2中是否仍然存在尚不确定?/div>
 
 
二?/span>hibernate的查询缓?/div>
 
       查询~存的实现机制与二~存基本一_最大的差异在于攑օ~存中的key是查询的语句Qvalue是查询之后得到的l果集的id列表。表面看来这LҎg能解决hql利用~存的问题,但是需要注意的是,构成key的是Qhql生成的sql、sql的参数、排序、分信息等。也是说如果你的hql有小的差异Q比如第一条hql?-50条数据,W二条hql?0-60条数据,那么hibernate会认是两个完全不同的keyQ无法重复利用缓存。因此利用率也不高?/div>
 
       另外一个需要注意的问题是,查询~存和二U缓存是有关联关pȝQ他们不是完全独立的两套东西。假如一个查询条件hql_1Q第一ơ被执行的时候,它会从数据库取得数据Q然后把查询条g作ؓkeyQ把q回数据的所有id列表作ؓvalueQ请注意仅仅是idQ放到查询缓存中Q同时整个结果集攑ֈclass~存Q也是二~存Q,key是idQvalue是pojo对象。当你再ơ执行hql_1Q它会从~存中得到id列表Q然后根据这些列表一个一个的到class~存里面Lpojo对象Q如果找不到向数据库发h询。也是_如果二~存配置了超时时_或者发呆时_Q就有可能出现查询缓存命中了Q获得了id列表Q但是class里面相应的pojo已经因ؓ时(或发?被失效,hibernate׃Ҏid清单Q一个一个的d数据库查询,有多个idQ就执行多少个sql。该情况导致性能下降严重?/div>
 
       查询~存的失效机制也由hibernate控制Q数据进入缓存时会有一个timestampQ它和数据表的timestamp对应。当hibernate环境内发生save、update{操作时Q会更新被操作数据表的timestamp。用户在获取~存的时候,一旦命中就会检查它的timestamp是否和数据表的timestamp匚wQ如果不Q缓存会被失效。因此查询缓存的失效控制是以数据表ؓ_度的,只要数据表中M一条记录发生一点修改,整个表相关的所有查询缓存就都无效了。因此查询缓存的命中率可能会很低?/div>
 
l论QXXXX不应把hibernate二~存作ؓ优化的主要手D,一般情况下不要使用?/div>
 
原因如下Q?/div>
1?nbsp;XXXX的上层业务中索条仉比较复杂Q尤其是涉及多表操作的地斏V很出现重复执行一个排序、分c参C致的查询Q因此命中率很难提高?/div>
2?nbsp;查询~存必须配合二~存一起用,否则极易出现1+N的情况,否则性能不升反降
3?nbsp;使用查询~存必须在执行查询之前显C用Query.setCacheable(true)才能Ȁzȝ存,q势必会对已有的hibernate装cd来问题?/div>
 
 
 
ȝ
       详细分析hibernate的二U缓存和查询~存之后Q针对XXXX目的具体情况做出结论,在底层用通用~存Ҏ的想法基本上是不可取的。比较好的做法是在高层次中(业务逻辑层面Q,针对具体的业务逻辑状况手动使用数据~存Q不仅可以完全控制缓存的生命周期Q还可以针对业务具体调整~存Ҏ提交命中率。Cluster中的~存同步可以完全交给~存本n的同步机制来完成。比如开源缓存swarmcache采用invalidate的机Ӟ可以Ҏ用户指定的策略,在需要的时候向|络中的其他swarmcache节点发送失效消息,q一机制和XXXX1.0中已l采用的MappingCache的同步方案基本一致。徏议采用?/div>

JAVA之\ 2007-08-28 14:35 发表评论
]]>Hibernate二~存ȝhttp://www.tkk7.com/xixidabao/archive/2007/04/08/109196.htmlJAVA之\JAVA之\Sun, 08 Apr 2007 02:04:00 GMThttp://www.tkk7.com/xixidabao/archive/2007/04/08/109196.html

Hibernate二~存ȝ

2006-10-27 15:59 作者: AreYouOK 出处Q?JAVAEYE 责Q~辑Q?a title=向本~辑提问 target=_blank>方舟


       很多人对二~存都不太了解,或者是有错误的认识Q我一直想写一文章介l一下hibernate的二U缓存的Q今天终于忍不住了?

  我的l验主要来自hibernate2.1版本Q基本原理和3.0?.1是一LQ请原谅我的固不化?

  hibernate的session提供了一U缓存,每个sessionQ对同一个idq行两次loadQ不会发送两条sqll数据库Q但是session关闭的时候,一U缓存就失效了?br>
  二~存是SessionFactoryU别的全局~存Q它底下可以使用不同的缓存类库,比如ehcache、oscache{,需要设|hibernate.cache.provider_classQ我们这里用ehcacheQ在2.1中就?hibernate.cache.provider_class=net.sf.hibernate.cache.EhCacheProvider如果使用查询~存Q加上hibernate.cache.use_query_cache=true

  ~存可以单的看成一个MapQ通过key在缓存里面找value?br>
  Class的缓?/strong>

  对于一条记录,也就是一个PO来说Q是ҎID来找的,~存的key是IDQvalue是POJO。无论listQloadq是iterateQ只要读Z个对象,都会填充~存。但是list不会使用~存Q而iterate会先取数据库select id出来Q然后一个id一个id的loadQ如果在~存里面有,׃~存取,没有的话去数据库load。假设是d~存Q需要设|:

Qcache usage="read-write"/Q?

  如果你用的二~存实现是ehcache的话Q需要配|ehcache.xml

Qcache name="com.xxx.pojo.Foo" maxElementsInMemory="500" eternal="false" timeToLiveSeconds="7200" timeToIdleSeconds="3600" overflowToDisk="true" /Q?

  其中eternal表示~存是不是永q不时QtimeToLiveSeconds是缓存中每个元素Q这里也是一个POJOQ的时旉Q如果eternal="false"Q超q指定的旉Q这个元素就被移C。timeToIdleSeconds是发呆时_是可选的。当往~存里面put的元素超q?00个时Q如果overflowToDisk="true"Q就会把~存中的部分数据保存在硬盘上的时文仉面?

  每个需要缓存的class都要q样配置。如果你没有配置Qhibernate会在启动的时候警告你Q然后用defaultCache的配|,q样多个class会共享一个配|?

  当某个ID通过hibernate修改Ӟhibernate会知道,于是U除~存?

  q样大家可能会想Q同L查询条gQ第一ơ先listQ第二次再iterateQ就可以使用到缓存了。实际上q是很难的,因ؓ你无法判断什么时候是W一ơ,而且每次查询的条仉常是不一LQ假如数据库里面?00条记录,id??00Q第一ơlist的时候出了前50个idQ第二次iterate的时候却查询?0?0号idQ那?0-50是从~存里面取的Q?1?0是从数据库取的,共发?+20条sql。所以我一直认为iterate没有什么用QL会有1+N的问题?

  Q题外话Q有说法说大型查询用list会把整个l果集装入内存,很慢Q而iterate只select id比较好,但是大型查询L要分|的,谁也不会真的把整个结果集装进来,假如一?0条的话,iterate共需要执?1条语句,list虽然选择若干字段Q比iterateW一条select id语句慢一些,但只有一条语句,不装入整个结果集hibernateq会Ҏ数据库方a做优化,比如使用mysql的limitQ整体看来应该还是list快。)

  如果惌对list或者iterate查询的结果缓存,p用到查询~存?br>
  查询~存

  首先需要配|hibernate.cache.use_query_cache=true

  如果用ehcacheQ配|ehcache.xmlQ注意hibernate3.0以后不是net.sf的包名了Q?br>
Qcache name="net.sf.hibernate.cache.StandardQueryCache"
maxElementsInMemory="50" eternal="false" timeToIdleSeconds="3600"
timeToLiveSeconds="7200" overflowToDisk="true"/Q?
Qcache name="net.sf.hibernate.cache.UpdateTimestampsCache"
maxElementsInMemory="5000" eternal="true" overflowToDisk="true"/Q?

  然后

query.setCacheable(true);//ȀzL询缓?
query.setCacheRegion("myCacheRegion");//指定要用的cacheRegionQ可?

  W二行指定要使用的cacheRegion是myCacheRegionQ即你可以给每个查询~存做一个单独的配置Q用setCacheRegion来做q个指定Q需要在ehcache.xml里面配置它:

Qcache name="myCacheRegion" maxElementsInMemory="10" eternal="false" timeToIdleSeconds="3600" timeToLiveSeconds="7200" overflowToDisk="true" /Q?

  如果省略W二行,不设|cacheRegion的话Q那么会使用上面提到的标准查询缓存的配置Q也是Qnet.sf.hibernate.cache.StandardQueryCache

  对于查询~存来说Q缓存的key是根据hql生成的sqlQ再加上参数Q分늭信息Q可以通过日志输出看到Q不q它的输Z是很可读Q最好改一下它的代码)?

  比如hqlQ?

from Cat c where c.name like ?

  生成大致如下的sqlQ?

select * from cat c where c.name like ?

  参数?tiger%"Q那么查询缓存的key*大约*是这L字符Ԍ我是凭记忆写的,q不_Q不q看了也该明白了Q:

select * from cat c where c.name like ? , parameter:tiger%

  q样Q保证了同样的查询、同L参数{条件下h一Lkey?

  现在说说~存的valueQ如果是list方式的话Qvalue在这里ƈ不是整个l果集,而是查询出来的这一串ID。也是_不管是listҎq是iterateҎQ第一ơ查询的时候,它们的查询方式很它们qx的方式是一LQlist执行一条sqlQiterate执行1+N条,多出来的行ؓ是它们填充了~存。但是到同样条gW二ơ查询的时候,都和iterate的行Z样了Q根据缓存的keyȝ存里面查CvalueQvalue是一串idQ然后在到class的缓存里面去一个一个的load出来。这样做是ؓ了节U内存?

  可以看出来,查询~存需要打开相关cȝclass~存。list和iterateҎW一ơ执行的时候,都是既填充查询缓存又填充class~存的?
q里q有一个很Ҏ被忽视的重要问题Q即打开查询~存以后Q即使是listҎ也可能遇?+N的问题!相同条gW一ơlist的时候,因ؓ查询~存中找不到Q不class~存是否存在数据QL发送一条sql语句到数据库获取全部数据Q然后填充查询缓存和class~存。但是第二次执行的时候,问题来了,如果你的class~存的超时时间比较短Q现在class~存都超时了Q但是查询缓存还在,那么listҎ在获取id串以后,会一个一个去数据库loadQ因此,class~存的超时时间一定不能短于查询缓存设|的时旉Q如果还讄了发呆时间的话,保证class~存的发呆时间也大于查询的缓存的生存旉。这里还有其他情况,比如class~存被程序强制evict了,q种情况p自己注意了?br>
  另外Q如果hql查询包含select字句Q那么查询缓存里面的value是整个l果集了?br>
  当hibernate更新数据库的时候,它怎么知道更新哪些查询~存呢?

  hibernate在一个地方维护每个表的最后更新时_其实也就是放在上面net.sf.hibernate.cache.UpdateTimestampsCache所指定的缓存配|里面?

  当通过hibernate更新的时候,hibernate会知道这ơ更新媄响了哪些表。然后它更新q些表的最后更新时间。每个缓存都有一个生成时间和q个~存所查询的表Q当hibernate查询一个缓存是否存在的时候,如果~存存在Q它q要取出~存的生成时间和q个~存所查询的表Q然后去查找q些表的最后更新时_如果有一个表在生成时间后更新q了Q那么这个缓存是无效的?

  可以看出Q只要更新过一个表Q那么凡是涉及到q个表的查询~存失效了Q因此查询缓存的命中率可能会比较低?br>
Collection~存

  需要在hbm的collection里面讄Q?br>
Qcache usage="read-write"/Q?

  假如class是CatQcollection叫childrenQ那么ehcache里面配置

Qcache name="com.xxx.pojo.Cat.children"
maxElementsInMemory="20" eternal="false" timeToIdleSeconds="3600" timeToLiveSeconds="7200"
overflowToDisk="true" /Q?

  Collection的缓存和前面查询~存的list一P也是只保持一串idQ但它不会因个表更新q就失效Q一个collection~存仅在q个collection里面的元素有增删时才失效?

  q样有一个问题,如果你的collection是根据某个字D|序的Q当其中一个元素更C该字D|Q导致顺序改变时Qcollection~存里面的顺序没有做更新?br>
  ~存{略

  只读~存Qread-onlyQ:没有什么好说的

  ?写缓存(read-writeQ?E序可能要的更新数据

  不严格的?写缓存(nonstrict-read-writeQ:需要更新数据,但是两个事务更新同一条记录的可能性很,性能比读写缓存好
事务~存QtransactionalQ:~存支持事务Q发生异常的时候,~存也能够回滚,只支持jta环境Q这个我没有怎么研究q?br>
  d~存和不严格d~存在实C的区别在于,d~存更新~存的时候会把缓存里面的数据换成一个锁Q其他事务如果去取相应的~存数据Q发现被锁住了,然后q接取数据库查询?

  在hibernate2.1的ehcache实现中,如果锁住部分~存的事务发生了异常Q那么缓存会一直被锁住Q直?0U后时?

  不严D写缓存不锁定~存中的数据?使用二~存的前|条件?br>
  你的hibernateE序Ҏ据库有独占的写访问权Q其他的q程更新了数据库Qhibernate是不可能知道的。你操作数据库必需直接通过hibernateQ如果你调用存储q程Q或者自׃用jdbc更新数据库,hibernate也是不知道的。hibernate3.0的大扚w更新和删除是不更CU缓存的Q但是据?.1已经解决了这个问题?

  q个限制相当的棘手,有时候hibernate做批量更新、删除很慢,但是你却不能自己写jdbc来优化,很郁闷吧?

  SessionFactory也提供了U除~存的方法,你一定要自己写一些JDBC的话Q可以调用这些方法移除缓存,q些Ҏ是:

void evict(Class persistentClass)
Evict all entries from the second-level cache.
void evict(Class persistentClass, Serializable id)
Evict an entry from the second-level cache.
void evictCollection(String roleName)
Evict all entries from the second-level cache.
void evictCollection(String roleName, Serializable id)
Evict an entry from the second-level cache.
void evictQueries()
Evict any query result sets cached in the default query cache region.
void evictQueries(String cacheRegion)
Evict any query result sets cached in the named query cache region.

  不过我不q样做,因ؓq样很难l护。比如你现在用JDBC扚w更新了某个表Q有3个查询缓存会用到q个表,用evictQueries(String cacheRegion)U除?个查询缓存,然后用evict(Class persistentClass)U除了class~存Q看上去好像完整了。不q哪天你d了一个相x询缓存,可能会忘记更新这里的U除代码。如果你的jdbc代码到处都是Q在你添加一个查询缓存的时候,q知道其他什么地方也要做相应的改动吗Q?br>
  ȝQ?

  不要惛_然的以ؓ~存一定能提高性能Q仅仅在你能够驾驭它q且条g合适的情况下才是这L。hibernate的二U缓存限制还是比较多的,不方便用jdbc可能会大大的降低更新性能。在不了解原理的情况下ؕ用,可能会有1+N的问题。不当的使用q可能导致读数据?
如果受不了hibernate的诸多限Ӟ那么q是自己在应用程序的层面上做~存吧?

  在越高的层面上做~存Q效果就会越好。就好像管盘有缓存,数据库还是要实现自己的缓存,管数据库有~存Q咱们的应用E序q是要做~存。因为底层的~存它ƈ不知道高层要用这些数据干什么,只能做的比较通用Q而高层可以有针对性的实现~存Q所以在更高的别上做缓存,效果也要好些吧?br>

JAVA之\ 2007-04-08 10:04 发表评论
]]>Hibernate 的原理与配置快速入?/title><link>http://www.tkk7.com/xixidabao/archive/2006/05/17/46596.html</link><dc:creator>JAVA之\</dc:creator><author>JAVA之\</author><pubDate>Wed, 17 May 2006 03:46:00 GMT</pubDate><guid>http://www.tkk7.com/xixidabao/archive/2006/05/17/46596.html</guid><description><![CDATA[  也许你听说过Hibernate的大名,但可能一直不了解它,也许你一直望用它q行开发,那么本文正是你所需要的Q在本文中,我向大家重点介绍Hibernate的核心API调用库,q讲解一下它的基本配|?br><br>  看完本文后,我相信你对什么是ORMQ对?关系映射Q以及它的优点会有一个深ȝ认识Q我们先通过一个简单的例子开始来展现它的威力?br><br>  正如一些传l的l典计算机文章大都会通过一?#8220;hello,world”的例子开始讲解一P我们也不例外Q我们也从一个相对简单的例子来阐qHibernate的开发方法,但如果要真正阐述Hibernate的一些重要思想Q仅仅靠在屏q上打印一些字W是q远不够的,在我们的CZE序中,我们创Z些对象,q将其保存在数据库中Q然后对它们q行更新和查询?br><br> <table borderColor=#ffcc00 cellSpacing=4 width="90%" align=center border=1> <tbody> <tr> <td colSpan=2> <div align=center><font color=#ff0000><strong>阅读D</strong></font></div> </td> </tr> <tr> <td width="35%"><a target=_blank><font color=#0000ff>“Hello World”</font></a></td> <td width="65%">“Hello world”CZE序让您对Hibernate有一个简单的认识<strong>?/strong></td> </tr> <tr> <td><a target=_blank><font color=#0000ff>理解Hibernate的架?/font></a></td> <td>介绍Hibernate接口的主要功能?/td> </tr> <tr> <td><a target=_blank><font color=#0000ff>核心接口</font></a></td> <td>Hibernate?个核心接口,通过q几个接口开发h员可以存储和获得持久对象Qƈ且能够进行事务控?/td> </tr> <tr> <td><a target=_blank><font color=#0000ff>一个重要的术语QType</font></a></td> <td>Type是Hibernate发明者发明的一个术语,它在整个构架中是一个非常基、有着强大功能的元素,一个Type对象能将一个Javacd映射到数据库中一个表的字D中厅R?/td> </tr> <tr> <td><a target=_blank><font color=#0000ff>{略接口</font></a></td> <td>Hibernate与某些其它开源Y件不同的q有一点――高度的可扩展性,q通过它的内置{略机制来实现?/td> </tr> <tr> <td><a target=_blank><font color=#0000ff>基础配置</font></a></td> <td>Hibernate可以配置成可在Q何Java环境中运行,一般说来,它通常被用?Q?层的C/S模式的项目中Qƈ被部|在服务端?/td> </tr> <tr> <td height=36><a target=_blank><font color=#0000ff>创徏一个SessionFactory对象</font></a></td> <td>要创Z个SessionFactory对象Q必dHibernate初始化时创徏一个Configurationcȝ实例Qƈ已写好的映文件交由它处理?/td> </tr> </tbody> <strong>“Hello World”</strong><br><br>  Hibernate应用E序定义了一些持久类Qƈ且定义了q些cM数据库表格的映射关系。在我们q个“Hello world”CZE序中包含了一个类和一个映文件。让我们看看q个单的持久cd含有一些什么?映射文g是怎样定义的?另外Q我们该怎样用Hibernate来操作这个持久类?br><br>  我们q个单示例程序的目的是将一些持久类存储在数据库中,然后从数据库取出来,q将其信息正文显C给用户。其中Message正是一个简单的持久c:Q它包含我们要显C的信息Q其源代码如下:<br><br>  列表1 Message.Java 一个简单的持久c?br><br> </table> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>package hello;<br>public class Message {<br> private Long id;<br> private String text;<br> private Message nextMessage;<br> private Message() {}<br> public Message(String text) {<br>  this.text = text;<br> }<br> public Long getId() {<br>  return id;<br> }<br> private void setId(Long id) {<br>  this.id = id;<br> }<br> public String getText() {<br>  return text; <br> }<br> public void setText(String text) {<br>  this.text = text;<br> }<br> public Message getNextMessage() {<br>  return nextMessage;<br> }<br> public void setNextMessage(Message nextMessage) {<br>  this.nextMessage = nextMessage;<br> }<br>} </td> </tr> </tbody> </table> <br>  MessagecL三个属性:Message的id 、消息正文、以及一个指向下一条消息的指针。其中id属性让我们的应用程序能够唯一的识别这条消息,通常它等同于数据库中的主键,如果多个Messagecȝ实例对象拥有相同的idQ那它们代表数据库某个表的同一个记录。在q里我们选择了长整型作ؓ我们的id|但这不是必需的。Hibernate允许我们使用L的类型来作ؓ对象的id|在后面我们会Ҏ作详l描q?br><br>  你可能注意到Messagecȝ代码cM于JavaBean的代码风|q且它有一个没有参数的构造函敎ͼ在我们以后的代码中我l用这U风格来~写持久cȝ代码?br><br>  Hibernate会自动管理Messagecȝ实例Qƈ通过内部机制使其持久化,但实际上Message对象q没有实CQ何关于Hibernate的类或接口,因此我们也可以将它作Z个普通的JavacL使用Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Message message = new Message("Hello World");<br>System.out.println( message.getText() ); </td> </tr> </tbody> </table> <br>  以上q段代码正是我们所期望的结果:它打?#8220;hello world”到屏q上。但qƈ不是我们的最l目标;实际上Hibernate与诸如EJB容器q样的环境在持久层实现的方式上有很大的不同。我们的持久c?Messagec?可以用在与容器无关的环境中,不像EJB必须要有EJB容器才能执行。ؓ了能更清楚地表现q点Q以下代码将我们的一个新消息保存到数据库中去Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Session session = getSessionFactory().openSession();<br>Transaction tx = session.beginTransaction();<br>Message message = new Message("Hello World");<br>session.save(message);<br>tx.commit();<br>session.close(); </td> </tr> </tbody> </table> <br>  以上q段代码调用了Hibernate的Session和Transaction接口Q关于getSessionFactory()Ҏ我们会马上提到Q。它相当于我们执行了以下SQL语句Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>insert into MESSAGES (MESSAGE_ID, MESSAGE_TEXT, NEXT_MESSAGE_ID)<br>values (1, 'Hello World', null) </td> </tr> </tbody> </table> <br>  在以上的SQL语句中,MESSAGE_ID字段到底被初始化成了什么值呢Q由于我们ƈ没有在先前的代码中ؓmessage对象的id属性赋与初始|那它是否为null呢?实际上Hibernate对id属性作了特D处理:׃它是一个对象的唯一标识Q因此当我们q行save()调用ӞHibernate会ؓ它自动赋予一个唯一的|我们在后面内容中讲q它是如何生成这个值的Q?br><br>  我们假设你已l在数据库中创徏了一个名为MESSAGE的表Q那么既然前面这D代码让我们Message对象存入了数据库中,那么现在我们p它们一一取出来。下面这D代码将按照字母序Q将数据库中的所有Message对象取出来,q将它们的消息正文打印到屏幕上:<br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Session newSession = getSessionFactory().openSession();<br>Transaction newTransaction = newSession.beginTransaction();<br>List messages =newSession.find("from Message as m order by m.text asc");<br>System.out.println( messages.size() + " message(s) found:" );<br>for ( Iterator iter = messages.iterator(); iter.hasNext(); ) {<br> Message message = (Message) iter.next();<br> System.out.println( message.getText() );<br>}<br>newTransaction.commit();<br>newSession.close(); </td> </tr> </tbody> </table> <br>  在以上这D代码中Q你可能被find()Ҏ的这个参数困扰着Q?from Message as m order by m.text asc"Q其实它是Hibernate自己定义的查询语aQ全U叫Hibernate Query Language(HQL)。通俗地讲HQL与SQL的关pd不多是方言与普通话之间的关p,咋一看,你会觉得它有点类gSQL语句。其实在find()调用ӞHibernate会将q段HQL语言译成如下的SQL语句Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>select m.MESSAGE_ID, m.MESSAGE_TEXT, m.NEXT_MESSAGE_ID<br>from MESSAGES m<br>order by m.MESSAGE_TEXT asc </td> </tr> </tbody> </table> <br>  以下是q行l果Q?<br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>1 message(s) found:<br>Hello World </td> </tr> </tbody> </table> <br>  如果你以前没有ORMQ对象-关系映射Q的开发经验,那你可能惛_代码的某个地方去Lq段SQL语句Q但在Hibernate中你可能会失望:它根本不存在Q所有就SQL语句都是Hibernate动态生成的?<br><br>  也许你会觉得q缺点什么,对!仅凭以上代码Hibernate是无法将我们的MessagecL久化的。我们还需要一些更多的信息Q这是映射定义表!q个表在Hibernate中是以XML格式来体现的Q它定义了Messagecȝ属性是怎样与数据库中的MESSAGES表的字段q行一一对应的,列表2是这个示例程序的映射配置文g清单Q?br><br>  列表2Q示例程序的对象Q关pL表 <br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Q?xml version="1.0"?Q?br>Q?DOCTYPE hibernate-mapping PUBLIC<br>"-//Hibernate/Hibernate Mapping DTD//EN"<br>"http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd"Q?br>Qhibernate-mappingQ?br>Qclass name="hello.Message" table="MESSAGES"Q?br> Qid name="id" column="MESSAGE_ID"Q?br>  Qgenerator class="increment"/Q?br> Q?idQ?br> Qproperty name="text" column="MESSAGE_TEXT"/Q?br> Qmany-to-one name="nextMessage" cascade="all" column="NEXT_MESSAGE_ID"/Q?br>Q?classQ?br>Q?hibernate-mappingQ?</td> </tr> </tbody> </table> <br>  以上q个文档告诉Hibernate怎样MessagecL到MESSAGES表中Q其中Messagecȝid属性与表的MESSAGE_ID字段对应Qtext属性与表的MESSAGE_TEXT字段对应QnextMessage属性是一个多对一的关p,它与表中的NEXT_MESSAGE_ID相对应?<br><br>  相对于有些开源项目来_Hibernate的配|文件其实是很容易理解的。你可以LC改与l护它。只要你定义好了持久cM数据库中表字D늚对应关系p了,Hibernate会自动帮你生成SQL语句来对Message对象q行插入、更新、删除、查扑ַ作,你可以不写一句SQL语句Q甚至不需要懂得SQL语言Q?br><br>  现在让我们做一个新的试验,我们先取出第一个Message对象Q然后修改它的消息正文,最后我们再生成一个新的Message对象Qƈ它作ؓW一个Message对象的下一条消息,其代码如下:<br><br>  列表3 更新一条消?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Session session = getSessionFactory().openSession();<br>Transaction tx = session.beginTransaction();<br>// 1 is the generated id of the first message<br>Message message =(Message) session.load( Message.class, new Long(1) );<br>message.setText("Greetings Earthling");<br>Message nextMessage = new Message("Take me to your leader (please)");<br>message.setNextMessage( nextMessage );<br>tx.commit();<br>session.close(); </td> </tr> </tbody> </table> <br>  以上q段代码在调用时QHibernate内部自动生成如下的SQL语句Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>select m.MESSAGE_ID, m.MESSAGE_TEXT, m.NEXT_MESSAGE_ID<br>from MESSAGES m<br>where m.MESSAGE_ID = 1<br><br>insert into MESSAGES (MESSAGE_ID, MESSAGE_TEXT, NEXT_MESSAGE_ID)<br>values (2, 'Take me to your leader (please)', null)<br><br>update MESSAGES<br>set MESSAGE_TEXT = 'Greetings Earthling', NEXT_MESSAGE_ID = 2<br>where MESSAGE_ID = 1 </td> </tr> </tbody> </table> <br>  当第一个Message对象的text属性和nextMessage被程序修ҎQ请注意Hibernate是如何检到q种变化Qƈ如何在数据库中自动对它更新的。这实际上是Hibernate的一个很有h值的特色Q我们把它称?#8220;自动脏数据检?#8221;QHibernate的这个特色得当我们修改一个持久对象的属性后Q不必显式地通知Hibernated它在数据库中q行更新。同LQ当W一个Message对象调用setNextMessage()Ҏ第二个Message对象作ؓ它的下一条消息的引用ӞW二条消息会无需调用save()ҎQ便可以自动C存在数据库中。这U特色被UCؓ“U联保存”Q它也免M我们昑ּ地对W二个Message对象调用save()Ҏ之苦?br><br>  如果我们再运行先前的那段数据库中所有的Message对象都打印出来的代码Q那它的q行l果如下Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>2 message(s) found:<br>Greetings Earthling<br>Take me to your leader (please) </td> </tr> </tbody> </table> <br>  “Hello world”CZE序现在介绍完毕。我们ȝ对Hibernate有了一个简单的认识Q下面我们将回过头来Q对Hibernate的主要API调用作一下简要的介绍Q?br><br><br><strong>理解Hibernate的架?/strong><br><br>  当你想用Hibernate开发自qZ持久层的应用ӞW一件事情应当是熟悉它的~程接口。Hibernate的API接口设计得尽量简z明了,以方便开发h员。然而实际上׃ORM的复杂性,它的API一般都不可能设计得很简单。但是别担心Q你没有必要一下子了解所有的Hibernate的API接口。下面这张图描述了Hibernate在应用层和持久层中的一些重要的接口c:<br><br> <table width="90%" align=center border=0> <tbody> <tr> <td> <div align=center><img onerror="this.src='http://www.yesky.com/image20010518/189639.jpg';" hspace=3 src="http://www.tkk7.com/image20010518/189639.jpg" align=center vspace=1 border=1></div> </td> </tr> </tbody> </table> <br>  在上图中Q我们将应用层放在了持久层的上部Q实际上在传l的目中,应用层充当着持久层的一个客L角色。但对于一些简单的目来说Q应用层和持久层q没有区分得那么清楚Q这也没什么,在这U情况下你可以将应用层和持久层合q成了一层?br><br>  在上图中QHibernate的接口大致可以分Z下几U类型:<br><br>  · 一些被用户的应用程序调用的Q用来完成基本的创徏、读取、更新、删除操作以及查询操作的接口。这些接口是Hibernate实现用户E序的商业逻辑的主要接口,它们包括Session、Transaction和Query?br><br>  · Hibernate用来d诸如映射表这c配|文件的接口Q典型的代表有ConfigurationcR?br><br>  · 回调(Callback)接口。它允许应用E序能对一些事件的发生作出相应的操作,例如Interceptor、Lifecycle和Validatable都是q一cL口?br><br>  · 一些可以用来扩展Hibernate的映机制的接口Q例如UserType、CompositeUserType和IdentifierGenerator。这些接口可qL序来实现Q如果有必要Q?br><br>  Hibernate使用了J2EE架构中的如下技术:JDBC、JTA、JNDI。其中JDBC是一个支持关pL据库操作的一个基层;它与JNDI和JTA一L合,使得Hibernate可以方便地集成到J2EE应用服务器中厅R?br><br>  在这里,我们不会详细地去讨论Hibernate API接口中的所有方法,我们只简要讲一下每个主要接口的功能Q如果你想了解得更多的话Q你可以在Hibernate的源码包中的net.sf.hibernate子包中去查看q些接口的源代码。下面我们依ơ讲一下所有的主要接口Q?br><br>  <strong>核心接口</strong><br><br>  以下5个核心接口几乎在M实际开发中都会用到。通过q些接口Q你不仅可以存储和获得持久对象,q且能够q行事务控制?br><br>  Session接口<br><br>  Session接口对于Hibernate 开发h员来说是一个最重要的接口。然而在Hibernate中,实例化的Session是一个轻量的类Q创建和销毁它都不会占用很多资源。这在实际项目中实很重要,因ؓ在客L序中Q可能会不断地创Z及销毁Session对象Q如果Session的开销太大Q会l系l带来不良媄响。但值得注意的是Session对象是非U程安全的,因此在你的设计中Q最好是一个线E只创徏一个Session对象?br><br>  在Hibernate的设计者的头脑中,他们session看作介于数据q接与事务管理一U中间接口。我们可以将session惌成一个持久对象的~冲区,Hibernate能检到q些持久对象的改变,q及时刷新数据库。我们有时也USession是一个持久层理器,因ؓ它包含这一些持久层相关的操作,诸如存储持久对象x据库Q以及从数据库从获得它们。请注意QHibernate 的session不同于JSP应用中的HttpSession。当我们使用sessionq个术语Ӟ我们指的是Hibernate中的sessionQ而我们以后会HttpSesion对象UCؓ用户session?br><br>  SessionFactory 接口<br><br>  q里用到了一个设计模式――工厂模式,用户E序从工厂类SessionFactory中取得Session的实例?br><br>  令你感到奇怪的是SessionFactoryq不是轻量的!实际上它的设计者的意图是让它能在整个应用中׃n。典型地来说Q一个项目通常只需要一个SessionFactory够了,但是当你的项目要操作多个数据库时Q那你必Mؓ每个数据库指定一个SessionFactory?br>SessionFactory在Hibernate中实际vC一个缓冲区的作用,它缓冲了Hibernate自动生成的SQL语句和一些其它的映射数据Q还~冲了一些将来有可能重复利用的数据?br><br>  Configuration 接口<br><br>  Configuration接口的作用是对Hibernateq行配置Q以及对它进行启动。在Hibernate的启动过E中QConfigurationcȝ实例首先定位映射文档的位|,dq些配置Q然后创Z个SessionFactory对象?br><br>  虽然Configuration接口在整个Hibernate目中只扮演着一个很的角色Q但它是启动hibernate时你所遇到的每一个对象?br><br>  Transaction 接口<br><br>  Transaction接口是一个可选的APIQ你可以选择不用这个接口,取而代之的是Hibernate的设计者自己写的底层事务处理代码?Transaction接口是对实际事务实现的一个抽象,q些实现包括JDBC的事务、JTA中的UserTransaction、甚臛_以是CORBA事务。之所以这栯计是能让开发者能够用一个统一事务的操作界面,使得自己的项目可以在不同的环境和容器之间方便地移倹{?br><br>  Query和Criteria接口<br><br>  Query接口让你方便地对数据库及持久对象q行查询Q它可以有两U表达方式:HQL语言或本地数据库的SQL语句。Queryl常被用来绑定查询参数、限制查询记录数量,q最l执行查询操作?br><br>  Criteria接口与Query接口非常cMQ它允许你创建ƈ执行面向对象的标准化查询?br><br>  值得注意的是Query接口也是轻量U的Q它不能在Session之外使用?br><br>  Callback 接口 <br><br>  当一些有用的事g发生时――例如持久对象的载入、存储、删除时QCallback接口会通知HibernateL收一个通知消息。一般而言QCallback接口在用L序中q不是必ȝQ但你要在你的项目中创徏审计日志Ӟ你可能会用到它?br><br><br><br><br><br>  <strong>一个重要的术语QType</strong><br><br>  Hibernate的设计者们发明了一个术语:TypeQ它在整个构架中是一个非常基、有着强大功能的元素。一个Type对象能将一个Javacd映射到数据库中一个表的字D中去(实际上,它可以映到表的多个字段中去Q。持久类的所有属性都对应一个type。这U设计思想使用Hibernate有着高度的灵zL和扩展性?br><br>  Hibernate内置很多typecdQ几乎包括所有的Java基本cdQ例如Java.util.Currency、Java.util.calendar、byte[]和Java.io.Serializable?br><br>  不仅如此QHibernateq支持用戯定义的typeQ通过实现接口UserType和接口CompositeUserTypeQ你可以加入自己的type。你可以利用q种特色让你的项目中使用自定义的诸如Address、Nameq样的typeQ这样你可以获得更大的便利Q让你的代码更优雅。自定义type在Hibernate中是一Ҏ心特Ԍ它的设计者鼓׃多多使用它来创徏一个灵zR优雅的目Q?br><br>  <strong>{略接口</strong><br><br>  Hibernate与某些其它开源Y件不同的q有一点――高度的可扩展性,q通过它的内置{略机制来实现。当你感觉到Hibernate的某些功能不I或者有某些~陷Ӟ你可以开发一个自q{略来替换它Q而你所要做的仅仅只是承它的某个策略接口,然后实现你的新策略就可以了,以下是它的策略接口:<br><br>  · 主键的生?(IdentifierGenerator 接口) <br><br>  · 本地SQL语言支持 (Dialect 抽象c? <br><br>  · ~冲机制 (Cache 和CacheProvider 接口) <br><br>  · JDBC q接理 (ConnectionProvider接口) <br><br>  · 事务理 (TransactionFactory, Transaction, ?TransactionManagerLookup 接口) <br><br>  · ORM {略 (ClassPersister 接口) <br><br>  · 属性访问策?(PropertyAccessor 接口) <br><br>  · 代理对象的创?(ProxyFactory接口)<br><br>  HibernateZ上所列的机制分别创徏了一个缺省的实现Q因此如果你只是要增强它的某个策略的功能的话Q只需单地l承q个cd可以了,没有必要从头开始写代码?br><br>  以上是Hibernate的一些核心接口,但当我们真正开始用它进行开发时Q你的脑里可能M有一个疑问:我是通过什么方式,q从哪里取得Session的呢Q以下我们就解答q个问题?br><br>  <strong>基础配置</strong><br><br>  现在回顾一下我们先前的内容Q我们写Z一个示例程序,q简要地讲解了Hibernate的一些核心类。但要真正你的目q行hQ还有一件事必须要做Q配|。Hibernate可以配置成可在Q何Java环境中运行,一般说来,它通常被用?Q?层的C/S模式的项目中Qƈ被部|在服务端。在q种目中,Web览器、或Java GUIE序充当者客L。尽我们的焦点主要是集中在多层web应用Q但实际上在一些基于命令行的应用中也可以用Hibernate。ƈ且,对Hibernate的配|在不同的环境下都会不同QHibernateq行在两U环境下Q可理环境和不可管理环?br><br>  · 可管理环境――这U环境可理如下资源Q池资源理Q诸如数据库q接池和Q还有事务管理、安全定义。一些典型的J2EE服务器(JBoss、Weblogic、WebSphereQ已l实Cq些?br><br>  · 不可理环境――只是提供了一些基本的功能Q诸如像Jetty或Tomcatq样的servlet容器环境。一个普通的Java桌面应用或命令行E序也可以认为是处于q种环境下。这U环境不能提供自动事务处理、资源管理或安全理Q这些都必须由应用程序自己来定义?br><br>  Hibernate的设计者们这两种环境设计了一个统一的抽象界面,因此对于开发者来说只有一U环境:可管理环境。如果实际项目是建立在诸如Tomcatq类不可理的环境中Ӟ那Hibernate会使用它自q事务处理代码和JDBCq接池,使其变ؓ一个可理环境?br>对于可管理的环境而言QHibernate会将自己集成在这U环境中。对于开发者而言Q你所要做的工作非常简单:只需从一个ConfigurationcM创徏一个SessionFactorycd可以了?br><br><br><br><br><br><strong>创徏一个SessionFactory对象</strong><br><br>  Z能创Z个SessionFactory对象Q你必须在Hibernate初始化时创徏一个Configurationcȝ实例Qƈ已写好的映文件交由它处理。这PConfiguration对象可以创Z个SessionFactory对象Q当SessionFactory对象创徏成功后,Configuration对象没有用了,你可以简单地抛弃它。如下是CZ代码Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>Configuration cfg = new Configuration();<br>cfg.addResource("hello/Message.hbm.xml");<br>cfg.setProperties( System.getProperties() );<br>SessionFactory sessions = cfg.buildSessionFactory(); </td> </tr> </tbody> </table> <br>  在以上代码中QMessage.hb.xmlq个映射文g的位|比较特D,它与当前的classpath相关。例如classpath包含当前目录Q那在上qC码中的Message.hbm.xml映射文g可以保存在当前目录下的hello目录中?br><br>  作ؓ一U约定,Hibernate的映文仉认以.htm.xml作ؓ其扩展名。另一个约定是坚持为每一个持久类写一个配|文Ӟ想一惛_果你所有持久类的映写入一个单独的配置文g中的话,那这个配|文件肯定非常庞大,不易l护。但q里又出C一个新问题Q如果ؓ每个cd一个配|文件的话,q么多的配置文g应该存放在哪里呢Q?br><br>  Hibernate推荐你将每个映射文g保存在与持久cȝ同的目录下,q且与持久类同名。例如我们第一个示例程序中的Message持久cL在hello目录下,那你必须在这个目录下存放名ؓMessage.hbm.xml的映文件。这样一个持久类都有自己的一个映文Ӟ避免了出现像struts目中的“struts-config.xml地狱”的情c如果你不遵循这U规定,那你必须手动地用addResource()Ҏ一个个的映文件蝲入;但你如果遵@q种规定Q那你可以方便地用addClass()Ҏ同时持久类和它的映文件蝲入,以下是体现这U便利性的CZ代码Q?br><br> <table borderColor=#ffcc66 width="90%" align=center bgColor=#e6e4dd border=1> <tbody> <tr> <td>SessionFactory sessions = new Configuration()<br>.addClass(org.hibernate.auction.model.Item.class)<br>.addClass(org.hibernate.auction.model.Category.class)<br>.addClass(org.hibernate.auction.model.Bid.class)<br>.setProperties( System.getProperties() )<br>.buildSessionFactory(); </td> </tr> </tbody> </table> <br>  当然QHibernate的映文件还有很多其它的配置选项Q比如数据库q接的设定,或是能够改变Hibernateq行时行为的一些设定。所有的讄可能是非常庞杂的Q以让你喘不过气来Q但是不必担心,因ؓHibernate为绝大多数值都讑֮了一个合理缺省|你只需要修改这些配|文件中的极一部分倹{?br><br>  你可以通过以下几种方式来修改Hibernate的系l配|参敎ͼ<br><br>  · 一个Java.util.Properties实例作ؓ参数传给ConfigurationcȝsetProperties()Ҏ?br><br>  · 在Hibernate启动时用Java –Dproperty=value的方式设|倹{?br><br>  · 在classpath可以扑ֈ的\径下创徏一个名为hibernate.properties的配|文件?<br><br>  · 在classpath可以扑ֈ的\径下创徏一个名为hibernate.cfg.xml的文Ӟq在ӞpropertyQ标{中定义属性倹{?br><br>  以上是对Hibernate的一个大致介l,如果你想知道得更多,那本文还是远q不够的Q我陆l推出更多关于Hibernate的资料。但有一Ҏ毫无疑问的:它的是一个非怼U的持久层解决ҎQ?br><br><br><br><br> <img src ="http://www.tkk7.com/xixidabao/aggbug/46596.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.tkk7.com/xixidabao/" target="_blank">JAVA之\</a> 2006-05-17 11:46 <a href="http://www.tkk7.com/xixidabao/archive/2006/05/17/46596.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss> <footer> <div class="friendship-link"> <p>лǵվܻԴȤ</p> <a href="http://www.tkk7.com/" title="亚洲av成人片在线观看">亚洲av成人片在线观看</a> <div class="friend-links"> </div> </div> </footer> վ֩ģ壺 <a href="http://www91pao.com" target="_blank">91ƷѲ</a>| <a href="http://wilbysec.com" target="_blank">޾Ʒþһ</a>| <a href="http://zz4466.com" target="_blank">þþƷһѿ</a>| <a href="http://19520888.com" target="_blank">޳avƬ롿</a>| <a href="http://zhidianzh.com" target="_blank">þþƷƷް </a>| <a href="http://hs718.com" target="_blank">ĻӰ߸</a>| <a href="http://zfzz008.com" target="_blank">Ʒ99þѹۿ</a>| <a href="http://yeshenghuowang.com" target="_blank">޴߶ר </a>| <a href="http://www16am8.com" target="_blank">պϵ</a>| <a href="http://3y0r.com" target="_blank">պƬӰ</a>| <a href="http://0827fang.com" target="_blank">þþƷһԡ</a>| <a href="http://www827556.com" target="_blank">޹Ļ߹ۿ</a>| <a href="http://wwkk3.com" target="_blank">AƬһ</a>| <a href="http://daohang123456.com" target="_blank">߿Ƶվ</a>| <a href="http://6006769.com" target="_blank">www.avѹۿ</a>| <a href="http://chenglide.com" target="_blank">޺ݺۺϾþ</a>| <a href="http://928348.com" target="_blank">ҹӰ߹ۿ </a>| <a href="http://hn-hshb.com" target="_blank">baoyu777Ƶ</a>| <a href="http://gycat.com" target="_blank">è˳վ߹ۿ</a>| <a href="http://www-070755.com" target="_blank">޹Ʒѹۿ</a>| <a href="http://www-qwh.com" target="_blank">У԰ɫС˵</a>| <a href="http://8b22.com" target="_blank">þþþAVר</a>| <a href="http://qq2071.com" target="_blank">պav</a>| <a href="http://tedegold.com" target="_blank">߾þþƷĹ</a>| <a href="http://zjyxc.com" target="_blank">ѿƬ</a>| <a href="http://tzkanglong.com" target="_blank">ŷͬgvվۿ</a>| <a href="http://bjhuicui.com" target="_blank">ĻƵww</a>| <a href="http://baocaoluoli.com" target="_blank">ý̫ˬƵѹ</a>| <a href="http://ac839.com" target="_blank">jyzzjyzzѹۿ</a>| <a href="http://gzweida88.com" target="_blank">˳Ƶ߲</a>| <a href="http://xuanzhicity.com" target="_blank">Ʒavɫ</a>| <a href="http://dghxm168.com" target="_blank">AVۺɫһ</a>| <a href="http://yuduruizhi.com" target="_blank">Ƶ</a>| <a href="http://600c20.com" target="_blank">ۺϾþþƷɫ</a>| <a href="http://27simnjingmiguan.com" target="_blank">۾ƷҹӰ</a>| <a href="http://cdlsy.com" target="_blank">޾Ʒ˳߹ۿ鶹</a>| <a href="http://gayhh.com" target="_blank">þպƷһ</a>| <a href="http://gyqcy.com" target="_blank">˳ɵӰ</a>| <a href="http://apguangyu.com" target="_blank">AVþþƷ </a>| <a href="http://gzqhit.com" target="_blank">aѹۿþav</a>| <a href="http://bearsou.com" target="_blank">ĻþƷƵ</a>| <script> (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })(); </script> </body>