Spring和Hibernate動態建表及動態加載映射文件(無需Session factory...

         J.Office2有一功能是工做流支持動態表單設計,設計後能夠動態生成數據庫表,而且支持實時查詢(單表及多表都可)。java

 

         因爲J.Office2版本中採用了Hibernate做爲底層的ORM框架,結合Spring框架,Spring容器啓動後,SessionFactory就會被注入到各個業務的Dao層中去。sql

 

        動態建表功能比較容易實現,咱們能夠new一個SessionFactory,而後把它的配置屬性hibernate.hbm2ddl.auto改成update或create,就能夠達到動態修改表結構的效果。數據庫

 

            但若要加入新的hbm或class,須要從新調用SessionFactoryBean來獲取一個全新的SessionFactory,這種方案試過了, 效果並不理想。從新加載,會致使大量的hbm或class文件從新加載,實在有點慢。而且嚴重影響如今注入SessionFactory的Dao。若 Dao採用動態構建SessionFactory,性能又是一問題。而Hibernate沒有提供SessionFactory動態加入hbm或 Class文件。因此實在機關用盡。session

 

      因此最終仍是回到如何擴展Hibernate的SessionFactory類中去了,這想法已經有很多開發人員嘗試過,JE也有一帖子專門討論這個。不 過僅是一Demo,不完善。咱們提供了兩個擴展的類(修改Hibernate中的兩類,使其支持動態加入配置文件,而且能實時查詢。app

 

咱們僅須要修改兩個類,一個是Configuration,在其裏面加一方法,以下:框架

     public void doComplie(){
         secondPassCompile();
     }eclipse

 

     修改ide

 

在SessonFactoryImpl類中加入如下方法,(有一些變量值不能修改的,請改成可修改)post

 

Java代碼   收藏代碼
  1. //add by csx  
  2. public void addNewConfig(Configuration cfg){  
  3.     log.info("add NewConfig.....");  
  4.       
  5.     Mapping mapping=this.configuration.getMapping();  
  6.     this.filters.putAll( cfg.getFilterDefinitions() );  
  7.     //Generators:  
  8.     Iterator classes = cfg.getClassMappings();  
  9.     while ( classes.hasNext() ) {  
  10.         PersistentClass model = (PersistentClass) classes.next();  
  11.           
  12.         if ( !model.isInherited() ) {  
  13.             IdentifierGenerator generator = model.getIdentifier().createIdentifierGenerator(  
  14.                     settings.getDialect(),  
  15.                     settings.getDefaultCatalogName(),  
  16.                     settings.getDefaultSchemaName(),  
  17.                     (RootClass) model  
  18.                 );  
  19.             identifierGenerators.put( model.getEntityName(), generator );  
  20.         }  
  21.     }  
  22.   
  23.     ///////////////////////////////////////////////////////////////////////  
  24.     // Prepare persisters and link them up with their cache  
  25.     // region/access-strategy  
  26.   
  27.     String cacheRegionPrefix = settings.getCacheRegionPrefix() == null ? "" : settings.getCacheRegionPrefix() + ".";  
  28.       
  29.     Map entityAccessStrategies = new HashMap();  
  30.     Map tmpEntityPersisters=new HashMap();  
  31.     Map tmpClassMetadata=new HashMap();  
  32.       
  33.       
  34.     this.configuration.getClassMap().putAll(cfg.getClassMap());  
  35.     classes = cfg.getClassMappings();  
  36.   
  37.     while ( classes.hasNext() ) {  
  38.           
  39.         PersistentClass model = (PersistentClass) classes.next();  
  40.           
  41.         model.prepareTemporaryTables( mapping, settings.getDialect() );  
  42.         String cacheRegionName = cacheRegionPrefix + model.getRootClass().getCacheRegionName();  
  43.           
  44.         // cache region is defined by the root-class in the hierarchy...  
  45.         EntityRegionAccessStrategy accessStrategy = ( EntityRegionAccessStrategy ) entityAccessStrategies.get( cacheRegionName );  
  46.         if ( accessStrategy == null && settings.isSecondLevelCacheEnabled() ) {  
  47.             AccessType accessType = AccessType.parse( model.getCacheConcurrencyStrategy() );  
  48.             if ( accessType != null ) {  
  49.                   
  50.                 log.trace( "Building cache for entity data [" + model.getEntityName() + "]" );  
  51.                 EntityRegion entityRegion = settings.getRegionFactory().buildEntityRegion( cacheRegionName, properties, CacheDataDescriptionImpl.decode( model ) );  
  52.                 accessStrategy = entityRegion.buildAccessStrategy( accessType );  
  53.                 entityAccessStrategies.put( cacheRegionName, accessStrategy );  
  54.                 allCacheRegions.put( cacheRegionName, entityRegion );  
  55.             }  
  56.         }  
  57.         EntityPersister cp = PersisterFactory.createClassPersister( model, accessStrategy, this, cfg.getMapping() );  
  58.         tmpEntityPersisters.put( model.getEntityName(), cp );  
  59.         tmpClassMetadata.put( model.getEntityName(), cp.getClassMetadata() );  
  60.           
  61.     }  
  62.       
  63.     //Named Queries:  
  64.     namedQueries.putAll(cfg.getNamedQueries());  
  65.     namedSqlQueries.putAll( cfg.getNamedSQLQueries() );  
  66.     sqlResultSetMappings.putAll(cfg.getSqlResultSetMappings());  
  67.     imports.putAll(cfg.getImports());  
  68.   
  69.     entityPersisters.putAll(tmpEntityPersisters);  
  70.   
  71.     classMetadata.putAll(tmpClassMetadata);  
  72.       
  73.     Map tmpEntityToCollectionRoleMap = new HashMap();  
  74.     Map tempCollectionPersisters=new HashMap();  
  75.       
  76.     this.configuration.getCollectionMap().putAll(cfg.getCollectionMap());  
  77.     Iterator collections = cfg.getCollectionMappings();  
  78.   
  79.     while ( collections.hasNext() ) {  
  80.         Collection model = (Collection) collections.next();  
  81.         final String cacheRegionName = cacheRegionPrefix + model.getCacheRegionName();  
  82.         final AccessType accessType = AccessType.parse( model.getCacheConcurrencyStrategy() );  
  83.         CollectionRegionAccessStrategy accessStrategy = null;  
  84.         if ( accessType != null && settings.isSecondLevelCacheEnabled() ) {  
  85.             log.trace( "Building cache for collection data [" + model.getRole() + "]" );  
  86.             CollectionRegion collectionRegion = settings.getRegionFactory().buildCollectionRegion( cacheRegionName, properties, CacheDataDescriptionImpl.decode( model ) );  
  87.             accessStrategy = collectionRegion.buildAccessStrategy( accessType );  
  88.             entityAccessStrategies.put( cacheRegionName, accessStrategy );  
  89.             allCacheRegions.put( cacheRegionName, collectionRegion );  
  90.         }  
  91.         CollectionPersister persister = PersisterFactory.createCollectionPersister(this.getConfiguration(), model, accessStrategy, this) ;  
  92.         tempCollectionPersisters.put( model.getRole(), persister.getCollectionMetadata() );  
  93.         Type indexType = persister.getIndexType();  
  94.         if ( indexType != null && indexType.isAssociationType() && !indexType.isAnyType() ) {  
  95.             String entityName = ( ( AssociationType ) indexType ).getAssociatedEntityName( this );  
  96.             Set roles = ( Set ) tmpEntityToCollectionRoleMap.get( entityName );  
  97.             if ( roles == null ) {  
  98.                 roles = new HashSet();  
  99.                 tmpEntityToCollectionRoleMap.put( entityName, roles );  
  100.             }  
  101.             roles.add( persister.getRole() );  
  102.         }  
  103.         Type elementType = persister.getElementType();  
  104.         if ( elementType.isAssociationType() && !elementType.isAnyType() ) {  
  105.             String entityName = ( ( AssociationType ) elementType ).getAssociatedEntityName( this );  
  106.             Set roles = ( Set ) tmpEntityToCollectionRoleMap.get( entityName );  
  107.             if ( roles == null ) {  
  108.                 roles = new HashSet();  
  109.                 tmpEntityToCollectionRoleMap.put( entityName, roles );  
  110.             }  
  111.             roles.add( persister.getRole() );  
  112.         }  
  113.           
  114.     }  
  115.     //加入新的  
  116.     collectionPersisters.putAll(tempCollectionPersisters);  
  117.   
  118.     Iterator itr = tmpEntityToCollectionRoleMap.entrySet().iterator();  
  119.     while ( itr.hasNext() ) {  
  120.         final Map.Entry entry = ( Map.Entry ) itr.next();  
  121.         entry.setValue( Collections.unmodifiableSet( ( Set ) entry.getValue() ) );  
  122.     }  
  123.       
  124.     collectionRolesByEntityParticipant.putAll( tmpEntityToCollectionRoleMap);  
  125.   
  126.     // after *all* persisters and named queries are registered  
  127.     Iterator iter = tmpEntityPersisters.values().iterator();  
  128.     while ( iter.hasNext() ) {  
  129.         ( (EntityPersister) iter.next() ).postInstantiate();  
  130.     }  
  131.     iter = tempCollectionPersisters.values().iterator();  
  132.     while ( iter.hasNext() ) {  
  133.         ( (CollectionPersister) iter.next() ).postInstantiate();  
  134.     }  
  135.       
  136.     queryPlanCache=new QueryPlanCache(this);  
  137.       
  138.       
  139. }  
 

咱們動態加入實體,動態可進行查詢,Hibernate提供幾種實體的查詢策略,其中一個是咱們經常使用的pojo,若採用該方法,咱們得加上hbm與 class類或僅是含註解的class至hibernate的SessionFactory,這種方案而且沒有問題,但會遇到當咱們修改表單字段,從新生 成對應的實體時,咱們就會遇到原有的實體class不能在ClassLoader裏卸載。使用的仍是舊的Class,達不到動態查詢的效果。若使用動態的 ClassLoader,代碼將變得很複雜。(嘗試過,代碼相對比較繁雜)性能

 

        還好Hibernate提供了另外一種實體查詢策略,基於Map的動態實體。基於這種方式,咱們配置一個一對多的表,其示例代碼以下:

 

MainEntity.hbm.xml

Java代碼   收藏代碼
  1. <?xml version="1.0"?>  
  2. <!DOCTYPE hibernate-mapping PUBLIC  
  3.         "-//Hibernate/Hibernate Mapping DTD 3.0//EN"  
  4.         "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">  
  5. <hibernate-mapping>  
  6.     <class table="main_entity" entity-name="MainEntity">  
  7.         <id name="mainId" column="mainId" type="java.lang.Long">  
  8.             <generator class="native"/>  
  9.         </id>  
  10.         <property name="itemSubject" type="java.lang.String" length="128"/>  
  11.         <property name="itemDescp" type="java.lang.String" />  
  12.         <property name="createtime" type="java.util.Date" />  
  13.         <bag name="subEntitys"  
  14.              table="sub_entity"  
  15.              lazy="false"  
  16.              inverse="true"  
  17.              cascade="save-update,delete-orphan"  
  18.         >  
  19.             <key>  
  20.                 <column name="mainId"/>  
  21.             </key>  
  22.             <one-to-many entity-name="SubEntity"/>  
  23.         </bag>  
  24.     </class>  
  25. </hibernate-mapping>  

 SubEntity.hbm.xml

 

 

Java代碼   收藏代碼
  1. <?xml version="1.0"?>  
  2. <!DOCTYPE hibernate-mapping PUBLIC  
  3.         "-//Hibernate/Hibernate Mapping DTD 3.0//EN"  
  4.         "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">  
  5. <hibernate-mapping>  
  6.     <class table="sub_entity" entity-name="SubEntity">  
  7.         <id name="subId" column="subId" type="java.lang.Long">  
  8.             <generator class="native"/>  
  9.         </id>  
  10.         <property name="subject" type="java.lang.String" length="128"/>  
  11.         <property name="createtime" type="java.util.Date"/>  
  12.         <many-to-one name="mainEntity" entity-name="MainEntity" not-null="false" fetch="select">  
  13.             <column name="mainId"></column>  
  14.         </many-to-one>  
  15.     </class>  
  16. </hibernate-mapping>  
 

爲兩實體插入數據後,可動態測試以下:

 

 

Java代碼   收藏代碼
  1. public static void main(String[]args){  
  2.         ClassPathXmlApplicationContext ctx=new ClassPathXmlApplicationContext("classpath:app-context.xml");  
  3.   
  4.         MyAnnotationSessionFactoryBean sessionFactoryBean=(MyAnnotationSessionFactoryBean)ctx.getBean("&sessionFactory");  
  5.           
  6.         SessionFactoryImpl sessionFactoryImpl=(SessionFactoryImpl)sessionFactoryBean.getObject();  
  7.       
  8.         Configuration cfg=new Configuration();    
  9.         //cfg.configure();  
  10.           
  11.         cfg.addFile("D:/download/eclipse/workspace2/SpringHibernate/src/com/hotent/entity/MainEntity.hbm.xml");  
  12.         cfg.addFile("D:/download/eclipse/workspace2/SpringHibernate/src/com/hotent/entity/SubEntity.hbm.xml");  
  13.         cfg.doComplie();  
  14.   
  15.         sessionFactoryImpl.addNewConfig(cfg);  
  16.   
  17.         MainEntityDao dao=(MainEntityDao)ctx.getBean("mainEntityDao");  
  18.   
  19.         List list=dao.query();  
  20.   
  21.         for(int i=0;i<list.size();i++){  
  22.             Map map=(Map)list.get(i);  
  23.             Iterator it=map.keySet().iterator();  
  24.             while(it.hasNext()){  
  25.                 //String key=(String)it.next();  
  26.                 Object key=it.next();  
  27.                 Object val=map.get(key);  
  28.                 System.out.println("--------------->key:"+key );  
  29.             }  
  30.         }  
  31.   
  32.         Configuration cfg2=new Configuration();  
  33.         cfg2.addFile("D:/dev/product/SpringHibernate/src/com/hotent/entity/MainEntity2.hbm.xml");  
  34.         cfg2.addFile("D:/dev/product/SpringHibernate/src/com/hotent/entity/SubEntity2.hbm.xml");  
  35.         cfg2.doComplie();  
  36.         sessionFactoryImpl.addNewConfig(cfg2);  
  37.           
  38.         List list2=dao.query();  
  39.           
  40.         for(int i=0;i<list2.size();i++){  
  41.             Map map=(Map)list2.get(i);  
  42.             Iterator it=map.keySet().iterator();  
  43.             while(it.hasNext()){  
  44.                 //String key=(String)it.next();  
  45.                 Object key=it.next();  
  46.                 Object val=map.get(key);  
  47.                 System.out.println("key:"+key );  
  48.             }  
  49.         }  
  50.           
  51.     }  
 
Java代碼   收藏代碼
  1. MainEntity2.hbm.xml與SubEntity2.hbm.xml文件相對原來的文件增長了一些列或刪除了一些列  

 

執行後,能夠實時看到不一樣的結果,而且不會影響現有的dao。

相關文章
相關標籤/搜索