背景:有时候需要同步数据库的表结构和部分数据,同步全表数据非常大,也不适合。还有一个种办法是使用数据库的dump命令执行备份,无法进入服务器?没有权限怎么办?
这里只要能访问服务器中的 information_schema数据库就能够进行导出。
1、查询 information_schema 中目标数据库的所有表名和存储引擎。
- SELECT table_name,engine
- FROM information_schema.tables
- WHERE table_schema="数据库名";
2、查询 information_schema 中目标数据库的字符集。
- SELECT CCSA.character_set_name,T.table_name
- FROM information_schema.`TABLES` T,
- information_schema.`COLLATION_CHARACTER_SET_APPLICABILITY` CCSA
- WHERE CCSA.collation_name = T.table_collation
- AND T.table_schema = '数据库名';
3、拿到目标数据库中的所有数据表名之后,查询表结构。
SELECT COLUMN_NAME 数据表的字段名
COLUMN_TYPE 字段的数据类型
COLUMN_COMMENT 字段的注释
COLUMN_DEFAULT 字段的默认值
EXTRA 字段的拓展信息,AUTO_INCREMENT
- SELECT COLUMN_NAME,COLUMN_TYPE,COLUMN_COMMENT,COLUMN_DEFAULT,EXTRA
- FROM information_schema.columns
- WHERE table_schema='数据库名'
- AND table_name = '表名'
4、拿到表结构字段信息拼接表结构语句。
- CREATE TABLE IF NOT EXISTS pur_accounting_log (
- id int(11) auto_increment PRIMARY KEY COMMENT 'ID',
- accounting_time datetime default '0000-00-00 00:00:00' COMMENT '核算时间',
- supplier_code varchar(20) default '' COMMENT '核算维度编码',
- purchase_name varchar(20) default '' COMMENT '核算维度主体',
- is_accounting tinyint(1) default 0 COMMENT '是否核算'
- ) ENGINE=InnoDB DEFAULT CHARSET=utf8;
5、查询表中的前 100 条数据,拼接 INSERT 语句。
- INSERT INTO pur_accounting_log (`id`,`accounting_time`,`supplier_code`,`purchase_name`,`is_accounting`)
- VALUES ('1','2023-09-25 00:00:00','A294494176','HK','0');
完成的代码:
- /**
- * PDO:
- * PHP 数据对象 (PDO :PHP Data Objects) 扩展为PHP访问数据库定义了一个轻量级的一致接口。
- * PDO 提供了一个数据访问抽象层,这意味着,不管使用哪种数据库,都可以用相同的函数(方法)来查询和获取数据。
- */
- $filePath = './down_database.sql';// SQL文件存储位置
-
- const DB_NAME = 'purchase';// 数据库名称
- const DB_HOST = '127.0.0.1';// 数据库IP
- const DB_USERNAME = 'root';// 用户名
- const DB_PASSWORD = '123456';// 密码
- const LIMIT = 5;
- const DSN = 'mysql:host='.DB_HOST.';dbname=' . DB_NAME;
-
- $options = [
- PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,// 返回索引数组格式
- ];
- $connectObj = new PDO(DSN, DB_USERNAME, DB_PASSWORD, $options);
-
- // 查询目标数据库中的所有表名称
- $query_tables = "SELECT engine,table_name
- FROM information_schema.tables
- WHERE table_schema='" . DB_NAME . "'";
- $tables_list = $connectObj->query($query_tables)->fetchAll();
-
- // 查询所有表的存储字符集
- $query_tables_charset = "SELECT CCSA.character_set_name,T.table_name
- FROM information_schema.`TABLES` T,
- information_schema.`COLLATION_CHARACTER_SET_APPLICABILITY` CCSA
- WHERE CCSA.collation_name = T.table_collation
- AND T.table_schema = '" . DB_NAME . "'";
-
- $tables_charset = $connectObj->query($query_tables_charset)->fetchAll();
- $tables_charset = array_column($tables_charset,'character_set_name','table_name');
-
- // 创建数据库并使用它
- $create_database = "CREATE DATABASE `" . DB_NAME . "`;\n\n";
- $use_database = "USE `" . DB_NAME . "`;\n\n";
-
- rewriteSqlToFile($filePath, $create_database);
- rewriteSqlToFile($filePath, $use_database);
-
- foreach ($tables_list as $value) {
- $table_name = $value['table_name'];
- $engine = $value['engine'];
- $charset = isset($tables_charset[$table_name]) ? $tables_charset[$table_name] : null;
-
- echo "开始导出表:" . DB_NAME . "." . $table_name;
- if (is_numeric($table_name)) {
- echo " ---> 失败:表明为数字无法导出\n";
- continue;
- }
-
- $query_table = "SELECT COLUMN_NAME,COLUMN_TYPE,COLUMN_COMMENT,COLUMN_DEFAULT,EXTRA
- FROM information_schema.columns
- WHERE table_schema='" . DB_NAME . "'
- AND table_name = '" . $table_name . "'";
- $tables_info = $connectObj->query($query_table)->fetchAll();
-
- // 拼接表结构
- $create_table_column = '';
- foreach ($tables_info as $column_value) {
- if (stripos($column_value['COLUMN_TYPE'], 'int') !== false) {
- $COLUMN_DEFAULT = intval($column_value['COLUMN_DEFAULT']);
- } else {
- $COLUMN_DEFAULT = "'" . strval($column_value['COLUMN_DEFAULT']) . "'";
- }
-
- $EXTRA = '';
- if ($column_value['EXTRA'] and $column_value['EXTRA'] == 'auto_increment') {
- $EXTRA = " " . $column_value['EXTRA'] . " PRIMARY KEY";// 自增主键
- } else {
- $EXTRA .= " default " . $COLUMN_DEFAULT;
- }
-
- $create_table_column .= "\t" . $column_value['COLUMN_NAME']
- . " " . $column_value['COLUMN_TYPE']
- . $EXTRA
- . " COMMENT '" . $column_value['COLUMN_COMMENT'] . "',\r\n";
- }
-
- $create_table = "CREATE TABLE IF NOT EXISTS " . $table_name . " ( \r\n" .
- trim($create_table_column, ",\r\n") . "\r\n"
- . ") ENGINE=" .$engine;
-
- if( $charset ) $create_table .= " DEFAULT CHARSET=" . $charset;
- $create_table .= ";\r\n";
-
- rewriteSqlToFile($filePath, $create_table);
-
-
- // 拼接数据集合
- $query_table = "SELECT *
- FROM " . DB_NAME . "." . $table_name . "
- WHERE 1=1
- LIMIT " . LIMIT;
- $tables_data_list = $connectObj->query($query_table)->fetchAll();
-
- if($tables_data_list){
- $insert_list = [];
- foreach ($tables_data_list as $item) {
- $row_insert_sql = "INSERT INTO {$table_name} (`" . implode("`,`", array_keys($item)) . "`) "
- . "VALUES ('" . implode("','", array_values($item)) . "');";
- $insert_list[] = $row_insert_sql;
- }
-
- rewriteSqlToFile($filePath, implode("\r\n", $insert_list));
-
- rewriteSqlToFile($filePath, "");
- rewriteSqlToFile($filePath, "");
- }
-
- echo " ---> 成功\n";
- }
-
- /**
- * 数据写入到SQL文件中
- * @param $filePath
- * @param $sql
- */
- function rewriteSqlToFile($filePath, $sql)
- {
- file_put_contents($filePath, $sql . PHP_EOL, FILE_APPEND);
- }
-
- echo "同步成功";
- exit;
导出过程:
导出SQL的部分示例: