??xml version="1.0" encoding="utf-8" standalone="yes"?> 一位ID是jackson1225的网友在javaeye询问?a title="一个大型Webpȝ的架构和部v选型问题" target="_blank">一个大型Webpȝ的架构和部v选型问题Q希望能提高现有的基于Java的Web应用的服务能力。由于架构模式和部v调优一直是JavaC的热门话题,q个问题引发了很多热心网友的讨论Q其中一些意见对其它大型Web目也有很好的指导意义。在讨论之初jackson1225q样描述了当前的应用的架构和部vҎ(gu)Q?/p>
目前pȝ架构如下: web层调用EJBq程接口来访问中间g层。web层首先通过一个XML配置文g中配|的EJB接口信息来调用相应的EJBq程接口Q?/p>
该系l中一ơ操作涉?qing)到两个ORACLE库以?qing)一个SQL SERVER库的讉K和操作,x三个数据库连接,在一个事务中完成?/p>
q样的架构其实很多公叔R在用,因ؓ(f)Struts和Tomcat分别是最行的Java Web MVC框架和Servlet容器Q而F5公司的负载均衡是横向扩展常见的解x案(例如配置session stickyҎ(gu)Q。由于这个系l中有跨数据源的事务Q所以用Weblogic Server EJB容器和支持两阶段提交的数据库驱动可以保证跨数据源的事物完整性(当然Q容器管理的分布式事务ƈ非是唯一和最优的解决Ҏ(gu)Q?/p>
但是随着Rod Johnson重量U的著作《J2EE Development without EJB》和其中的Spring框架的流行,轻量U框架和轻量U容器的概念已经深入人心。所以对于jackson1225提出的这个场景,大多数网友都提出了置疑,认ؓ(f)q个pȝ滥用了技术,完全是在费钱。网友们大都认ؓ(f)SLSBQ无状态会(x)话BeanQ完全没有必要出现在q个场景中,认ؓ(f)SLSB通过q程接口讉K本地资源?x)有很大的性能开销Q这U观点也是Rod johnson在without EJB中批判EJB 2.x中的一大反模式?/p>
׃JavaEE是一个以模式见长的解x案,模式和架构在JavaEE中占有很重要的地位,所以很多业内专家也都警?#8220;反模式(Anti-patternsQ?#8221;的出现。对于上面所q的Ҏ(gu)是否是反模式Qjackson1225马上站出来申辩:(x) 我们目是把EJB作ؓ(f)一个FacadeQ只是提供给WEB层调用的q程接口Q而且只用了无状态会(x)话BeanQ所以性能上还可以的?/p>
q个解释很快得到了一些网友的认可Q但是大家很快意识到架构的好坏决定于是否能够满用户的需求,davexinQ可能是jackson1225的同事)描述了这个系l的用户和ƈ发情况:(x) 现在有用?000万,马上要和另一个公司的?x)员pȝ合ƈQ加h一共有9000万用戗数据量单表中有一亿条以上的数据。这是基本的情况Q其实我觉得现在的架构还是可以的Q现在支持的q发大概5000q发用户左右Q接下来?x)进行系l改造,目标支持1万个q发用户?/p>
具体的ƈ发量公布后又有网友置疑这个数据,认ؓ(f)q个pȝ的Servlet容器支持的ƈ发数太小Q怀疑是否配|不够优化。davexin又补充了该项目的服务器配|:(x) pȝ前端tomcat都是用的刀片,配置?G内存Qcpu大概?.0GQ每台机器也支?50-400个ƈ发,再多的话Q就?x)相应时间非常的常,?0U,失去了意?Q所以我们才得出q样的结论的?/p>
一位ID是cauherk的网友提Z比较中肯的意见,他没有从Web容器单纯的ƈ发支持能力上提出改进Ҏ(gu)Q而是提出了对于类似的应用?a title="一些通用的改q提C? target="_blank">一些通用的改q提C?/a>Q这里摘要一下:(x) 可以按照业务、区域等{特性对数据库进行配|,可以考虑分库、用rac、分区、分表等{策略,保数据库能正常的进行交易?/p>
要在两个数据库中操作Q那么必考虑到分布式事务。你应该仔细的设计你的系l,来避免用分布式事务Q以避免分布式事务带来更多的数据库压力和其它问题。推荐你采用延迟提交的策?q不保证数据的完?Q来避免分布式事务的问题Q毕竟commitp|的几率很低?/p>
静态、图片独立用不同的服务器,对于常态的静态文Ӟ采用E-TAG或者客L(fng)~存Q?google很多是q样q的。对于热点的功能Q考虑使用完全装蝲到内存,保证l对的响应速度Q对于需要频J访问的热点数据Q采用集中缓?多个可以采用负蝲均衡)Q减L据库的压力?/p>
对于几乎除二q制文gQ都应该在L4上配|基于硬件的压羃Ҏ(gu)Q减网l的量。提高用户用的感知?/p>
可以考虑采用镜像、多路网l接入、基于DNS的负载均衡。如果有_的投资,可以采用CDN(内容分发|?Q减M的服务器压力?/p>
cauherk的这个分析比较到位,其中ETags的方案是最q的一个热点,InfoQ?#8220;使用ETags减少Web应用带宽和负?/a>”里面对这U方案有很详l的介绍。一般以数据库ؓ(f)中心的Web应用的性能瓉都在数据库上Q所以cauherk把数据库和事务问题放C前两位来讨论。但是davexin解释在所讨论的这个项目中数据库ƈ非瓶颈:(x) 我们的压力不在数据库层,在web层和F5?当高峰的时?QF5也被Ҏ(gu)了,是每秒点击过30万,web动态部分根本承受不了。根据我们程序记录,20台web最多承?000个ƈ发,如果再多Qtomcat׃响应了。就像死了一栗?/p>
q个回复让接下来的讨论都集中于Web容器的性能优化Q但?a title="JavaEye站长robbin发表了自q意见" target="_blank">JavaEye站长robbin发表了自q意见Q将话题引回了这个项目的架构本nQ?/p>
performance tuning最重要的就是定位瓶颈在哪里Q以?qing)瓶颈是怎么产生的?/strong> 我的推测是瓶颈还是出在EJBq程Ҏ(gu)调用上! tomcat上面的java应用要通过EJBq程Ҏ(gu)调用Q来讉Kweblogic上面的无状态SessionBeanQ这L(fng)q程Ҏ(gu)调用一般都?00ms~500msU别Q或者更多。而如果没有远E方法调用,即大量采用spring的动态反,一ơ完整的webh处理在本地JVM内部的完成时间一般也不过20ms而已。一ơwebh需要过长的执行旉Q就?x)导致servletU程被占用更多的旉Q从而无法及(qing)时响应更多的后箋h?/p>
如果q个推测是成立的话,那么我的是既然你没有用到分布式事务Q那么就q脆LEJB。weblogic也可以全部撤掉,业务层用spring取代EJBQ不要搞分布式架构,在每个tomcat实例上面部v一个完整的分层l构?/p>
另外在高q发情况下,apache处理静态资源也很耗内存和CPUQ可以考虑用轻量web server如lighttpd/litespeed/nginx取代之?/p>
robbin的推断得C|友们的支持Qdavexin也认同robbin的看法,但是他解释说公司认ؓ(f)攑ּSLSB存在风险Q所以公司們于通过Tomcat替换为Weblogic Server 10来提升系l的用户支撑能力?a title="robbin则马上批评了q种做法" target="_blank">robbin则马上批评了q种做法Q?/p>
坦白说我q从来没有听说过大规模互联网应用使用EJB的先例。ؓ(f)什么大规模互联|应用不能用EJBQ其实就是因为EJB性能太差Q用了EJB几乎必然出现性能障碍?/p>
web容器的性能说到底无非就是ServletU程调度能力而已QTomcat不像WebLogic那样附加n多管理功能,跑得快很正常。对比测试一下WebLogic的数据库q接池和C3P0q接池的性能也会(x)发现cM的结论,C3P0可要比WebLogic的连接池快好几倍了。这不是说WebLogic性能不好Q只不过weblogic要实现更多的功能Q所以在单一的速度斚w׃(x)牺牲很多东西?/p>
以我的经验来判断Q用tomcat5.5以上的版本,配置a(chn)pr支持Q进行必要的tuningQ用BEA JRockit JVM的话Q在你们目前的刀片上面,支撑500个ƈ发完全是可以做到的。结合你们目?0个刀片的gQ那么达?万ƈ发是没问题的。当然这样做的前提是必须扔掉EJBQƈ|web层和业务层在同一个JVM内部?/p>
接下来robbinq针对davexin对话题中的应用分别在tomcat和weblogic上的试数据q行了分析:(x) 2?台weblogic10 ExpressQ相当于1台tomcatQ用于发布jsp应用Q加1台weblogic10Q发布ejb应用Q,能支?000个ƈ发用?..... q说明瓶颈还不在EJBq程调用上,但是问题已经逐渐清楚了。ؓ(f)什么weblogic充当web容器发vq程EJB调用的时候可以支?000个ƈ发,但是tomcat只能?50个?只有两个可能的原因:(x)
接着springside目发v者江南白衣也提出了一个M的优化指|(x) 1.基础配置优化 tomcat 6Q?tomcat参数调优? 2.业务层优?/p>
部分功能本地化,而不调remote session bean? 3.展示层优?/p>
动态页面发布ؓ(f)静态页面? davexin在调整了Tomcat配置后应验了robbin对tomcat配置问题的质疑,davexinq样描述l过配置优化以后的测试结果:(x) l过试Qƈ发h数是可以辑ֈ像robbin所说的一P能够?00人左叻I如果压到q发700人,有15%左右的失败,虽然在调整上面参C后,q发人数上去了,但是在同L(fng)旉内所完成的事务数量下降了10%左右Qƈ且响应时间gq了1U左叻I但从整体上来_(d)牺牲一点事务吞吐量和响应时_(d)q发人数能够提高500Q觉得还是值得的?/p>
xq个话题有了一个比较好的结果。这个话题ƈ非完全针对一个具体的目才有意义Q更重要的是在分析和讨论问题的过E中|友们解决问题的思\Q尤其是cauherk、robbin、江南白衣等几位|友提出的意见可以让q大Java Web目开发者了解到中、大型项目所需要考虑的架构和部v所需要考虑的关键问题,也消除了很多人对轻量Servlet容器与EJB容器性能的一些误解?/p>
在讨Zq有一些小插曲Q如davexin和江南白衣讨ZJRocket的实ӞRealtimeQ版本是否可以提升Servlet容器的相应能?/a>Q答案是不可以。还有ID为mfc42d的网友从Servlet容器的ƈ发支持能力引甛_?a title="Java的线E调度能力和NIO对Servelet容器的意? target="_blank">Java的线E调度能力和NIO对Servelet容器的意?/a>Q他推荐了自q两篇不错的blog“java的线E实?/a>”?#8220;javaq程使用的最大内存的数?/a>”Qblog文章里面从JVM源码U别分析了Java的线E支持能力,面(f)JVM性能调优问题的网友可以认真阅M下?/p>
Ganesh的原文:(x) 我以前翻译的Hibernate的session的javadocQ当时好像从字面上没有什么大的区别,配合q个帖子Q要重新想一?D
引用Q?
......
4?台tomcat4.1?台weblogic8Q只能支?50个ƈ发用Ptomcatpl超Ӟ说明此种l构瓉在tomcat?
JRockit JVM? JVM参数调优Q?br />
Apache+Squid 处理静态内容?
异步提交操作,JMSQ?br />
cache热点数据Q?
Cache部分动态页面内容?
]]>
很多q样不是OO的,很多Hibernate的细节造成它复杂,有h说Hibernate影响了他们的工作效率Q等{,反正反对者站出来很多。我惌是可喜的?br />说明在Java的Killer-APPl合Spring+Hibernate后面q是有很多持怀疑态度的h的,我们随时都需要提问题的大脑!
所以,我们也思考一下吧Q?br />1、你一定要量的不断地熟?zhn)你所使用的技术,可以是Hibernate也可以是JDBCQ时M持对相关知识的饥渴。这样减因Z的无知造成的失误?br />2、思考别人的批评Q或者说ORM不好Q或者说q种那种ORM不好Q你都要听着Q然后思考,然后坚持你思考的l果Q直C一ơ思考。这个过E最好保持一定的技术偏执,因ؓ(f)没有性格的程序员不是好程序员?br />3、放眼各U技术,了解怼技术,q样你不?x)在城头更换大王旗的时候失?D。现在我们可以回头看看JDBC Dataset和JPAQ向上向下都要看齐嘛。了解了怼技术更有助于你的思考?br />4、知道你在讨Z么,知道背景。你需要知道Hibernate主要在Java or JavaEE中用,当然有Hibernate.netQ但是它不怎么火。还有,Hibernate的基q是SQL和DBMSQ你需要仔l了解这两种东西?br />5、?zhn)道?/p>
Hibernate - Difference between session's get() and load()
http://gmarwaha.blogspot.com/2007/01/hibernate-difference-between-sessions.html
TSS上的讨论Q?br />Difference between Hibernate's get() and load()?
http://www.theserverside.com/news/thread.tss?thread_id=43887
http://www.tkk7.com/iamtin/archive/2006/03/06/33910.html
BTWQ因为做手术在家休养Q所以才有时间发发牢?D
]]>
其实关键问题是DetachedCriteria.getExecutableCriteriaq回的Criteria实例中已l包含了投媄和排序信息,如果你想先查LQ?
得先去掉这些信息;查完LQ再查实际数据时Q又得加上这些信息。而Criteria可没提供对它Ҏ(gu)改去的接口?
不过Q仔l考察一下DetachedCriteria可以发现Q它cM一U暂存了查询条g的值对象,通过与一个session对象l合生成
一个“可执行的”CriteriaQ相当于先执行session.createCriteriaQ然后把自己保存的条件设|进厅R?
既然如此Q我q脆自己提供一个类似的c,暂存查询条gQ与sessionl合生成executableCriteriaQ但我先提供l用户一个只包含
查询条g的Criteria实例Q让用户先拿着它去查LQ然后再提供加入投媄和排序信息的Ҏ(gu)Q让用户拿着它去查数据,不就解决问题了?
-------------- CZ代码Q?
Update 2006-6-6 20:52Q?br />恩,谢谢怽。想了想q是DetachedCriteria设计的目的和我想要的不一栗DetachedCriteria目前只是Zqsession可以构造,而ƈ不是Z反复使用?br />因ؓ(f)Q?、它里面只包装了一个CriteriaImplQ所以实际上两者生命周期比较一致。而且每次getExecutableCriteria都直接返回这个CriteriaImplQ而不是重新创建,q就造成了容易被意外修改?2、它没有提供与add对应的removeҎ(gu)Q这造成它只能篏U而不能擦拭(对于OrderQ,用反就太脏了?br />whimet写的q个MyDetachedCriteria倒是可以解决问题Q更W合我们惌复用DetachedCriteria的需求,不过我觉得调用v来接口上不太l一。Order和Projection一般不用复用,我想可以不用实现DetachedCriteria接口Q干脆修改GenericDAO的方法好了,写个接受List<Criteria>或者Criteria...的方法对应就可以了。反正觉得DetachedCriteria目前的实现是不如意,hack它吧……我想提供removeq些Order和Projection的方法?br />q里用反解决了他遇到的Order的问题,大家可以参考,看来DetachedCriteria目前实现的还不够令h满意Q?br />关于Hibernate的DetachedCriteria查询的addOrder问题的解军_?/a>
Java应用E序与Hibernate之间的主要运行时接口。它是抽象了持久化服务概늚核心抽象APIcR?br />
Session的生命周期绑定在一个物理的事务QtansactionQ上面。(长的事务可能跨越多个数据库事物。)
Session的主要功能是提供Ҏ(gu)的实体cd例的创徏Q读取和删除操作。实例可能以下面三种状态存在:(x)
自由状态(transientQ? 不曾q行持久化,未与MSession相关?br />持久化状态(persistentQ? 仅与一?tt>Session相关?br />游离状态(detachedQ? 已经q行q持久化Q但当前未与MSession相关?br />
游离状态的实例可以通过调用save()?tt>persist()或?tt>saveOrUpdate()Ҏ(gu)q行持久化。持久化实例可以通过调用 delete()变成游离状态。通过get()?tt>load()Ҏ(gu)得到的实例都是持久化状态的。游ȝ态的实例可以通过调用 update()?saveOrUpdate()?tt>lock()或?tt>replicate()q行持久化。游L者自q态下的实例可以通过调用merge()Ҏ(gu)成ؓ(f)一个新的持久化实例?br />
save()?tt>persist()会(x)引发SQL?tt>INSERTQ?tt>delete()?x)引发SQLDELETEQ?tt>update()?tt>merge()?x)引发SQLUPDATE。对持久化(persistentQ?/i>实例的修改在h提交的时候会(x)被检到Q它也会(x)引vSQLUPDATE?tt>saveOrUpdate()或?tt>replicate()?x)引发SQLINSERT或?tt>UPDATE?br />
其具体实现ƈ不一定是U程安全的。每个线E?事务应该从一?tt>SessionFactory获取自己的session实例?br />
如果其持久化对象cL可序列化的,?tt>Session实例也是可序列化的?br />
一个典型的事务应该使用下面的Ş式:(x)
Session sess = factory.openSession(); Transaction tx; try { tx = sess.beginTransaction(); //do some work ... tx.commit(); } catch (Exception e) { if (tx!=null) tx.rollback(); throw e; } finally { sess.close(); }
SessionFactory
Method Summary | |
Transaction | beginTransaction() 开始一个工作单元ƈ且返回相兌?tt>事务QTransactionQ?/tt>对象? |
void | cancelQuery() l止执行当前查询?/td> |
void | clear() 完整的清除这个session?/td> |
Connection | close() 停止q个SessionQ通过中断JDBCq接q且清空Qcleaning upQ它?/td> |
Connection | connection() 获取q个Session的JDBCq接?br /> 如果q个session使用了积极的collection释放{略Q如CMT-容器控制事务的环境下Q,关闭q个调用的连接的职责应该由当前应用程序负责? |
boolean | contains(Object object) 查这个对象实例是否与当前?tt>Session兌Q即是否为Persistent状态)?/td> |
Criteria | createCriteria(Class persistentClass) 为给定的实体cL它的类创徏一个新?tt>Criteria实例?/td> |
Criteria | createCriteria(Class persistentClass, String alias) Ҏ(gu)l定的实体类或者它的超cdZ个新?tt>Criteria实例Qƈ赋予它(实体c)一个别名?/td> |
Criteria | createCriteria(String entityName) Ҏ(gu)l定的实体的名称QnameQ,创徏一个新?tt>Criteria实例?/td> |
Criteria | createCriteria(String entityName, String alias) Ҏ(gu)l定的实体的名称QnameQ,创徏一个新?tt>Criteria实例Qƈ赋予它(实体c)一个别?/td> |
Query | createFilter(Object collection, String queryString) Ҏ(gu)l定的collection和过滤字W串Q查询条Ӟ创徏一个新?tt>Query实例?/td> |
Query | createQuery(String queryString) Ҏ(gu)l定的HQL查询条g创徏一个新?tt>Query实例?/td> |
SQLQuery | createSQLQuery(String queryString) Ҏ(gu)l定的SQL查询条g创徏一个新?tt>SQLQuery实例?/td> |
void | delete(Object object) 从数据库中移除持久化QpersistentQ对象的实例?/td> |
void | delete(String entityName, Object object) 从数据库中移除持久化QpersistentQ对象的实例?/td> |
void | disableFilter(String filterName) 用当前session的名U过滤器?/td> |
Connection | disconnect() 断开Session与当前的JDBCq接?/td> |
Filter | enableFilter(String filterName) 打开当前session的名U过滤器?/td> |
void | evict(Object object) 当前对象实例从session~存中清除?/td> |
void | flush() 强制提交hQflushQ?tt>Session?/td> |
Object | get(Class clazz, Serializable id) Ҏ(gu)l定标识和实体类q回持久化对象的实例Q如果没有符合条件的持久化对象实例则q回null?/td> |
Object | get(Class clazz, Serializable id, LockMode lockMode) Ҏ(gu)l定标识和实体类q回持久化对象的实例Q如果没有符合条件的持久化对象实例则q回null?/td> |
Object | get(String entityName, Serializable id) q回与给定的实体命名和标识匹配的持久化实例,如果没有对应的持久化实例则返回null?/td> |
Object | get(String entityName, Serializable id, LockMode lockMode) q回与给定的实体cd标识所匚w的持久化实例Q如果没有对应的持久化实例则q回null?/td> |
CacheMode | getCacheMode() 得到当前的缓存模式?/td> |
LockMode | getCurrentLockMode(Object object) 给定对象当前的锁定U别?/td> |
Filter | getEnabledFilter(String filterName) Ҏ(gu)名称获取一个当前允许的qo(h)器(filterQ?/td> |
EntityMode | getEntityMode() 获取q个session有效的实体模式?/td> |
String | getEntityName(Object object) q回一个持久化对象的实体名U?/td> |
FlushMode | getFlushMode() 获得当前的刷新提交(flushQ模式?/td> |
Serializable | getIdentifier(Object object) 获取l定的实体对象实例在Session的缓存中的标识,如果该实例是自由状态(TransientQ的或者与其它Session兌则抛Z个异常?/td> |
Query | getNamedQuery(String queryName) 从映文件中Ҏ(gu)l定的查询的名称字符串获取一?tt>QueryQ查询)实例?/td> |
Session | getSession(EntityMode entityMode) Ҏ(gu)l定的实体模式(Entity ModeQ开始一个新的有效的Session?/td> |
SessionFactory | getSessionFactory() 获取创徏q个session?tt>SessionFactory实例? |
SessionStatistics | getStatistics() 获取q个session的统计信息?/td> |
Transaction | getTransaction() 获取与这个session兌?tt>TransactionQ事务)实例?instance associated with this session. |
boolean | isConnected() 查当?tt>Session是否处于q接状态?/td> |
boolean | isDirty() 当前Session是否包含需要与数据库同步的Q数据状态)变化 Q如果我们刷新提交(flushQ这个session是否?x)有SQL执行Q?/td> |
boolean | isOpen() 查当?tt>Session是否仍然打开?/td> |
Object | load(Class theClass, Serializable id) 在符合条件的实例存在的情况下Q根据给定的实体cd标识q回持久化状态的实例?/td> |
Object | load(Class theClass, Serializable id, LockMode lockMode) 在符合条件的实例存在的情况下Q根据给定的实体cR标识及(qing)指定的锁定等U返回持久化状态的实例?/td> |
void | load(Object object, Serializable id) 与l定的标C对应的持久化状态(|复制到给定的自由状态(trasientQ实例上?/td> |
Object | load(String entityName, Serializable id) 在符合条件的实例存在的情况下Q根据给定的实体cd标识q回持久化状态的实例?/td> |
Object | load(String entityName, Serializable id, LockMode lockMode) 在符合条件的实例存在的情况下Q根据给定的实体cR标识及(qing)指定的锁定等U返回持久化状态的实例?/td> |
void | lock(Object object, LockMode lockMode) 从给定的对象上获取指定的锁定U别?/td> |
void | lock(String entityName, Object object, LockMode lockMode) 从给定的对象上获取指定的锁定U别?/td> |
Object | merge(Object object) 给定的对象的状态复制到h相同标识的持久化对象上?/td> |
Object | merge(String entityName, Object object) 给定的对象的状态复制到h相同标识的持久化对象上?/td> |
void | persist(Object object) 一个自q态(transientQ的实例持久化?/td> |
void | persist(String entityName, Object object) 一个自q态(transientQ的实例持久化?/td> |
void | reconnect() 不推荐的?/b> 手工的重新连接只应用于应用程序提供连接的情况Q在q种情况下或许应该?a href="/org/hibernate/Session.html#reconnect(java.sql.Connection)"> reconnect(java.sql.Connection) ?/i> |
void | reconnect(Connection connection) 重新q接到给定的JDBCq接?/td> |
void | refresh(Object object) 从数据库中重新读取给定实例的状态?/td> |
void | refresh(Object object, LockMode lockMode) Ҏ(gu)指定?tt>锁定模式QLockModeQ?/tt>Q从数据库中重新dl定实例的状态?/td> |
void | replicate(Object object, ReplicationMode replicationMode) 使用当前的标识值持久化l定的游ȝ态(TransientQ的实体?/td> |
void | replicate(String entityName, Object object, ReplicationMode replicationMode) 使用当前的标识值持久化l定的游ȝ态(TransientQ的实体?/td> |
Serializable | save(Object object) 首先为给定的自由状态(TransientQ的对象Q根据配|)生成一个标识ƈ赋|然后其持久化?/td> |
Serializable | save(String entityName, Object object) 首先为给定的自由状态(TransientQ的对象Q根据配|)生成一个标识ƈ赋|然后其持久化?/td> |
void | saveOrUpdate(Object object) Ҏ(gu)l定的实例的标识属性的|注:(x)可以指定unsaved-value。一般默认null。)来决定执?save() ?tt>update()操作?/td> |
void | saveOrUpdate(String entityName, Object object) Ҏ(gu)l定的实例的标识属性的|注:(x)可以指定unsaved-value。一般默认null。)来决定执?save() ?tt>update()操作?/td> |
void | setCacheMode(CacheMode cacheMode) 讄h提交模式?/td> |
void | setFlushMode(FlushMode flushMode) 讄h提交模式?/td> |
void | setReadOnly(Object entity, boolean readOnly) 一个未l更改的持久化对象设|ؓ(f)只读模式Q或者将一个只d象标Cؓ(f)可以修改的模式?/td> |
void | update(Object object) Ҏ(gu)l定的detachedQ游ȝ态)对象实例的标识更新对应的持久化实例?/td> |
void | update(String entityName, Object object) Ҏ(gu)l定的detachedQ游ȝ态)对象实例的标识更新对应的持久化实例?/td> |
equals:
是判断是否相等的方法?BR>基本U定如下Q?BR>自反Q就是x.equals(x)应该q回true
对称Qx.equals(y)q回true则y.equals(x)q回true
传递:(x)x.euqls(y)==trueQy.equals(z)==trueQ那么x.equals(z)也必返回true
持箋性:(x)是说如果x.equals(y)==trueQ不它调用多少ơ都应该q回true
非空引用Qx.equals(null)应该q回false
准则原因q里Q?BR>http://java.sun.com/j2se/1.3/docs/api/java/lang/Object.html#hashCode()
在实际Domain中如何实C们比较好呢?从eclipseWork中的模板生成如下Q?/P>