MySQL 8.0+ 创建用户需分步:先CREATE USER指定主机和密码,再GRANT最小权限;禁用DELETE FROM mysql.user,须用DROP USER删除;验证权限应使用SHOW GRANTS而非查表。
MySQL 8.0+ 默认使用 mysql_native_password 插件,且不支持直接在 CREATE USER 中指定密码过期策略或账户锁定状态(这些需后续 ALTER USER 设置)。最安全、兼容性最好的建用户方式是分两步:先创建空密码/随机密码用户,再授权。
CREATE USER 'dev_user'@'192.168.1.%' IDENTIFIED BY 'StrongPass123!'; —— 指定 IP 段访问,避免用 '%' 开放所有主机(尤其生产)
显式指定认证插件:CREATE USER 'dev_user'@'localhost' IDENTIFIED WITH mysql_native_password BY 'Pass123';
GRANT 显式赋权,MySQL 8.0 不再允许 CREATE USER ... GRANT 合并在一条语句中SELECT, INSERT 而非 ALL PRIVILEGES,例如:GRANT SELECT, INSERT ON mydb.orders TO 'dev_user'@'192.168.1.%';
DROP USER 是唯一安全删除用户的命令,DELETE FROM mysql.user 直接删表记录会导致权限系统不一致、缓存未刷新、甚至 mysqld 重启失败。
DROP USER 'old_user'@'localhost'; —— 必须带 host,'old_user' 和 'old_user'@'localhost' 是两个不同账户REVOKE
--skip-grant-tables 启动修复,非常麻烦KILL 主动终止残留会话:KILL 12345;(ID 来自 SHOW PROCESSLIST;)别依赖 SELECT * FROM mysql.user 查看用户——字段含义复杂、加密密码不可读、host 匹配逻辑易误解。应使用系统视图和内置命令。
SELECT User, Host FROM mysql.user WHERE User = 'test_user';
SHOW GRANTS FOR 'dev_user'@'192.168.1.%'; —— 输出的是可重放的授权语句,最直观SHOW GRANTS;(不带 FOR),这对调试连接后权限异常特别有用Host 字段支持通配符(如 192.168.%),但匹配优先级按“最长前缀”规则,192.168.1.100 会优先匹配 192.168.1.% 而非 %
CREATE USER 'api_user'@'10.0.2.%' IDENTIFIED BY 'ApiToken$2025'; GRANT SELECT, UPDATE ON app_db.users TO 'api_user'@'10.0.2.%'; GRANT SELECT ON app_db.products TO 'api_user'@'10.0.2.%'; FLUSH PRIVILEGES;
最后一行 FLUSH PRIVILEGES 在绝大多数情况下不是必须的——只要用 CREATE USER/GRANT/DROP USER 等 DDL 命令操作,权限表已自动重载。仅当你手动修改了 mysql.user 表才需要它,而这种做法本身就不该发生。