Hibernate 中的SchemaUpdate工具源码分析使用

//导出数据库本对象的应用
package org.hibernate.tool.hbm2ddl;

import java.io.FileInputStream;
import java.io.FileWriter;
import java.io.Writer;
import java.sql.Connection;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;

import org.hibernate.HibernateException;
import org.hibernate.JDBCException;
import org.hibernate.cfg.Configuration;
import org.hibernate.cfg.Environment;
import org.hibernate.cfg.NamingStrategy;
import org.hibernate.cfg.Settings;
import org.hibernate.dialect.Dialect;
import org.hibernate.jdbc.util.FormatStyle;
import org.hibernate.jdbc.util.Formatter;
import org.hibernate.jdbc.util.SQLStatementLogger;
import org.hibernate.util.PropertiesHelper;
import org.hibernate.util.ReflectHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A commandline tool to update a database schema. May also be called from
 * inside an application.
 *
 * @author Christoph Sturm
 */
public class SchemaUpdate {

 private static final Logger log = LoggerFactory.getLogger( SchemaUpdate.class );
 private ConnectionHelper connectionHelper;
 private Configuration configuration;
 private Dialect dialect;
 private List exceptions;
 private boolean haltOnError = false;
 private boolean format = true;
 private String outputFile = null;
 private String delimiter;
 private Formatter formatter;
 private SQLStatementLogger sqlStatementLogger;

 public SchemaUpdate(Configuration cfg) throws HibernateException {
  this( cfg, cfg.getProperties() );
 }

 public SchemaUpdate(Configuration cfg, Properties connectionProperties) throws HibernateException {
  this.configuration = cfg;
  dialect = Dialect.getDialect( connectionProperties );
  Properties props = new Properties();
  props.putAll( dialect.getDefaultProperties() );
  props.putAll( connectionProperties );
  connectionHelper = new ManagedProviderConnectionHelper( props );
  exceptions = new ArrayList();
  formatter = ( PropertiesHelper.getBoolean( Environment.FORMAT_SQL, props ) ? FormatStyle.DDL : FormatStyle.NONE ).getFormatter();
 }

 public SchemaUpdate(Configuration cfg, Settings settings) throws HibernateException {
  this.configuration = cfg;
  dialect = settings.getDialect();
  connectionHelper = new SuppliedConnectionProviderConnectionHelper(
    settings.getConnectionProvider()
  );
  exceptions = new ArrayList();
  sqlStatementLogger = settings.getSqlStatementLogger();
  formatter = ( sqlStatementLogger.isFormatSql() ? FormatStyle.DDL : FormatStyle.NONE ).getFormatter();
 }

 public static void main(String[] args) {
  try {
   Configuration cfg = new Configuration();

   boolean script = true;
   // If true then execute db updates, otherwise just generate and display updates
   boolean doUpdate = true;
   String propFile = null;

   for ( int i = 0; i < args.length; i++ ) {
    if ( args[i].startsWith( "--" ) ) {
     if ( args[i].equals( "--quiet" ) ) {
      script = false;
     }
     else if ( args[i].startsWith( "--properties=" ) ) {
      propFile = args[i].substring( 13 );
     }
     else if ( args[i].startsWith( "--config=" ) ) {
      cfg.configure( args[i].substring( 9 ) );
     }
     else if ( args[i].startsWith( "--text" ) ) {
      doUpdate = false;
     }
     else if ( args[i].startsWith( "--naming=" ) ) {
      cfg.setNamingStrategy(
        ( NamingStrategy ) ReflectHelper.classForName( args[i].substring( 9 ) ).newInstance()
      );
     }
    }
    else {
     cfg.addFile( args[i] );
    }

   }

   if ( propFile != null ) {
    Properties props = new Properties();
    props.putAll( cfg.getProperties() );
    props.load( new FileInputStream( propFile ) );
    cfg.setProperties( props );
   }

   new SchemaUpdate( cfg ).execute( script, doUpdate );
  }
  catch ( Exception e ) {
   log.error( "Error running schema update", e );
   e.printStackTrace();
  }
 }

 /**
  * Execute the schema updates
  *
  * @param script print all DDL to the console
  */
 public void execute(boolean script, boolean doUpdate) {

  log.info( "Running hbm2ddl schema update" );

  Connection connection = null;
  Statement stmt = null;
  Writer outputFileWriter = null;

  exceptions.clear();

  try {

   DatabaseMetadata meta;
   try {
    log.info( "fetching database metadata" );
    connectionHelper.prepare( true );
    connection = connectionHelper.getConnection();
    meta = new DatabaseMetadata( connection, dialect );
    stmt = connection.createStatement();
   }
   catch ( SQLException sqle ) {
    exceptions.add( sqle );
    log.error( "could not get database metadata", sqle );
    throw sqle;
   }

   log.info( "updating schema" );

   
   if ( outputFile != null ) {
    log.info( "writing generated schema to file: " + outputFile );
    outputFileWriter = new FileWriter( outputFile );
   }
   
   String[] createSQL = configuration.generateSchemaUpdateScript( dialect, meta );
   for ( int j = 0; j < createSQL.length; j++ ) {

    final String sql = createSQL[j];
    String formatted = formatter.format( sql );
    try {
     if ( delimiter != null ) {
      formatted += delimiter;
     }
     if ( script ) {
      System.out.println( formatted );
     }
     if ( outputFile != null ) {
      outputFileWriter.write( formatted + "\n" );
     }
     if ( doUpdate ) {
      log.debug( sql );
      stmt.executeUpdate( formatted );
     }
    }
    catch ( SQLException e ) {
     if ( haltOnError ) {
      throw new JDBCException( "Error during DDL export", e );
     }
     exceptions.add( e );
     log.error( "Unsuccessful: " + sql );
     log.error( e.getMessage() );
    }
   }

   log.info( "schema update complete" );

  }
  catch ( Exception e ) {
   exceptions.add( e );
   log.error( "could not complete schema update", e );
  }
  finally {

   try {
    if ( stmt != null ) {
     stmt.close();
    }
    connectionHelper.release();
   }
   catch ( Exception e ) {
    exceptions.add( e );
    log.error( "Error closing connection", e );
   }
   try {
    if( outputFileWriter != null ) {
     outputFileWriter.close();
    }
   }
   catch(Exception e) {
    exceptions.add(e);
    log.error( "Error closing connection", e );
   }
  }
 }

 /**
  * Returns a List of all Exceptions which occured during the export.
  *
  * @return A List containig the Exceptions occured during the export
  */
 public List getExceptions() {
  return exceptions;
 }

 public void setHaltOnError(boolean haltOnError) {
  this.haltOnError = haltOnError;
 }

 public void setFormat(boolean format) {
  this.formatter = ( format ? FormatStyle.DDL : FormatStyle.NONE ).getFormatter();
 }

 public void setOutputFile(String outputFile) {
  this.outputFile = outputFile;
 }

 public void setDelimiter(String delimiter) {
  this.delimiter = delimiter;
 }

}

设置适当的方法实现相应的功能

 

测试代码

package org.hibernate.test.schemaupdate;

import junit.framework.Test;
import junit.framework.TestSuite;

import org.hibernate.cfg.Configuration;
import org.hibernate.junit.UnitTestCase;
import org.hibernate.tool.hbm2ddl.SchemaExport;
import org.hibernate.tool.hbm2ddl.SchemaUpdate;

/**
 *  */
public class MigrationTest extends UnitTestCase {

 public MigrationTest(String str) {
  super( str );
 }

 public static Test suite() {
  return new TestSuite( MigrationTest.class );
 }

 public void testSimpleColumnAddition() {
  String resource1 = "org/hibernate/test/schemaupdate/1_Version.hbm.xml";
  String resource2 = "org/hibernate/test/schemaupdate/2_Version.hbm.xml";

  Configuration v1cfg = new Configuration();
  v1cfg.addResource( resource1 );
  new SchemaExport( v1cfg ).execute( false, true, true, false );

  SchemaUpdate v1schemaUpdate = new SchemaUpdate( v1cfg );
  v1schemaUpdate.execute( true, true );

  assertEquals( 0, v1schemaUpdate.getExceptions().size() );

  Configuration v2cfg = new Configuration();
  v2cfg.addResource( resource2 );

  SchemaUpdate v2schemaUpdate = new SchemaUpdate( v2cfg );
  v2schemaUpdate.execute( true, true );
  assertEquals( 0, v2schemaUpdate.getExceptions().size() );

 }

}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值