将数据从MySQL表传输到另一个表

时间:2015-10-27 10:41:10

标签: mysql sql migrate

我有一个数据库,让我们在我的计算机上简单地称它为'db',其中有几个表在这些表中有多列和数据。 我有一个使用这个数据库的软件来存储配置元素和其他一些东西。

现在,我发布了我的软件的新版本,只是对数据库进行了少量修改,即某些列可能已添加到表中,或已删除(但没有重命名列)。

我必须保留所有数据,因此我想将其转移到我的数据库的新“版本”。

我的想法:

  • 将'db'重命名为'db_old'。
  • 将新数据库安装为“db_new”,并使用新列中的默认值
  • 对于每个表,获取“db_new”中存在的“db_old”中所有列的列表
  • 使用INSERT INTO ... SELECT将旧的东西放回'db_new'。
  • 删除旧数据库并使用我的新数据库。

你认为它能奏效吗?你有什么简单的解决方案吗?

另外,我绝对不是SQL专家......我试过这个(不看是否已经删除了列):

SELECT 
    GROUP_CONCAT(COLUMN_NAME
        SEPARATOR ',')
INTO @colList FROM
    INFORMATION_SCHEMA.COLUMNS
WHERE
    TABLE_SCHEMA = 'db_old'
        AND TABLE_NAME = 'configuration';
INSERT 
INTO db_new.configuration (SELECT @colList)
SELECT @colList FROM
    db_old.configuration;

但它无法用有效列表替换第二个@colList ......你能帮我解决这个问题吗?

谢谢大家,祝你有个美好的一天!

2 个答案:

答案 0 :(得分:0)

首先应该转储数据库数据库并创建.sql文件。根据您的数据库数据,此文件甚至可以使用GB。此SQL文件将包含所有表以及这些表中的所有数据。我建议你打开并查看文件。 然后,您应该使用这个新创建的文件并使用它将所有数据导入新数据库。它会将所有这些表格,数据放入这个新的数据库中。

以下是如何做到这一点。首先创建SQL文件:

mysqldump -h [SeverIpAddress] -u [UserName] -p[password] YourDbname > db_backup.sql

在远程服务器的情况下使用-h [SeverIpAddress]。如果它在你自己的系统中,你不需要使用它。

然后你应该创建你的新数据库,比方说DB_new。创建后,使用use命令切换到它。

use DB_new

完成后,现在导入我们在使用source命令之前创建的.SQl文件。

source YourSQLFilePath

在您的情况下,source db_backup.sql

答案 1 :(得分:0)

行。如果有人遇到同样的问题,这就是解决方案。

首先,承认你有一个名为'myDatabase'的数据库,你想要“升级”一个名为'myTable'的表,即你想通过添加/删除列来修改表结构,但保留数据。< / p>

第一步是删除外键(如果有)并重命名“myTable”:

USE `myDatabase`;

ALTER TABLE `myTable` DROP FOREIGN KEY `my_fk_constraint`;
ALTER TABLE `myTable` RENAME TO `old_myTable`;

第二步是使用SOURCE导入新的表结构。

SOURCE C:/new_table_structure.sql

第三步是可选的,但如果您的表有很多列,则可能需要这样做:

USE `myDatabase`;    
SET GLOBAL group_concat_max_len = 4294967295;

第四步是存储以下例程:

delimiter //

DROP PROCEDURE IF EXISTS updateConf//

CREATE PROCEDURE updateConf(IN dbName TEXT, IN old_table TEXT, IN new_table TEXT, IN primary_key_name TEXT)
BEGIN

-- get column count in old table
SELECT count(*)  
INTO @colNb 
FROM information_schema.COLUMNS 
WHERE TABLE_SCHEMA = dbName
AND TABLE_NAME = old_table;

-- get string with all column names from old_table
SELECT GROUP_CONCAT(COLUMN_NAME)
INTO @colNames1
FROM INFORMATION_SCHEMA.COLUMNS
 WHERE TABLE_SCHEMA = dbName
AND TABLE_NAME = old_table;
SET @colNames1 = CONCAT(@colNames1, ',');

-- get string with all column names from new_table
SELECT GROUP_CONCAT(COLUMN_NAME)
INTO @colNames2
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_SCHEMA = dbName
AND TABLE_NAME = new_table;

    -- variables initialization
    SET @cpt = 1; -- column number counter
    SET @pos = 1; -- position of column name first char
    SET @vir = 1; -- next comma position
    -- start of loop
    label: LOOP
        IF @cpt <= @colNb THEN
            SET @vir = LOCATE(',',@colNames1,@pos); -- localize next comma
            SET @colName = SUBSTRING(@colNames1, @pos, @vir - @pos); -- get column name
            SET @pos = @vir + 1; -- update next column position
            -- if column is in both tables
            IF FIND_IN_SET(@colName, @colNames2) AND @colName != primary_key_name THEN 
                SET @execut = CONCAT("INSERT INTO ", new_table, " (", primary_key_name, ",", @colName, ") SELECT ", primary_key_name, ",", @colName, " FROM ", old_table, " ON DUPLICATE KEY UPDATE ", new_table, ".", @colName, " = ", old_table, ".", @colName);
                PREPARE stmt FROM @execut;
                EXECUTE stmt;           
            END IF;
            SET @cpt = @cpt + 1; -- counter increment
        -- when all columns parsed
        ELSE 
            LEAVE label; -- end of loop
        END IF;
    END LOOP label;
END //

delimiter ;

最后一步是调用表上的过程,并删除临时表:

CALL updateConf( 'myDatabase', 'old_myTable', 'myTable', 'primaryKeyName' );

DROP TABLE `old_myTable`;

瞧!只是不要忘记放回你丢弃的外键:)

肯定可以用更好的方式完成,但我能让它正常工作。

谢谢大家!