【Paddy】如何将物理表分割成动态数据表与静态数据表
前言
一般来说,物理表的增、删、改、查都受到数据量的制约,进而影响了性能。
很多情况下,你所负责的业务关键表中,每日变动的数据库与不变动的数据量比较,相差非常大。
这里我们将变动的数据称为动态数据,不变动的数据称为静态数据。
举个例子,1张1000W的表,每日动态数据只有1W条,999W条的数据都为静态。往往select或者重复改变的数据都在动态数据中。比如订单表。
所以,如果将动态数据库从表中剥离出来,分割两张表,一张动态数据表,一张静态数据表,从数据量的角度来看,性能是不是就会自然提高了?
下图为一张现实表和一张理想化的结构 (蓝色为静态数据:1000W、黄色为动态数据:100W):

希望动态数据和静态数据分割开

想法再好,也比不上现实的残酷。很多朋友可能都想到这种架构,但实现起来问题非常多:
1. 如何将动态数据自动从静态数据中剥离?如何维护?
2. 程序代码改动量不可预计。尤其核心表的使用遍布各地,分割成两张表后,所有代码基本都要重构。
3. 表分区是否可以达到同样目的,NO~因为表分区无法100%区分动态与静态,而且分区列的使用决定了效率。
下面开始,进入本次主题,如何在SQL Server中达到动态数据、静态数据分割;两表数据如何自动维护;程序统一访问逻辑表名,零维护成本。
正文
实现方法:利用视图 + 视图触发器
没错,没有了~就是靠这两货,就解决了,下面我们来看看逻辑图(画的不好,见谅)。

最后,让我们来看看具体事例
1. 创建demo数据库,和动态表(TB_A_NEW)、历史表(TB_A_OLD)
CREATE DATABASE demo
GO
ALTER DATABASE demo ADD FILEGROUP HekatonFG CONTAINS MEMORY_OPTIMIZED_DATA;
GO
ALTER DATABASE demo ADD FILE( NAME = 'Memory', FILENAME ='C:\Memory.ndf') TO FILEGROUP [HekatonFG];
GO
CREATE TABLE TB_A_OLD
(
id INT IDENTITY(1,1) PRIMARY KEY,
NAME NVARCHAR(36)
GO
CREATE TABLE TB_A_NEW
(
id INT IDENTITY(1, 1) PRIMARY KEY,
NAME NVARCHAR(36)
)
GO
INSERT INTO TB_A_OLD
SELECT Newid()
GO 10
这个时候的两张表的数据如下:

然后,我们为了保证动态表的新增数据的自增值是静态表的当前标识值+1,这样两表记录合并才能衔接。
DECLARE @IDENT BIGINT
SELECT @IDENT = Ident_current('TB_A_OLD') + 1
DBCC checkident('TB_A_NEW', reseed, @IDENT)
创建视图(View_TB_A)
CREATE VIEW [dbo].[View_TB_A]
AS
SELECT *
FROM TB_A_NEW
UNION ALL
SELECT old.*
FROM TB_A_OLD old
LEFT JOIN TB_A_NEW new
ON old.id = new.id
WHERE new.NAME IS NULL GO
创建视图触发器(Insert、Update)
USE demo GO CREATE TRIGGER [dbo].[TR_Insert]
ON [dbo].[View_TB_A]
INSTEAD OF INSERT
AS
BEGIN
SET NOCOUNT ON INSERT INTO TB_A_NEW
(NAME)
SELECT NAME
FROM inserted
END GO USE demo GO CREATE TRIGGER [dbo].[TR_Update]
ON [dbo].[View_TB_A]
INSTEAD OF UPDATE
AS
BEGIN
SET NOCOUNT ON --1. 判断update中,包含的记录是否都在new表中
DECLARE @count BIGINT SELECT @count = Count(1)
FROM deleted UPDATE TB_A_NEW
SET NAME = b.NAME
FROM TB_A_NEW a,
inserted b
WHERE a.id = b.id IF @@ROWCOUNT < @count
BEGIN
SET IDENTITY_INSERT TB_A_NEW ON INSERT INTO TB_A_NEW
(id,
NAME)
SELECT i.id,
i.NAME
FROM inserted i
LEFT JOIN TB_A_NEW n
ON i.id = n.id
WHERE n.NAME IS NULL
END
END GO
下面,我们来测试下SELECT、UPDATE、INSERT
Select

SELECT的结果就是刚才的插入10条记录,通过!
Insert

INSERT到视图中,会在TB_A_NEW表中新增一条记录,TB_A_OLD并没有任何变化。也就是自动分配到了动态表中,通过!
Update

然后我们继续更新id=2的记录,看看结果如何

最后,我们继续查询视图VIEW_TB_A,看看最终返回的结果。

看到最终结果11条,并且之前通过视图的更新,插入都返回了正确结果,通过!
至此,测试结束,通过视图和视图触发器,实现了统一名称访问,底册物理表分割动态表和静态表,测试过程并不包含delete,感兴趣的童鞋稍微琢磨下就出来了。
如果需要合并动态表与静态表,只需将id相同记录,更新到静态表,静态表不存在的id对应动态数据,插入到静态表中即可。
本次内容主意在实现方法,具体性能提升测试,日后再做说明,简单说下,就是触发器的逻辑处理,放在了物理表插入数据之前。将数据分类存储。所以逻辑处理开销会增加,但是物理存储分割。在数据量影响性能明显的场景中会非常适用。
如果你的数据库是SQL Server 2014,那么将静态和动态表定义到内存表中,动态数据又剥离了,具体提升多少这里不做猜测,有环境的朋友可以试试哦~
另:
利用以上方法,也可以解决肖桑(大菠萝)和笑东风提到的的主键int修改为bigint型问题
(http://www.cnblogs.com/TeyGao/p/4463389.html)
如果每日一次合并动态表与静态表数据,动态数据表就是当日的差异数据哦,扩展使用非常方便。
好~就到这里,再见吧~
【Paddy】如何将物理表分割成动态数据表与静态数据表的更多相关文章
- [转]在Sql Server中将字符串分割成表格数据示例
本文转自:http://www.lmwlove.com/ac/ID718 比如我们有一个字符串 ) select @appName ='UserID=admin,Account=ABC' 然后我们要以 ...
- 随笔 JS 字符串 分割成字符串数组 并动态添加到指定ID的DOM 里
JS /* * 字符串 分割成字符串数组 并动态添加到指定ID的DOM 里 * @id 要插入到DOM元素的ID * * 输入值为图片URL 字符串 * */ function addImages(i ...
- sql server 将字符串分割成表函数 strsplitetotable
在sql server里,调用存储过程时,经常需要将数据拼成字符串做为参数调用存储过程,而在储存过程中分割字符串虽然简单但麻烦,封装了该函数,可以将拼串分割成内存表返回,方便使用,返回的表字段从a,b ...
- SOLOv 2:实例分割(动态、更快、更强)
SOLOv 2:实例分割(动态.更快.更强) SOLOv2: Dynamic, Faster and Stronger 论文链接: https://arxiv.org/pdf/2003.10152. ...
- PHP+Mysql+easyui点击左侧tree菜单对应表名右侧动态生成datagrid加载表单数据(二)
关于tree菜单生成,参考我的另一篇博文地址tree 菜单 实现功能:点击左侧tree菜单中的table,右侧通过datagrid加载出该表对用的所有数据 难点:获取该表的所有列名,动态生成datag ...
- js 字符串分割成字符串数组 遍历数组插入指定DOM里 原生JS效果
使用的TP3.2 JS字符串分割成字符串数组 var images='{$content.pictureurl} ' ;结构是这样 attachment/picture/uploadify/20141 ...
- oracle根据分隔符将字符串分割成数组函数
--创建表类型 create or replace type mytype as table of number;--如果定义成varchar--CREATE OR REPLACE type myty ...
- SQL Server单表已700w+将普通表转换成分区表1
最近项目中,某个表数据量爆发时增长,单表已700w+,读写性能急剧下降,所以考虑加入分区表以解燃眉之急,后续还是要分表分库,当然这是后话.下面简要说一下将普通表转为分区表的步骤. 一.创建文件组 ...
- MySQL数据库中,将一个字段的值分割成多条数据显示
本文主要记录如何在MySQL数据库中,将一个字符串分割成多条数据显示. 外键有时是以字符串的形式存储,例如 12,13,14 这种,如果以这种形式存储,则不能直接与其他表关联查询,此时就需要将该字段的 ...
随机推荐
- SQL Server 2012 新特性
--Concat示例 ,null,'RTM') --Format实例 DECLARE @d DATETIME = GETDATE(); SELECT FORMAT( @d, 'd', 'en-US' ...
- [NHibernate]查看NHibernate生成的SQL语句
最近接触到一个用Spring.Net结合NHIbernate的项目,第一次使用,有很多配置,数据操作一旦出问题,很难找到原因,那么如何查看NHibernate发送给数据库的SQL语句呢? 当然我们可以 ...
- Java多线程之构造与发布
资料来源 http://www.ibm.com/developerworks/library/j-jtp0618/ http://www.javaspecialists.eu/archive/Issu ...
- mac linux 删除一个文件下边所有文件和文件夹
sudo rm -r -f 目录
- [RxJava^Android]项目经验分享 --- 递归实现
介绍一下业务逻辑:获取接口数据,根据接口内容判断是否需要继续获取数据. 本文使用递归思路,通过RxJava来实现此功能,获取数据的Observable直接用模拟的Observable.just()替代 ...
- BZOJ4597: [Shoi2016]随机序列
Description 你的面前有N个数排成一行.分别为A1, A2, … , An.你打算在每相邻的两个 Ai和 Ai+1 间都插入一个加号或者 减号或者乘号.那么一共有 3^(n-1) 种可能的表 ...
- 使用Prerender.io为angular项目做SEO
现在的项目的为了更好的分工明确,降低耦合都开始采用前后端分离的形式进式开发,我们也采用这种开发形式,前端用angular开发.虽说刚开始也遇各种坑,但是后期熟悉了之后简直爽呆.一个比较大的坑就是SEO ...
- C#编写windows服务
项目要求: 数据库用有一张表,存放待下载文件的地址,服务需要轮训表将未下载的文件下载下来. 表结构如下: 过程: VS--文件-->新建项目-->windows-->windows服 ...
- 临时更换hadoop-ugi
在用spark读写hdfs数据时,有时候当前用户对要读写的hdfs路径没有权限,需要临时改变用户去读写hdfs,操作完后回到原来的用户.我们的hdfs是没有权限认证的,一开始通过下面代码的方式来实现. ...
- 安装WAMP 及 修改MYSQL用户名 、 密码
1,下载并安装WAMP 2,启动服务后,找到MYSQL--MYSQL console--弹出命令窗口(刚开始没有初始用户名跟密码,可直接回车执行) 3,首先输入 use mysq;l---然后修改用户 ...