如何将MySQL 5.7用户权限迁移至新服务器并备份相关库表?

2026-05-20 13:231阅读0评论SEO资讯
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计998个文字,预计阅读时间需要4分钟。

如何将MySQL 5.7用户权限迁移至新服务器并备份相关库表?

不能直接 连接

为什么 mysqldump mysql.user 表在 5.7 也不安全

很多人以为“只要版本一致,dump mysql.user 就能直接导入”,实际踩坑频繁。根本原因不是结构变,而是字段行为依赖运行时上下文:

  • password 字段在 5.7 是哈希值,但不同 MySQL 编译版本或 patch 级别对 old_passwords 模式处理不一致,导入后可能密码失效
  • plugin 字段若为 mysql_native_password,看似没问题,但若目标库启用了 skip_name_resolve=OFF(默认),而用户 host 是 'user'@'%.example.com',DNS 解析失败会导致权限不匹配
  • account_locked 字段在 5.7.6+ 才引入,旧版本目标库无此列,INSERT INTO mysql.user 会因字段数不匹配报错 ERROR 1136 (21S01): Column count doesn't match value count
  • max_questions/max_updates 等资源限制字段若为非零值,导入后会意外限流,且不易察觉

正确导出方式:只 dump 权限相关表 + 显式过滤

mysqldump 导出权限表是可行的,但必须严格限定表范围、跳过触发器/存储过程,并排除内置账号:

  • 执行命令:mysqldump -u root -p --skip-triggers --compact --no-create-info --where="User NOT IN ('mysql.session','mysql.sys','root')" mysql user db tables_priv columns_priv procs_priv proxies_priv > mysql_grants_57.sql
  • --no-create-info 防止生成 DROP TABLECREATE TABLE,避免覆盖目标库系统表结构
  • --where 过滤掉系统账号,防止误删或冲突;root 建议手工重建,不走自动导入
  • 导出后检查文件开头是否有 INSERT INTO `user` VALUES —— 若有,确认每行字段数与目标库 DESCRIBE mysql.user 输出一致(尤其注意 5.7.6+ 新增字段)

导入前必须做的三件事

导入不是把 SQL 扔进去就完事。5.7 的权限缓存机制要求显式刷新,且残留用户会导致权限叠加:

  • 先清理目标库同名用户:DROP USER 'app_user'@'192.168.%'; DROP USER 'app_user'@'localhost';(不能只删一个,@'host' 是独立账户)
  • 导入时加 SET FOREIGN_KEY_CHECKS = 0;SET UNIQUE_CHECKS = 0; 包裹,否则 mysql.procs_priv 等表可能因外键约束失败
  • 导入完成后**立刻执行**:FLUSH PRIVILEGES; —— 否则即使 mysql.user 表写入成功,内存权限缓存也不会更新,用户连得上但查不到库

验证权限是否真正生效

别只信 SHOW GRANTS 输出,它只反映语句层面;要验证实际行为是否一致:

  • 用目标用户连接:mysql -u app_user -p -h new-server-ip -e "SELECT USER(), CURRENT_USER();",确认返回的是 app_user@192.168.% 而非 app_user@localhost(后者说明 host 匹配失败)
  • 执行 SHOW GRANTS FOR 'app_user'@'192.168.%';,对比源库输出,注意 WITH GRANT OPTION 是否存在
  • 尝试执行一条被授权的操作,比如:CREATE TABLE test.t1 (id INT);,若报 ERROR 1142 (42000): CREATE command denied,说明权限未加载或 host 不匹配
  • SELECT User, Host, password FROM mysql.user WHERE User = 'app_user';,确认 password 字段非空且长度合理(5.7 哈希通常是 41 字符)

最易忽略的一点:导入脚本末尾没加 FLUSH PRIVILEGES;,或者用了普通账号导入导致部分 INSERT 被权限拒绝却没报错——结果是表数据进了,权限没生效,排查时绕一大圈。

标签:Mysql

本文共计998个文字,预计阅读时间需要4分钟。

如何将MySQL 5.7用户权限迁移至新服务器并备份相关库表?

不能直接 连接

为什么 mysqldump mysql.user 表在 5.7 也不安全

很多人以为“只要版本一致,dump mysql.user 就能直接导入”,实际踩坑频繁。根本原因不是结构变,而是字段行为依赖运行时上下文:

  • password 字段在 5.7 是哈希值,但不同 MySQL 编译版本或 patch 级别对 old_passwords 模式处理不一致,导入后可能密码失效
  • plugin 字段若为 mysql_native_password,看似没问题,但若目标库启用了 skip_name_resolve=OFF(默认),而用户 host 是 'user'@'%.example.com',DNS 解析失败会导致权限不匹配
  • account_locked 字段在 5.7.6+ 才引入,旧版本目标库无此列,INSERT INTO mysql.user 会因字段数不匹配报错 ERROR 1136 (21S01): Column count doesn't match value count
  • max_questions/max_updates 等资源限制字段若为非零值,导入后会意外限流,且不易察觉

正确导出方式:只 dump 权限相关表 + 显式过滤

mysqldump 导出权限表是可行的,但必须严格限定表范围、跳过触发器/存储过程,并排除内置账号:

  • 执行命令:mysqldump -u root -p --skip-triggers --compact --no-create-info --where="User NOT IN ('mysql.session','mysql.sys','root')" mysql user db tables_priv columns_priv procs_priv proxies_priv > mysql_grants_57.sql
  • --no-create-info 防止生成 DROP TABLECREATE TABLE,避免覆盖目标库系统表结构
  • --where 过滤掉系统账号,防止误删或冲突;root 建议手工重建,不走自动导入
  • 导出后检查文件开头是否有 INSERT INTO `user` VALUES —— 若有,确认每行字段数与目标库 DESCRIBE mysql.user 输出一致(尤其注意 5.7.6+ 新增字段)

导入前必须做的三件事

导入不是把 SQL 扔进去就完事。5.7 的权限缓存机制要求显式刷新,且残留用户会导致权限叠加:

  • 先清理目标库同名用户:DROP USER 'app_user'@'192.168.%'; DROP USER 'app_user'@'localhost';(不能只删一个,@'host' 是独立账户)
  • 导入时加 SET FOREIGN_KEY_CHECKS = 0;SET UNIQUE_CHECKS = 0; 包裹,否则 mysql.procs_priv 等表可能因外键约束失败
  • 导入完成后**立刻执行**:FLUSH PRIVILEGES; —— 否则即使 mysql.user 表写入成功,内存权限缓存也不会更新,用户连得上但查不到库

验证权限是否真正生效

别只信 SHOW GRANTS 输出,它只反映语句层面;要验证实际行为是否一致:

  • 用目标用户连接:mysql -u app_user -p -h new-server-ip -e "SELECT USER(), CURRENT_USER();",确认返回的是 app_user@192.168.% 而非 app_user@localhost(后者说明 host 匹配失败)
  • 执行 SHOW GRANTS FOR 'app_user'@'192.168.%';,对比源库输出,注意 WITH GRANT OPTION 是否存在
  • 尝试执行一条被授权的操作,比如:CREATE TABLE test.t1 (id INT);,若报 ERROR 1142 (42000): CREATE command denied,说明权限未加载或 host 不匹配
  • SELECT User, Host, password FROM mysql.user WHERE User = 'app_user';,确认 password 字段非空且长度合理(5.7 哈希通常是 41 字符)

最易忽略的一点:导入脚本末尾没加 FLUSH PRIVILEGES;,或者用了普通账号导入导致部分 INSERT 被权限拒绝却没报错——结果是表数据进了,权限没生效,排查时绕一大圈。

标签:Mysql