巧用*_his表记录操作历史
文章转载自「开发者圆桌」一个关于开发者入门、进阶、踩坑的微信公众号
许多OLTP应用的开发者都知道,一些重要的操作要记录操作历史,把操作前的数据备份到历史表,然后再执行相应的修改操作。这样可以获取某个时点的操作日志和操作前的记录值。
要记录操作历史有两个层面,一个是应用层即通过应用程序逻辑实现历史记录的保存,一个是数据库层即数据库归档、审计等DBA管理的功能。这里仅讨论前者。
比如有一个用户表t_user包含id,name,sex,age字段,因为t_user表的修改非常重要,要记录操作历史,一般的做法是建立和t_user结构相同的一个历史表命名为t_user_his,它包含id,name,sex,age字段,同时添加对应的his_id,oper_time,oper_remark等记录操作说明的字段。
OK,表设计完毕了,我们需要在修改时记录用户之前的信息,一般的做法是先来个insert插入历史表,然后再去修改原表数据,把这两个操作放到同一个事务中去处理,释义代码如下:
setAutoCommit(false);
insert into t_user_his(id,name,sex,age,his_id,oper_time,oper_remark) select id,name,sex,age,'历史表id',timestamp,'修改姓名' from t_user where id=18970;
update t_user set name='xxx' where id=18970;
commit();
部署上线一切运行正常,领导非常满意,突然有一天需要在t_user中添加一个phone字段来记录用户的电话号码,软件产品唯一不变的就是下面这个词
没办法只能修改了,需要按照下面几个步骤修改:
首先,修改t_user表添加一个phone字段。
其次,修改t_user_his表添加一个phone字段。
第三,修改涉及到的记录操作历史的SQL,如果有多处也需要一并修改:
setAutoCommit(false);
insert into t_user_his(id,name,sex,age,phone,his_id,oper_time,oper_remark) select id,name,sex,age,phone,'历史表id',timestamp,'修改姓名' from t_user where id=18970;
update t_user set name='xxx' where id=18970;
commit();
最后,编译程序,测试,提交,部署上线,一个流程下来可能要好多天。
问题来了,只能这样处理吗?如何规避字段修改带来的重复修改呢?有没有更好的解决方案呢?答案是肯定的,只需要通过一个简单的修改就可以做到。
我们可以这样来设计t_user_his表的字段,把历史操作相关字段前置,后续字段保持与t_user完全一致,例如t_user_his字段可以这样安排:his_id,oper_time,oper_remark,id,name,sex,age。
应用程序修改如下:
setAutoCommit(false);
insert into t_user_his select ,'历史表id',timestamp,'修改姓名' ,a.* from t_user a where id=18970;
update t_user set name='xxx' where id=18970;
commit();
这样的话,要添加一个新的phone字段,该如何操作呢?
首先,修改t_user表添加一个phone字段。
其次,修改t_user_his表添加一个phone字段与t_user字段顺序一致。
好了,两步就可以实现字段的自由添加、修改或者删除,不需要修改任何应用程序代码,也不再需要复杂的部署、测试、上线流程,仅在数据库中完成操作就可以了。
上述方法,修改一个字段就要记录整条数据到历史表,会占用较多的空间,要么就把字段写死。具体如何选择还是要分析系统的业务要求,不可盲目照搬。
巧用*_his表记录操作历史的更多相关文章
- MySQL数据库(3)_MySQL数据库表记录操作语句
附: MYSQL5.7版本sql_mode=only_full_group_by问题 .查询当前sql_mode: select @@sql_mode .查询出来的值为: set @@sql_mode ...
- Django中ORM简介与单表数据操作
一. ORM简介 概念:.ORM框架是用于实现面向对象编程语言种不同类型系统的数据之间的转换 构建模型的步骤:重点 (1).配置目标数据库信息,在seting.py中设置数据库信息 DATABASE ...
- Django之mysql表单操作
在Django之ORM模型中总结过django下mysql表的创建操作,接下来总结mysql表记录操作,包括表记录的增.删.改.查. 1. 添加表记录 class UserInfo(models.Mo ...
- Mycat读写分离、主从切换、分库分表的操作记录
系统开发中,数据库是非常重要的一个点.除了程序的本身的优化,如:SQL语句优化.代码优化,数据库的处理本身优化也是非常重要的.主从.热备.分表分库等都是系统发展迟早会遇到的技术问题问题.Mycat是一 ...
- excel vba 实现跨表单(sheet) 搜索 - 显示搜索行记录搜索历史
前两天,一个朋友问我,有没有办法在excel里实现一个表单里是原始数据,在另一个表单里显示搜索到的行,搜索关键词可用~分隔开,并把搜索历史记录下来? 我想了想,用vba实现肯定可以啊,但是我又在想,有 ...
- oracle中查找某用户执行某张表的操作操作记录
转载:http://www.cnblogs.com/nizuimeiabc1/p/9441937.html 1,首先查找表的操作记录 select * from v$sqlarea a where a ...
- {django模型层(二)多表操作}一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询、分组查询、F查询和Q查询
Django基础五之django模型层(二)多表操作 本节目录 一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询.分组查询.F查询和Q查询 六 xxx 七 ...
- ORM多表操作之创建关联表及添加表记录
创建关联表 关于表关系的几个结论 (1)一旦确立表关系是一对多:建立一对多关系----在多对应的表中创建关联字段. (2)一旦确立表关系是多对多:建立多对多关系----创建第三张关系表----id和两 ...
- MySQL(单表的表记录的操作)
一.表记录的增删改查 1.增加表记录 <1>插入一条记录: insert [into] tab_name (field1,filed2,.......) values (value1,va ...
随机推荐
- 在Delphi中隐藏程序进程
在开发某些软件的时候,为了保护程序自身,就需要用到隐藏程序进程.以下通过实例来讲解隐藏程序进程的方法: 1.创建一个新的项目 Project1 选择File,New Application.在表单Fo ...
- 添加redo日志组和添加日志组多元化
查看redo日志组的状态和日志的位置. SQL> 没有被使用,所以切几次日志,组合4已生效. SQL> select * from v$log; GROUP# THREAD# SEQ ...
- win7配置自己的IIS服务器亲自做的图文很详细
跟人网站爱好初学者必看的win7系统配置自己的IIS,可以在你自己的电脑上配置网站服务器发不到网上,下面就跟着我的步骤一起做吧100%成功. 步骤/方法 点击开始-------控制面板这个就是 ...
- (二)Hololens Unity 开发之 语音识别
学习源于官方文档 Voice input in Unity 笔记一部分是直接翻译官方文档,部分各人理解不一致的和一些比较浅显的保留英文原文 (二)Hololens Unity 开发之 语音识别 Hol ...
- RabbitMQ小白菜学习之在window下的安装配置
RabbitMQ安装 首先需要下载RabbitMQ的平台环境Erlang OTP平台和RabbitMQ Server(windows版): OTP 19.1 Windows 64-bit Binary ...
- opp(Object Oriented Programming)
嗯,昨天忙了一天没来及发,过年啊,打扫啊,什么搽窗户啊,拖地啊,整理柜子啊,什么乱七八糟的都有,就是一个字,忙. 好了,废话也不多说,把自己学到的放上来吧.嗯,说什么好呢,就说原型链啊 原型对象 每个 ...
- 没有苹果电脑打包iOS平台的 Ionic 2程序——《Ionic 2 实例开发》更新内容
没有苹果电脑打包iOS平台的 Ionic 2程序--<Ionic 2 实例开发>更新内容春节刚过,祝各位新的一年里万事如意,一帆风顺.<Ionic 2 实例开发>在这段时间里更 ...
- 【鸡年大吉】,不知道写点啥,放个demo(小球碰撞)吧,有兴趣的看看
最初的想法是仿写win7的泡泡屏保效果,但是对于小球的斜碰问题一直没搞明白(如果你会这个,欢迎留言或者做个demo),所以只是简单处理了碰撞后的速度,有时候会看起来很搞笑~~~funny guy 话不 ...
- SQLite:自学笔记(1)——快速入门
SQLite的安装和入门 了解 啥是SQLite? SQLite是一种轻巧迷你的关系型数据库管理系统.它的特点如下: 不需要一个单独的服务器进程或操作的系统(无服务器的). SQLite 不需要配置, ...
- 列存储段消除(ColumnStore Segment Elimination)
列存储索引是好的!对于数据仓库和报表工作量,它们是真正的性能加速器.与聚集列存储结合,你会在常规行存储索引(聚集索引,非聚集索引)上获得巨大的压缩好处.而且创建聚集列存储索引非常简单: CREATE ...