1. Backup and restoration policies

There are some factors to consider when performing backup or restore operations:

  • Determine whether the storage engine of the table to be backed up is transactional or non-transactional, and the two different storage engine backups handle data consistency differently.

  • Determine whether to use full backup or incremental backup. The advantage of full backup is that it keeps the latest backup and takes less time to restore. The disadvantage is that if there is a large amount of data, it will take a lot of time and cause a long time of pressure on the system. Incremental backup On the contrary, only incremental logs of each day are backed up, which takes less time and exerts less load. The disadvantage is that the recovery requires full backup and all the logs before the fault backup, which takes longer.

  • You can consider using the method of replication to do remote backup, but it can not replace backup, it can not do anything against the misoperation of the database.

  • Backup must be performed periodically, and the backup period must fully consider the recovery time supported by the system. Backup should be performed when system load is low

  • Make sure MySQL has log-bin enabled. With binlog, MySQL can do full restore when necessary, either point-in-time or location-based.

  • Do frequent backup and restore tests to ensure that backups are valid and recoverable.

Logical backup and restore

In MySQL, the biggest advantage of logical backup is that it can be backed up in the same way for all storage engines. Physical backup is different. Different storage engines have different backup methods. Therefore, logical backup is easier for databases mixed with different storage engines.

1. Backup

Logical backup in MySQL is to back up the data in the database into a text file, which can be viewed and edited. In MySQL, you can use mysqldump to perform a logical backup:

Shell > mysqldump [options] db_name [tables] // Back up one or more specified databases shell> mysqldump [options] --database  DB1 [DB2,DB3...] Shell > mysqldump [options] --all-databaseCopy the code

If no tables in the database are specified, all tables in all databases are exported by default.

Example:
1. Back up all databases:
shell>mysqldump -uroot -p --all-database > all.sqlCopy the code
2. Back up database test
shell>mysqldump -uroot -p test > test.sqlCopy the code
3. Back up emP of database test
shell> mysqldump -uroot -p test emp > emp.sqlCopy the code
4. Back up emP and DEPT in test
shell> mysqldump -uroot -p test emp dept > emp_dept.sql Copy the code
5. Back up all tables under test as comma-separated text to/TMP:
shell> mysqlddump -uroot -p -T /tmp test emp --fields-terminated-by ','
shell> more emp.txt  

1,z1
2,z2
3,z3
4,z4Copy the code

Note: In order to ensure the consistency of data backup, myISAM storage engine needs to add the -L parameter during the backup, which means that all tables are locked. During the backup, all tables can only be read and data cannot be updated. However, there is a better option for the transaction storage engine — single-Transaction. This option allows innoDB storage engine to take a snapshot to ensure consistency of the backup data.

2. Full recovery

Mysqldump is also very simple to restore, take backup as input:

mysql -uroot -p db_name < backfileCopy the code

Note that the data after the backup is restored is not complete, and the logs executed after the backup need to be redone:

mysqlbinlog binlog-file | mysql -uroot -pCopy the code
Mysqldump backup and restore example
1. Backup database at 2:00 am:
root@bogon:/usr/local/mysql/bin$ ./mysqldump -uroot -p -l -F t2 > t2.dmp
Enter password: Copy the code

Where -l indicates that all tables are read locked, and -f indicates that a new log file is generated. In this case, emP table data in T2 is as follows:

Reset master deletes all binlogs for testing purposes. MySQL [(none)]> reset master; Query OK, 0 rows affected (0.00 SEC) # select * from test; + -- -- -- -- -- - + -- -- -- -- -- -- + | | id name | + -- -- -- -- -- - + -- -- -- -- -- -- + | 1 | a | | 2 | b | + -- -- -- -- -- - + -- -- -- -- -- - + 2 rows in the set (0.00 SEC)Copy the code
2. After the backup, insert new data:
Mysql [T2]> insert into test values (3,'c'); mysql [t2]> insert into test values (3,'c'); Query OK, 1 row affected (0.00 SEC) MySQL [T2]> insert into test values (4,'d'); Query OK, 1 row affected (0.00 SEC) MySQL [T2]> insert into test values (4,'d'); Query OK, 1 row affected (0.00 SEC)Copy the code
3. The database suddenly fails (in fact, it is a friend who has nothing to do with deleting the database for practice), and the data cannot be accessed. Need to restore backup:

Delete library run:

Sql_log_bin is set to off for the current session. Mysql [T2]> set sql_log_bin = 0; mysql [t2]> sql_log_bin = 0; Query OK, 0 rows affected (0.00 SEC) MySQL [T2]> show variables like "%sql_log_bin%"; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | sql_log_bin | OFF | + -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- + 1 row in the set (0.00 SEC) MySQL (t2) > drop database t2; Query OK, 1 row affected (0.01sec) MySQL [(none)]> Flush logs; Query OK, 0 rows affected (0.22 SEC) MySQL [T2]> drop database t2; Query OK, 3 rows affected (0.23 SEC) MySQL [(none)]> exit; ByeCopy the code

Data recovery:

root@bogon:/usr/local/mysql/bin# ./mysql -e "create database t2" root@bogon:/usr/local/mysql/bin# ./mysql t2 < t2.dmp ******************************************************************* MySQL [t2]> select * from test; + -- -- -- -- -- - + -- -- -- -- -- -- + | | id name | + -- -- -- -- -- - + -- -- -- -- -- -- + | 1 | a | | 2 | b | + -- -- -- -- -- - + -- -- -- -- -- - + 2 rows in the set (0.00 SEC)Copy the code
4. Use mysqlbinlog to restore the binglog since mysqldump backup

Based on the previous operations, operations between the backup point in time and the database deletion point in time are recorded in the mysql-bin.000002 file

root@bogon:/usr/local/mysql/bin# ./mysqlbinlog --no-defaults /data/mysql/mysql-bin.000002 | ./mysql t2

*******************************************************
MySQL [t2]> select * from test;
+------+------+
| id   | name |
+------+------+
|    1 | a    |
|    2 | b    |
|    3 | c    |
|    4 | d    |
+------+------+
4 rows in set (0.00 sec)
Copy the code

The data is successfully recovered.

3. Point-in-time recovery

Due to misoperation, such as deleting a table by mistake, it is useless to use full recovery at this time, because there are still misoperation statements in the log. What we need is to restore to the state before the misoperation, and then skip the misoperation statement, and then restore the following statements to complete the recovery. This type of recovery is called incomplete recovery. In MySQL, incomplete recovery is divided into point-in-time recovery and location-based recovery. Procedure For point-in-time recovery:

(1) If a misoperation occurred at 10 am, you can use the following statement to restore the data before the fault with backup and binlog:
shell>mysqlbinlog --stop-date="2017-09-30 9:59:59" /data/mysql/mysql-bin.123456 | mysql -uroot -ppasswordCopy the code
(2) Skip the time point when the fault occurs and continue to execute the following binlog to complete the recovery.
shell>mysqlbinlog --start-date="2017-09-30 10:01:00" /data/mysql/mysql-bin.123456 | mysql -uroot -ppasswordCopy the code

4. Location-based recovery

This is similar to point-in-time recovery, but more precise, because many SQL statements may be executed at the same time. The restoration procedure is as follows:

(1) Execute the command under shell:
shell>mysqlbinlog --start-date="2017-09-30 9:59:59" --stop-date="2017-09-30 10:01:00" /data/mysql/mysql-bin.123456 > /tmp/mysql_restore.sqlCopy the code

This command will create a small text file in the/TMP directory and edit the file to know the location numbers before and after the error statement, for example, 368312 and 368315.

(2) After restoring the previous backup file, enter the following content from the command line:
shell>mysqlbinlog --stop-position="368312" /data/mysql/mysql-bin.123456 | mysql -uroot -ppassword  
shell>mysqlbinlog --start-position="368315" /data/mysql/mysql-bin.123456 | mysql -uroot -ppassword Copy the code

The first line above restores all transactions up to the stop position. The next line restores all transactions from the given starting position until the end of the binary log. Because the output of mysqlbinlog includes the set TIMESTAMP statement before each SQL statement is recorded, the recovered data and associated mysql logs will reflect the original time when the transaction was executed.

Physical backup and restoration

Physical backup can be divided into cold backup and hot backup. Compared with logical backup, physical backup has the biggest advantage of faster backup and recovery because physical backup is based on file CP.

1. Cold backup

Cold backup is actually a way to stop the database service, CP data files. (This method is rarely considered)

2. Hot backup

In MySQL, the hot backup method is different for different storage engines.

(1) MyISAM storage engine

Myisam storage engine has many methods of hot backup, essentially is to be backed up table read lock, and then cp data files to the backup directory. The following two methods are commonly used:

  • Use the mysqlHotCopy tool

Shell > mysqlhotCopy db_name [/path/to/new_directory]Copy the code
  • Manually lock the table copy

Mysql >flush tables for read; Cp data files to the backup directory.Copy the code
(2) InnoDB storage engine

Use third-party tools ibBackup, Xtrabackup and Innobacupex

4. Table import and export

In the daily maintenance of database, the import and export of tables is a very frequent operation.

1. Export

In some cases, it is often necessary to export the data in a table as plain data text separated by symbols, rather than SQL statements, for specific purposes:

  • Used as Excel display;

  • To save backup space;

  • For fast data loading, load Data is 20 times faster than normal SQL loading.

Use the select… into outfile … Command to export data. The syntax is as follows:
mysql> select * from tablename into outfile 'target_file' [option];Copy the code

The option parameter can be the following:

Fields terminated by 'string' // fields terminated by 'string' // default is tabbed '\t' fields [optionally] enclosed by 'char' // Optionally optionally only on char, vARCHar, text and other character fields '\' lines starting by 'string' // Fields escaped by 'char' // transfer character by default '\' lines' // The default '' lines terminated by 'string' // is '\n' # char which means the symbol can only be a single character and string which means it can be a string.Copy the code

For example, export the data in the test table as data text, where the field delimiter is “, “, the field reference is “”, and the record end character is carriage return:

MySQL [t2]> select * from test into outfile '/data/mysql/outfile.txt' fields terminated by "," enclosed by '"'; Query OK, 4 rows affected (0.02sec)Copy the code
zj@bogon:/data/mysql$ more outfile.txt 
"1","a","helloworld"
"2","b","helloworld"
"3","c","helloworld"
"4","d","helloworld"Copy the code

If the first column is numeric, do not want to enclose the column with quotation marks

MySQL [t2]> select * from test into outfile '/data/mysql/outfile2.txt' fields terminated by "," optionally enclosed by '"'; Query OK, 4 rows affected (0.03 SEC) zj@bogon:/data/mysql$more outfile2. TXT 1,"a"," helloWorld "2,"b"," helloWorld" 3,"c","helloworld" 4,"d","helloworld"Copy the code

To test the escape characters, the characters to be escaped in MySQL exported data mainly include the following three types:

  • Escape the character itself

  • Field separator

  • Record separator

MySQL [t2]> update test set content = '\\"##! aa' where id=1; Query OK, 1 row affected (0.05sec) Rows matched: 1 Changed: 1 Warnings: 0 MySQL [t2]> select * from test into outfile '/data/mysql/outfile3.txt' fields terminated by "," optionally enclosed by '"'; Query OK, 4 rows affected (0.03 SEC) * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * what zj had @ bogon: / data/mysql $more outfile3. TXT 1,"a","\\\"##! aa" 2,"b","helloworld" 3,"c","helloworld" 4,"d","helloworld"Copy the code
  • When the exported command contains a field reference character, the characters in the data containing the escape character and the field reference character need to be escaped.

  • When the export command does not contain field references, the characters in the data that contain the escape character itself and the field separator need to be escaped.

Note: select… into outfile … If the generated output file has a file with the same name in the destination directory, the file will not be created successfully and the source file will not be overwritten automatically.

The syntax for exporting data as text with mysqldump is as follows:

mysqldump -u username -T target_dir dbname tablename [option]

The option parameter can be the following:

  • — field-terminated -by=name (field separator);

  • –fields-enclosed-by=name (field reference);

  • — fields-infile enclosed-by=name (optionally only for char, varchar, test, etc.);

  • –fields-escaped-by=name (escaped character);

  • — lines-by =name (terminated);

Example:

root@bogon:/usr/local/mysql/bin# ./mysqldump -uroot -p -T /data/mysql/dump t2 test --fields-terminated-by ',' --fields-optionally-enclosed-by '"' **************** test.txt ********************** zj@bogon:/data/mysql/dump$ more test.txt 1,"a","\\\"##! aa" 2,"b","helloworld" 3,"c","helloworld" 4,"d","helloworld" ***************** test.sql ********************* zj@bogon:/data/mysql/dump$more test. SQL -- mysql dump 10.13 Distrib 5.7.18, for Linux (x86_64) -- -- Host: Localhost Database: t2 -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- Server version 5.7.18 - log / *! 40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */; / *! 40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */; / *! 40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */; / *! 40101 SET NAMES utf8mb4 */; / *! 40103 SET @OLD_TIME_ZONE=@@TIME_ZONE */; / *! 40103 SET TIME_ZONE='+00:00' */; / *! 40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='' */; / *! 40111 SET @OLD_SQL_NOTES=@@SQL_NOTES, SQL_NOTES=0 */; -- -- Table structure for table `test` -- DROP TABLE IF EXISTS `test`; / *! 40101 SET @saved_cs_client = @@character_set_client */; / *! 40101 SET character_set_client = utf8 */; CREATE TABLE `test` ( `id` int(11) DEFAULT NULL, `name` varchar(10) DEFAULT NULL, `content` varchar(100) DEFAULT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4; / *! 40101 SET character_set_client = @saved_cs_client */; / *! 40103 SET TIME_ZONE=@OLD_TIME_ZONE */; / *! 40101 SET SQL_MODE=@OLD_SQL_MODE */; / *! 40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */; / *! 40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */; / *! 40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */; / *! 40111 SET SQL_NOTES=@OLD_SQL_NOTES */; -- Dump completed on 2017-09-25 11:14:06Copy the code

Mysql > select * from mysqldump; mysqldump; into outfile … The options and syntax are very similar. Mysqldump actually calls the interface provided by the latter and adds some new functionality to it.

2. Import (import with select… Into outfile or mysqldump to export plain data text)

Similar to exporting, there are two different import methods: Load data infile… Like mysqlimport, they are essentially the same, except that one is executed inside MySQL and the other outside MySQL.

Use load data infile… Command. The syntax is as follows
mysql> load data [local]infile 'filename' into table tablename [option]  
Copy the code

Option can be the following:

  • Fields terminated by ‘string’ (field separator, defaults to TAB character ‘t’);

  • Fields [optionally] enclosed by ‘char’ (optionally) fields’ varchar text ‘ No references are used by default);

  • Fields escaped by ‘char’ (default ”)

  • Lines starting by ‘string’

  • Lines terminated by ‘string’ (terminated by ‘n’ by default)

  • Ignore number lines (ignore the first few lines in the input file)

  • (col_name_or_user_var,…) Load data in the order and number of fields listed.

  • set col_name = expr,… Convert the column to a certain value before loading it.

Fields, lines and before select… into outfile… The following example loads data from file ‘test.txt’ into table test:

MySQL [T2]> truncate table test; Query OK, MySQL [T2]> load data infile '/data/ MySQL /outfile.txt' into table test fields terminated by ',' enclosed by '"'; 2. Skipped over: 2. Skipped over: 2. Skipped over: 2. Skipped Skipped over: 2. +------+------+------------+ | id | name | content | +------+------+------------+ | 1 | a | helloworld | | 2 | b | The helloworld | | 3 | | c helloworld | | | | 4 d helloworld | + -- -- -- -- -- - + -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- - + 4 rows in the set (0.00 SEC)Copy the code

If you do not want to load the first two lines of the file, do the following:

MySQL [t2]> truncate table test; Query OK, MySQL [T2]> load data infile '/data/ MySQL /outfile.txt' into table test fields terminated by ',' enclosed by '"' ignore 2 lines; 2. Skipped: 0. Skipped: 0. 2. Skipped: 0. +------+------+------------+ | id | name | content | +------+------+------------+ | 3 | c | helloworld | | 4 | d | The helloworld | + -- -- -- -- -- - + -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- -- + 2 rows in the set (0.02 SEC)Copy the code

Implemented using mysqldump

Grammar:

shell> mysqlimport -uroot -p [--local] dbname order_tab.txt [option]  Copy the code

The option parameter can be the following:

  • –fields-terminated-by=name (field separator)

  • –fields-enclosed-by=name (field reference)

  • — fields-infile enclosed-by=name (optionally used in char, varchar, text, etc.)

  • –fields-escaped-by=name

  • — lines-by =name

  • –ignore-lines=number (ignore first few lines)

Note: If the import and export are cross-platform (Windows and Linux), be careful to set the parameter line-terminated-by, On Windows it is set to line-terminated-by=’rn’ and on Linux it is set to line-terminated-by=’n’.