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;
    }
}
Mysql根据一个基库生成其他库与其不同的库升级脚本的更多相关文章
- Linux静态库生成指南
		Linux静态库生成指南 Linux上的静态库,其实是目标文件的归档文件.在Linux上创建静态库的步骤如下: 写源文件,通过 gcc -c xxx.c 生成目标文件. 用 ar 归档目标文件,生 ... 
- iOS 静态库生成(引用第三方SDK、开源库、资源包)
		一.静态库创建 打开Xcode, 选择File ----> New ---> Project 选择iOS ----> Framework & Library ---> ... 
- QRCode.js一个生成二维码的javascript库
		前言 最近在开发中遇到一个需求:将后端返回的链接转换成二维码,那么如何来实现呢?我们可以使用QRCode.js来解决这一问题 什么是 QRCode.js? QRCode.js 是一个用于生成二维码的 ... 
- Linux下动态库生成和使用
		Linux下动态库生成和使用 一.动态库的基本概念 1.动态链接库是程序运行时加载的库,当动态链接库正确安装后,所有的程序都可以使用动态库来运行程序.动态链接库是目标文件的集合,目标文件在动态链接库中 ... 
- Linux下静态库生成和使用
		Linux下静态库生成和使用 一.静态库概念 1.库是预编译的目标文件(object files)的集合,它们可以被链接进程序.静态库以后缀为”.a”的特殊的存档(archive file)存储. ... 
- PHP5 GD库生成图形验证码(汉字)
		PHP5 GD库生成图形验证码且带有汉字的实例分享. 1,利用GD库函数生成图片,并在图片上写指定字符imagecreatetruecolor 新建一个真彩色图像imagecolorallocate ... 
- Windows环境下编译Assimp库生成Android可用的.so或.a文件
		在做项目过程中需要使用Assimp这个3D模型读取库来读取obj格式的模型,因为项目是基于Android平台,采用NDK开发,所以就打算编译Assimp库并生成.so文件.本文使用Assimp-v.5 ... 
- Linux动态库生成与使用指南
		相关阅读: Linux静态库生成指南 Linux下动态库文件的文件名形如 libxxx.so,其中so是 Shared Object 的缩写,即可以共享的目标文件. 在链接动态库生成可执行文件时,并不 ... 
- PHP验证码生成及图片处理(GD库)
		GD库是php处理图形的扩展库,GD库提供了一系列用来处理图片的API,使用GD库可以处理图片,或者生成图片,也可以给图片加水印. 本章实现了生成图片并绘画各种形状.图片的压缩.中文字符水印及图片水印 ... 
随机推荐
- 说说cglib动态代理
			前言 jdk中的动态代理通过反射类Proxy和InvocationHandler回调接口实现,要求委托类必须实现一个接口,只能对该类接口中定义的方法实现代理,这在实际编程中有一定的局限性. cglib ... 
- [SQL case when的两种用法]
			当我们需要从数据源上 直接判断数据显示代表的含义的时候 ,就可以在SQL语句中使用 Case When这个函数了. Case具有两种格式.简单Case函数和Case搜索函数. 第一种 格式 : 简单C ... 
- jdk的server模式修改无效(关于client和server模式)
			本机为64位操作系统,64位jdk,win10. 修改C:\Program Files\Java\jre8\lib\amd64\jvm.cfg无效. 我的文件的内容为: 原因参考如下: http:// ... 
- IT技术有感
			今天看技术文章,spring相关的,某一个点以前每次看一直不理解, 可是不知道为什么隔了1年左右,中间什么都没做,现在却都懂了. 在看懂的那一刻,笼罩在我心上的躁动突然平静了许多,我的心这一年来前所未 ... 
- django Form组件 上传文件
			上传文件 注意:FORM表单提交文件要有一个参数enctype="multipart/form-data" 普通上传: urls: url(r'^f1/',views.f1), u ... 
- python列表基础操作
			Python列表基本操作 记住一句话,叫做顾首不顾尾 首先我们来定义一个列表 name = ["jixuege","dajiba","boduoye& ... 
- POJ-1251 Jungle Roads---MST裸题(需要编号)
			题目链接: https://vjudge.net/problem/POJ-1251 题目大意: 首先给你一个图,需要你求出最小生成树,输入N个节点,用大写字母表示了节点,然后节点与节点之间有权值. 思 ... 
- codewars.DNA题目几种解法分析(字符串替换)
			题干: 意思就是字符串替换,"A"与"C"配对,"T"与"G"配对,DNA不为空. 解法一:我的解法,用for循环遍历字 ... 
- Azure AI 服务之文本翻译
			当下人工智能可谓是风头正劲,几乎所有的大厂都有相关的技术栈.微软在 AI 领域自然也是投入了重注,并且以 Azure 认知服务的方式投入了市场: 也就是说作为开发者我们不需要学习太多 AI 的理论知识 ... 
- JSONObject和JSONArray区别及基本用法
			一.JSONObject和JSONArray的数据表示形式 JSONObject的数据是用 { } 来表示的, 例如: { "id" : "123", & ... 
