SQLServer就有自增型整数,不用输入,也不能输入

解决方案 »

  1.   

    偶知道 ,数据库会自动生成呀
    但是偶不知道,cmp中如何不带ID,来create
    就是把home.create(id)转成home.create()
      

  2.   

    Integer ejbCreate(String name, ...) { // 没有id参数
        setName(...);
        set其他字段(...);
        return null;
    }
      

  3.   

    不带参数,在create里直接生成新数据,ID在现有ID上加1
      

  4.   

    J2EE - 如何在JBoss中解决自动增长键值问题
    【前言】 
    自动增长键值是关系数据库的一个显著功能特征,如MS Sql Server、MySql可以直接将一个字段设置成自动增长(auto-increment)类型,Oracle也提供了类似的sequence number功能。然而在EJB2.0规范之前,CMP部分并没有对自动增长键有相关的说明,这一缺陷一直深受J2EE开发人员的诟病,而应用服务器开发商为此也提出了各自的解决方案,但是在没有上升到规范的高度之前,这些解决方案都是非正式的,也就是说缺乏通用性,如闻名遐尔的Weblogic,在1.0规范的时候也仅仅是对一些主流数据库进行了支持。JBoss的3.0版本没有支持auto-increment特征,到3.2版本才正式支持。本篇是介绍在3.0版本下JBoss如何使用AutoNumber这个EJB插件来实现数据库表键值自动增长功能,在文章的后半部分介绍在3.2版本中如何使用“unknown keys”特征来真正实现自动键值增长。后台数据库使用的是MySql4.0.12版本。   EJB是一个仍在快速发展的技术,发展就意味着变化,而“快速”同样适用于“变化”二字,尤其在EJB的CMP部分,其变化尤为突出。CMP是EJB最精华的体现,由于使用了对象的方式来描述数据库,而目前大多数据库仍以关系类型为主,表达与被表达之间存在着不可忽略的差异,因而EJB规范在制定中需要考虑相关因素,不得不做出均衡的取舍。由于前面所提到的发展性,规范的最终成熟也不是一蹴而就,一个规范的推出可能迫于时机因素(技术或者是商业) 而推迟,开发人员在实际开发当中应该能够明确意识到这点,在遇到规范所引发的局限的地方,充分发挥主观能动性,在考虑问题解决问题方面能够突破到一个新境界。   JBoss3.0提供的AutoNumber处理方式 
    JBoss3.0给用户提供了一个AutoNumber插件来“伪”实现CMP的键值自动增长,之所以冠以一个“伪”字,是因为这个功能并不是通过数据库自身的功能来实现,而是利用了一个工厂类型的CMP Entity Bean来为各类实体Bean(对应数据库中的表)生成相应的键值,用户不必考虑如何来维护键值的唯一性和连续性,这一切都由AutoNumber来代理,用户要做的就是在使用这个“插件”之前先部署这个Bean Jar,并弄清楚其调用原理以知悉如何来使用这个CMP Bean。   在着手部署这个插件之前,先来看一段JBoss源代码文件AutoNumberFactory的代码注释: /**   * Gets the next key for the given collection.   * Note 1: you must deploy EJB AutoNumber   * Note 2: the keys are persistent in your database, independent of    * the actual table   * Note 3: you can only add instances to the collection which have a    * key generated by this method, otherwise the keys are not guaranteed   * to be unique   * Note 4: key values are >= 0   */ 大意如下:该类是为了给一个键集合获取下一个键值,为了使用这个工厂方法,你首先必须部署AutoNumber这个EJB,键集信息被存储在数据库中,与实际表无关,为了确保键值的唯一性,你只能通过这个工厂方法来生成键值,键值的范围是大于或等于0的整数。   部署AutoNumber CMP Entity Bean 
    AutoNumber插件的位置是在%JBOSS_HOME%/server/default/lib/ autonumber-plugin.jar,令人觉得惊奇的是JBoss没有为这个插件如何部署给出例子或者是指导性的说明文件(或者是本人尚未发现)。在研究了一下相关的源代码后,以下是一个部署的实例,以及在数据库中如何创建一个表来对应这个实体Bean。   首先在MySql中创建一张表,来对应AutoNumber bean。这张表很简单,只包含两个字段列name和value,name列用以存储键值名称,value用以存储当前可用的键值。DDL语句如下: CREATE TABLE `myauto` (   `name` varchar(20) NOT NULL default '',   `value` int(11) NOT NULL default '0',   PRIMARY KEY  (`name`),   UNIQUE KEY `name` (`name`) ) TYPE=InnoDB 这里将name设置为主键,并且为unique,约束了键名不可重复。   Ejb-jar.xml描述文件: <ejb-jar>     <enterprise-beans>         <entity>             <display-name>MyAutoNumber</display-name>             <ejb-name>AutoNumber</ejb-name>             <home>org.jboss.varia.autonumber.AutoNumberHome</home>             <remote>org.jboss.varia.autonumber.AutoNumber</remote>             <ejb-class>org.jboss.varia.autonumber.AutoNumberEJB2</ejb-class>             <persistence-type>Container</persistence-type>             <prim-key-class>java.lang.String</prim-key-class>             <reentrant>False</reentrant>             <cmp-version>2.x</cmp-version>             <abstract-schema-name>autonumberschema</abstract-schema-name>                     <cmp-field>                 <field-name>name</field-name>             </cmp-field>             <cmp-field>                 <field-name>value</field-name>             </cmp-field>             <primkey-field>name</primkey-field>         </entity>     </enterprise-beans>     <assembly-descriptor>         <container-transaction>             <method>                 <ejb-name>AutoNumber</ejb-name>                 <method-name>*</method-name>             </method>             <trans-attribute>Required</trans-attribute>         </container-transaction>     </assembly-descriptor> </ejb-jar>   Jbosscmp-jdbc.xml描述文件: <jbosscmp-jdbc>     <enterprise-beans>         <entity>             <ejb-name>AutoNumber</ejb-name>             <datasource>java:/MySqlDS</datasource>             <datasource-mapping>mySQL</datasource-mapping>             <create-table>false</create-table>             <remove-table>false</remove-table>             <read-only>false</read-only>             <table-name>myauto</table-name>             <cmp-field>                 <field-name>name</field-name>                 <column-name>NAME</column-name>             </cmp-field>             <cmp-field>                 <field-name>value</field-name>                 <column-name>VALUE</column-name>             </cmp-field>         </entity>     </enterprise-beans> </jbosscmp-jdbc> <datasource>标签值改成你相应的数据源名称,<table-name>及<field-name>标签值也改成你相应建立的数据库表及字段列名称。   Jboss.xml描述文件: <jboss>     <enterprise-beans>         <entity>             <ejb-name>AutoNumber</ejb-name>             <jndi-name>JBossUtilAutoNumber</jndi-name>         </entity>     </enterprise-beans> </jboss> 
      

  5.   

    在正式部署前,将部署文件置入一个meta-inf目录中,然后连同autonumber-plugin.jar文件一同jar进一个新的jar文件当中,最后是将该新jar文件copy至%JBOSS_HOME%/server/default/deploy目录并启动JBoss以观部署后效。<jndi-name>标签指明的jndi名必须是JBossUtilAutoNumber,因为AutoNumberFactory就是通过这个名字来调用AutoNumberBean的本地引用的。 AutoNumber的调用 
    AutoNumber这个cmp bean是通过AutoNumberFactory这个工厂类的静态类方法来调用的,调用方法很简单,假如你有一张表叫table1,里头有一整数类型的列充当了主键,那么可以将这个键值集合命名成“table1_pk”,通过AutoNumberFactory. getNextInteger(“table1_pk”)就可以获得表table1的当前一个Integer类型的键值。如果是第一次调用,那么获得的值为0,通过这个工厂类来调用AutoNumber不用做任何初始化工作,在调用的CMP类文件中要写这样一个import语句:import org.jboss.varia.autonumber.AutoNumberFactory; 此外AutoNumber还提供了键值初始化和reset的方法,详细参考AutoNumberFactory的源代码文件。AutoNumberFactory主要用在cmp bean的ejbCreate方法中,如: Integer myPk=AutoNumberFactory. getNextInteger(“table1_pk”); setId(myPk);   AutoNumber的原理 
    AutoNumber插件的核心是一个CMP bean,并且使用了工厂方式来生成自动增长键,不过这个工厂模式是建立在数据库上,可以同时为多个表生成键值。每次调用getNextInteger的时候,AutoNumberFactory通过JNDI名获得AutoNumber bean的本地接口,调用bean提供的方法,该方法要做的就是返回当前数据库中的键值,而后将键值加一。   小结 
    AutoNumber相当于开辟了另一条路来方便用户获取一个整数类型的键值,并确保通过该路径获取的键值能够保持唯一性。但是这种绕开数据库的方式有着很大的问题,因为它不是数据库自身来维护的,比如你在另一个数据访问程序(如mysql提供的客户端)来为一个表增加一条记录,如果你忘记了模拟AutoNumber的形式获得键值,也就是忘了在myauto表中将相应键值加一,那么你的ejb下一次获得的键值将会产生冲突。另外一个缺点就是键值只支持Integer类型。对于第一个缺点,是AutoNumber无法解决的,但是第二个缺点却是可以解决的,那就是创建我们自己的Factory方法,支持各种需要的主键数据类型,AutoNumber给了我们一个很好的设计范例。   在JBoss3.2中实现自动增长键值功能 
    先决条件 
    对于MySql的用户来说,你需要一个更好的引擎来支持更好的功能,为了让3.2版本能够在mysql中完成这个功能实现的一个例子,你首先要做的就是下载一个先进的JDBC引擎, MySQL Connector/J 3,该引擎支持了JDBC3.0,至于原因可参看【部署描述文件】部分。这里稍微引开一个话题,如果你是一个好的java程序员,那么你必须时刻对版本保持高度的敏感性,看看Jbuilder版本更新之快之大,JBoss4.0也在日程当中 ,更高的版本意味着更好的功能,但是,不要遗漏了版本之间的差异,不仅仅是主体的差异,还有联系体的差异(这里是JBoss是主体,mysql jdbc driver是联系体),不然你会头碰南墙欲哭无泪,“为伊消得人憔悴”,云云。闲话少说,将Connector/J 3的驱动jar文件拷贝至%JBOSS_HOME%/ server/default/lib/之下,记得将原有驱动删除。   一个CMP Bean场景 
    为了方便描述how to,使用一个简单的cmp bean作为实例来加以描述。假设有一张表auto_inc_test,只有两个字段id和name,其中id是自动增长类型,sql脚本如下: create table auto_inc_test (   id TINYINT(4) NOT NULL auto_increment,   name varchar(10),   primary key (id) ) type=InnoDB   至于如何设计和实现这个cmp bean,这里就不详加描述,对于Jbuilder而言仅仅是几个drag and drop的回合,我们将重心放在部署文件的配置上面。我们的bean 名称叫做TestAutoIncBean,有一个ejbCreate方法,只接受一个字符类型数据为参数,即name字段,id和name字段在bean中都有get/set方法。   部署描述文件 
    JBoss3.2较之3.0版本又有了极其不同的差异,一个就是对EJB2.0规范的支持,其体现在部署文件描述又变了。在三个描述文件ejb-jar.xml、jboss.xml和jbosscmp-jdbc.xml中,前两个与一般部署无异,最后一个部署文件的描述如下:   Jbosscmp-jdbc.xml描述文件 <!-- $Id: standardjbosscmp-jdbc.xml,v 1.39.2.12 2003/03/02 13:43:33 cooperfbi Exp $ -->   <jbosscmp-jdbc>     <defaults>       <datasource>java:/MySqlDS</datasource>       <datasource-mapping>mySQL</datasource-mapping>       <create-table>false</create-table>       <remove-table>false</remove-table>       <read-only>false</read-only>       <time-out>300</time-out>       <pk-constraint>true</pk-constraint>       <fk-constraint>false</fk-constraint>       <row-locking>false</row-locking>       <preferred-relation-mapping>foreign-key</preferred-relation-mapping>       <read-ahead>       <strategy>on-load</strategy>       <page-size>1000</page-size>       <eager-load-group>*</eager-load-group>       </read-ahead>       <list-cache-max>1000</list-cache-max>         <unknown-pk>          <key-generator-factory>UUIDKeyGeneratorFactory</key-generator-factory>          <unknown-pk-class>java.lang.Integer</unknown-pk-class>          <jdbc-type>INTEGER</jdbc-type>          <sql-type>INTEGER</sql-type>       </unknown-pk>         <entity-command name="mysql-get-generated-keys"/>     </defaults>     <enterprise-beans>         <entity>             <ejb-name>TestAutoInc</ejb-name>             <table-name>auto_inc_test</table-name>             <cmp-field>                 <field-name>id</field-name>                 <column-name>id</column-name>             </cmp-field>             <cmp-field>                 <field-name>name</field-name>                 <column-name>name</column-name>             </cmp-field>         </entity>     </enterprise-beans>     <dependent-value-classes>     </dependent-value-classes> </jbosscmp-jdbc>   这里把与数据库映射的描述部分放在<enterprise-beans>标签外部,意指这些数据库设置对所有entity bean适用,也可以放在<enterprise-beans>内部对单独的entity bean进行指定。<unknown-pk>是JBoss3.2新增的描述标签,用户可以在其中指定<key-generator-factory>,该标签指明获得键值工厂生成器的JNDI名称,这里是UUIDKeyGeneratorFactory,由JBoss服务器自身内部提供的,用户不必关注其中细节,另外用户要指定主键类的类型,这里是Integer type。另外值得一提的是<entity-command>标签,它指明了在创建entity bean的时候要调用哪个辅助插件,这里的值是"mysql-get-generated-keys",当创建entity bean的时候调用org.jboss.ejb.plugins.cmp.jdbc.mysql.JDBCMySQLCreateCommand插件,该插件则调用了Connector /J3提供的接口来处理与主键相关的事务。注意该描述文件的头一句<!-- $Id: standardjbosscmp-jdbc.xml,v 1.39.2.12 2003/03/02 13:43:33 cooperfbi Exp $ -->,虽然只是注释,但是对JBoss的xml解析器而言这个信息非常重要,因为JBoss仍要处理低版本的部署文件,这个id是为了告诉JBoss用什么规则来解析。   总结 
    通过以上两个对auto-increment问题在JBoss不同版本中的解决方案,可以看出J2EE的一些特点,对开发人员而言这些特点相当重要。J2EE的设计思想非常先进,但同时也带来了一定的的复杂度,需要考虑和处理的问题会有相应增加。   JBoss是一个开发源码而且完全免费的应用服务器,其与MySql的搭配可以用“天造一对,地设一双”来形容,对于需要实践锻炼的开发者而言是一个天大的福音,不仅可以免费在上面进行开发,还可以参考其设计思路(如第一节对AutoNumberFactory的设计分析)。JBoss在使用上面透露着二字,那就是“简洁”,但又不失其强大的J2EE引擎功能。 
      

  6.   

    另外可以参考http://www.javaresearch.org/article/showarticle.jsp?column=31&thread=3505将keygen作为一个session bean生成出key供cmp.creat(key)调用。
      

  7.   

    另外,如果你使用的是weblogic,可以注意一下这里在weblogic-cmp-rdbms-jar.xml中设置:
           <automatic-key-generation>
               <generator-type>Oracle</generator-type>
           </automatic-key-generation>
      

  8.   

    好东西,就是比较菜,看不太明白,one by one!