zoukankan      html  css  js  c++  java
  • Mysql根据一个基库生成其他库与其不同的库升级脚本

    今天研究了一下不同数据库之间如何做同步。弄了一个升级工具类,希望以后还能有所帮助。

    public class UpgradeDataBase {
    
        public static void main(String[] args) {
            //两个不同数据库名称
            List<String> sqls = UpgradeDataBaseToStandardDataBase("database1", "database2");
            for(String sql : sqls) {
                System.out.println(sql);
            }
        }
    
        /**
         * 将基准库里有的表字段,以名称为基准,生成同步到目标库中的脚本
         */
        public static List<String> UpgradeDataBaseToStandardDataBase(String standard, String target) {
    
            List<String> upgradeSqls = new ArrayList<>();
    
            List<String> standardTables = new ArrayList<>();
            List<String> targetTables = new ArrayList<>();
    
            Connection connStandard = null;
            Connection connTarget = null;
    
            Statement stmtStandard = null;
            Statement stmtTarget = null;
    
            ResultSet rsStandard = null;
            ResultSet rsTarget = null;
    
            try {
                Properties props = Resources.getResourceAsProperties("jdbc.properties");
                String url = props.getProperty("url");
                String driver = props.getProperty("driver");
                String username = props.getProperty("username");
                String password = props.getProperty("password");
    
                Class.forName(driver).newInstance();
                //拿到2个库的链接
                connStandard = DriverManager.getConnection(url, username, password);
                connStandard.setCatalog(standard);
    
                connTarget = DriverManager.getConnection(url, username, password);
                connTarget.setCatalog(target);
    
    
                stmtStandard = connStandard.createStatement();
                stmtTarget = connTarget.createStatement();
    
                //先把2个库所有的表查出来
                stmtStandard.execute("show tables");
                stmtTarget.execute("show tables");
    
                rsStandard = stmtStandard.getResultSet();
                rsTarget = stmtTarget.getResultSet();
    
                while (rsStandard.next()){
                    standardTables.add(rsStandard.getString(1));
                }
    
                while (rsTarget.next()){
                    targetTables.add(rsTarget.getString(1));
                }
                //循环基准库中每一张表
                for(String table : standardTables) {
                    if("SM_USER".equals(table)) {
                        continue;
                    }
                    if(targetTables.contains(table)) {
                        Map<String, Map<String, String>> standardColumns = new HashMap<>();
                        Map<String, Map<String, String>> targetColumns = new HashMap<>();
                        //检查每一个字段,
                        //1.首先先查出目标库和基准库该表的所有字段
                        stmtStandard.execute("show columns from " + table + " from " + standard);
                        rsStandard = stmtStandard.getResultSet();
                        while (rsStandard.next()){
    
                            Map<String, String> map = new HashMap<>();
                            map.put("Field", rsStandard.getString("Field"));//列名
                            map.put("Type", rsStandard.getString("Type"));//类型+长度
                            map.put("Null", rsStandard.getString("Null"));//是否可为空
                            map.put("Key", rsStandard.getString("Key"));//是否主键
                            map.put("Default", rsStandard.getString("Default"));//默认值
                            map.put("Extra", rsStandard.getString("Extra"));//其他(自增列,触发器等)
                            standardColumns.put(rsStandard.getString("Field"), map);
                        }
    
                        stmtTarget.execute("show columns from " + table + " from " + target);
                        rsTarget = stmtTarget.getResultSet();
                        while (rsTarget.next()){
    
                            Map<String, String> map = new HashMap<>();
                            map.put("Field", rsTarget.getString("Field"));//列名
                            map.put("Type", rsTarget.getString("Type"));//类型+长度
                            map.put("Null", rsTarget.getString("Null"));//是否可为空
                            map.put("Key", rsTarget.getString("Key"));//是否主键
                            map.put("Default", rsTarget.getString("Default"));//默认值
                            map.put("Extra", rsTarget.getString("Extra"));//其他(自增列,触发器等)
                            targetColumns.put(rsTarget.getString("Field"), map);
                        }
    
                        //2.以基准库为准,逐个列比较
                        //TODO 没有处理Key(没有做主键、自增处理)
                        for(String column : standardColumns.keySet()) {
                            if(targetColumns.containsKey(column)) {//存在这一列
                                boolean needGeneSql = false;
                                StringBuffer buffer = new StringBuffer();
                                //类型有变化, 但是不管类型有没有变化,后续的语句都需要
    //                            if(standardColumns.get(column).get("Type") != null && !standardColumns.get(column).get("Type").equals(targetColumns.get(column).get("Type"))) {
    //                                buffer.append(standardColumns.get(column).get("Type"));
    //                            }
                                buffer.append(standardColumns.get(column).get("Type"));
                                //默认值有变
                                if(standardColumns.get(column).get("Default") != null && !standardColumns.get(column).get("Default").equals(targetColumns.get(column).get("Default"))) {
                                    buffer.append(" default " + standardColumns.get(column).get("Default"));
                                    needGeneSql = true;
                                }
                                //是否可空有变
                                if(standardColumns.get(column).get("Null") != null && !standardColumns.get(column).get("Null").equals(targetColumns.get(column).get("Null"))) {
                                    buffer.append(("NO".equals(standardColumns.get(column).get("Null")) ? " not null " : " null "));
                                    needGeneSql = true;
                                }
                                //处理自增长等
                                if(standardColumns.get(column).get("Extra") != null && !standardColumns.get(column).get("Extra").equals(targetColumns.get(column).get("Extra"))) {
                                    buffer.append(" ").append(standardColumns.get(column).get("Extra"));
                                    needGeneSql = true;
                                }
                                if(needGeneSql) {
                                    String changeColumnSql = "alter table " + table + " change " + column + " " + column + " " + buffer.toString() + ";";
                                    upgradeSqls.add(changeColumnSql);
                                }
                            }
                            else{
                                String addColumnSql = "alter table " + table +
                                        " add column " + column + " " +
                                        standardColumns.get(column).get("Type") + " default " + standardColumns.get(column).get("Default") +
                                        ("NO".equals(standardColumns.get(column).get("Null")) ? " not null " : " null ") + ";";
                                upgradeSqls.add(addColumnSql);
                            }
                        }
                    }
                    else{//目标库中,没有基准库的表
                        stmtStandard.execute("show create table " + table);
                        rsStandard = stmtStandard.getResultSet();
                        String createSql = null;
                        while (rsStandard.next()){
                            //第2列是建表语句
                            createSql = rsStandard.getString(2);
                            upgradeSqls.add(createSql + ";");
                        }
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                try {
                    rsStandard.close();
                    rsTarget.close();
                    stmtStandard.close();
                    stmtTarget.close();
                    connStandard.close();
                    connTarget.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
    
            }
            return upgradeSqls;
    
        }
    }
    

      

      

  • 相关阅读:
    前端开发者应该知道的 CSS 小技巧
    css3制作六边形图片
    css3 绘制优惠券
    flex css 布局
    js 微信分享
    JS判断移动设备最佳方法 并实现跳转至手机版网页
    ajax 提交成功页面跳转问题
    css相关tips
    无阻塞加载和defer、async
    常用排序算法
  • 原文地址:https://www.cnblogs.com/owenma/p/7016596.html
Copyright © 2011-2022 走看看