不该。json字段会导致丢失索引、增加解析开销、sql查询脆弱;应结构化建表,用config_key作唯一索引、config_value按需存字符串,加env/is_active/version等字段支持多环境、热更新与分层治理。

配置表该不该用 JSON 字段存?
不该。MySQL 5.7+ 虽支持 JSON 类型,但配置项通常需要被 WHERE 条件过滤、被应用层高频读取、被运维人工核查——用 JSON 会丢失索引能力、增加解析开销、让 SQL 查询变得脆弱。
正确做法是结构化建表:
-
config_key作为主键或唯一索引(如'site_name'、'email_enabled') -
config_value用VARCHAR(512)或TEXT,按需转类型(布尔存 'true'/'false',数字存字符串再 cast) - 加
updated_at和is_active字段,方便灰度和回滚
多环境配置怎么避免手动改库?
靠数据库字段隔离,别靠库名或实例隔离。同一张 config 表里加 env 字段(值为 'dev' / 'staging' / 'prod'),应用启动时通过启动参数或环境变量传入当前 env,查询时带上 WHERE env = ? AND is_active = 1。
这样做的好处:
- 部署脚本无需区分环境执行不同 SQL
- 配置变更可走统一审核流程(比如只允许 prod 的
is_active由 DBA 手动开启) - 开发本地连测试库也能读到对应
dev配置,不污染 prod 数据
配置变更如何不重启服务生效?
MySQL 本身不推通知,得靠应用层轮询或监听。推荐轻量方案:在 config 表加 version 自增字段,每次更新配置就 UPDATE ... SET version = version + 1;应用缓存配置时记下当时读到的 version,每隔 30 秒查一次 MAX(version),发现变大就全量重载。
注意点:
- 别用
SELECT * FROM config全表扫——加WHERE env = ? AND is_active = 1条件,且(env, is_active, version)建联合索引 - 避免多个实例同时 reload 导致 DB 瞬时压力,可用随机 jitter(比如 30±5 秒)错开检查时间
- 如果配置极少变动,也可以接受 5 分钟缓存 + 重启生效,没必要为低频场景加复杂监听
通用项目结构里配置表放哪一层?
放在「数据访问层」,但接口要上提到「业务配置服务」。意思是:DAO 层只提供 getConfigByKey(env, key)、batchUpdateConfigs(env, configs) 这类原子操作;所有校验(比如 'timeout_ms' 必须是正整数)、默认兜底(key 不存在时返回预设值)、变更审计(谁在什么时候改了什么)都封装在上层 service 里。
这样分层后:
- Web 控制台、CLI 工具、定时任务都能复用同一套配置读写逻辑
- DBA 直接查表不会看到业务规则污染(比如不会看到 “status=1 表示启用” 这种隐含语义)
- 后续迁移到 Redis 或配置中心时,只需重写 DAO 实现,上层完全不动
真正难的不是建表,是定义清楚哪些算“配置”——比如用户等级规则、价格策略这类带业务逻辑的,不该塞进通用配置表,该进独立的业务规则表或 DSL 引擎。










