How to copy a row and insert in same table with a autoincrement field in MySQL?

后端 未结 13 1002
旧时难觅i
旧时难觅i 2020-11-29 15:55

In MySQL I am trying to copy a row with an autoincrement column ID=1 and insert the data into same table as a new row with

相关标签:
13条回答
  • 2020-11-29 16:41

    I was looking for the same feature but I don't use MySQL. I wanted to copy ALL the fields except of course the primary key (id). This was a one shot query, not to be used in any script or code.

    I found my way around with PL/SQL but I'm sure any other SQL IDE would do. I did a basic

    SELECT * 
    FROM mytable 
    WHERE id=42;
    

    Then export it to a SQL file where I could find the

    INSERT INTO table (col1, col2, col3, ... , col42) 
    VALUES (1, 2, 3, ..., 42);
    

    I just edited it and used it :

    INSERT INTO table (col1, col2, col3, ... , col42) 
    VALUES (mysequence.nextval, 2, 3, ..., 42);
    
    0 讨论(0)
  • 2020-11-29 16:45
    INSERT INTO `dbMyDataBase`.`tblMyTable` 
    (
        `IdAutoincrement`, 
        `Column2`, 
        `Column3`, 
        `Column4` 
    ) 
    
    SELECT 
        NULL,  
        `Column2`, 
        `Column3`, 
        'CustomValue' AS Column4 
    FROM `dbMyDataBase`.`tblMyTable` 
    WHERE `tblMyTable`.`Column2` = 'UniqueValueOfTheKey' 
    ; 
    /* mySQL 5.6 */
    
    0 讨论(0)
  • 2020-11-29 16:47

    IMO, the best seems to use sql statements only to copy that row, while at the same time only referencing the columns you must and want to change.

    CREATE TEMPORARY TABLE temp_table ENGINE=MEMORY
    
    SELECT * FROM your_table WHERE id=1;
    UPDATE temp_table SET id=NULL; /* Update other values at will. */
    
    INSERT INTO your_table SELECT * FROM temp_table;
    DROP TABLE temp_table;
    

    See also av8n.com - How to Clone an SQL Record

    Benefits:

    • The SQL statements 2 mention only the fields that need to be changed during the cloning process. They do not know about – or care about – other fields. The other fields just go along for the ride, unchanged. This makes the SQL statements easier to write, easier to read, easier to maintain, and more extensible.
    • Only ordinary MySQL statements are used. No other tools or programming languages are required.
    • A fully-correct record is inserted in your_table in one atomic operation.
    0 讨论(0)
  • 2020-11-29 16:47

    For a quick, clean solution that doesn't require you to name columns, you can use a prepared statement as described here: https://stackoverflow.com/a/23964285/292677

    If you need a complex solution so you can do this often, you can use this procedure:

    DELIMITER $$
    
    CREATE PROCEDURE `duplicateRows`(_schemaName text, _tableName text, _whereClause text, _omitColumns text)
    SQL SECURITY INVOKER
    BEGIN
      SELECT IF(TRIM(_omitColumns) <> '', CONCAT('id', ',', TRIM(_omitColumns)), 'id') INTO @omitColumns;
    
      SELECT GROUP_CONCAT(COLUMN_NAME) FROM information_schema.columns 
      WHERE table_schema = _schemaName AND table_name = _tableName AND FIND_IN_SET(COLUMN_NAME,@omitColumns) = 0 ORDER BY ORDINAL_POSITION INTO @columns;
    
      SET @sql = CONCAT('INSERT INTO ', _tableName, '(', @columns, ')',
      'SELECT ', @columns, 
      ' FROM ', _schemaName, '.', _tableName, ' ',  _whereClause);
    
      PREPARE stmt1 FROM @sql;
      EXECUTE stmt1;
    END
    

    You can run it with:

    CALL duplicateRows('database', 'table', 'WHERE condition = optional', 'omit_columns_optional');
    

    Examples

    duplicateRows('acl', 'users', 'WHERE id = 200'); -- will duplicate the row for the user with id 200
    duplicateRows('acl', 'users', 'WHERE id = 200', 'created_ts'); -- same as above but will not copy the created_ts column value    
    duplicateRows('acl', 'users', 'WHERE id = 200', 'created_ts,updated_ts'); -- same as above but also omits the updated_ts column
    duplicateRows('acl', 'users'); -- will duplicate all records in the table
    

    DISCLAIMER: This solution is only for someone who will be repeatedly duplicating rows in many tables, often. It could be dangerous in the hands of a rogue user.

    0 讨论(0)
  • 2020-11-29 16:49

    I tend to use a variation of what mu is too short posted:

    INSERT INTO something_log
    SELECT NULL, s.*
    FROM something AS s
    WHERE s.id = 1;
    

    As long as the tables have identical fields (excepting the auto increment on the log table), then this works nicely.

    Since I use stored procedures whenever possible (to make life easier on other programmers who aren't too familiar with databases), this solves the problem of having to go back and update procedures every time you add a new field to a table.

    It also ensures that if you add new fields to a table they will start appearing in the log table immediately without having to update your database queries (unless of course you have some that set a field explicitly)

    Warning: You will want to make sure to add any new fields to both tables at the same time so that the field order stays the same... otherwise you will start getting odd bugs. If you are the only one that writes database interfaces AND you are very careful then this works nicely. Otherwise, stick to naming all of your fields.

    Note: On second thought, unless you are working on a solo project that you are sure won't have others working on it stick to listing all field names explicitly and update your log statements as your schema changes. This shortcut probably is not worth the long term headache it can cause... especially on a production system.

    0 讨论(0)
  • 2020-11-29 16:52

    You can also pass in '0' as the value for the column to auto-increment, the correct value will be used when the record is created. This is so much easier than temporary tables.

    Source: Copying rows in MySQL (see the second comment, by TRiG, to the first solution, by Lore)

    0 讨论(0)
提交回复
热议问题