SQLSERVER将一个文件组的数据移动到另一个文件组

移动数据:

1、有表分区

2、没有表分区

我这里只讨论没有表分区的情况


例子

比如:你有三个文件组,其中一个是主文件组

测试脚本:

  1. USE master
  2. GO
  3.  
  4. IF EXISTS(SELECT * FROM sys.[databases] WHERE [database_id]=DB_ID('Test'))
  5. DROP DATABASE [Test]
  6.  
  7. --1.创建数据库
  8. CREATE DATABASE [Test]
  9. GO
  10.  
  11. USE [Test]
  12. GO
  13.  
  14. --2.创建文件组
  15. ALTER DATABASE [Test]
  16. ADD FILEGROUP [FG_Test_Id_01]
  17.  
  18. ALTER DATABASE [Test]
  19. ADD FILEGROUP [FG_Test_Id_02]
  20.  
  21. --3.创建文件
  22. ALTER DATABASE [Test]
  23. ADD FILE
  24. (NAME = N'FG_TestUnique_Id_01_data',FILENAME = N'E:\FG_TestUnique_Id_01_data.ndf',SIZE = 1MB, FILEGROWTH = 1MB )
  25. TO FILEGROUP [FG_Test_Id_01];
  26.  
  27. ALTER DATABASE [Test]
  28. ADD FILE
  29. (NAME = N'FG_TestUnique_Id_02_data',FILENAME = N'E:\FG_TestUnique_Id_02_data.ndf',SIZE = 1MB, FILEGROWTH = 1MB )
  30. TO FILEGROUP [FG_Test_Id_02];
  31.  
  32. --4.创建表,这个表的数据存放在[FG_Test_Id_01] 文件组上
  33. CREATE TABLE aa(id INT ,cname NVARCHAR(4000)) ON [FG_Test_Id_01]
  34. GO
  35.  
  36. --5.插入数据
  37. INSERT INTO [dbo].[aa]
  38. SELECT 1,REPLICATE('s',3000)
  39. GO 500
  40.  
  41. --6.查询数据
  42. SELECT * FROM [dbo].[aa]
  43.  
  44. --7.创建聚集索引在[FG_Test_Id_02]文件组上
  45. CREATE CLUSTERED INDEX PK_ID ON [dbo].[aa]([id]) WITH(ONLINE=ON) ON [FG_Test_Id_02]
  46. GO
  47.  
  48. --8.我们查看一下文件组的逻辑文件名
  49. EXEC [sys].[sp_helpdb] @dbname = TEST -- sysname
  50.  
  51. --9.收缩一下FG_Test_Id_01文件组文件
  52. DBCC SHRINKFILE(FG_TestUnique_Id_01_data,1)
  53.  
  54. --10.你可以选择drop掉聚集索引,也可以选择不drop掉聚集索引
  55. DROP INDEX PK_ID ON [dbo].[aa]
  56.  
  57. USE master
  58. GO
  59. DROP DATABASE [Test]

上面的脚本虽然简单,但是隐藏了非常多的知识点

知识点1:创建了两个文件组,现在数据库有三个文件组,包括主文件组,当你不指定任何参数的时候默认创建出来的数据文件是1MB大小

知识点2:插入数据,因为表是创建在[FG_Test_Id_01]文件组上,所以数据都会放在E:\FG_TestUnique_Id_01_data.ndf

  1. CREATE TABLE aa(id INT ,cname NVARCHAR(4000)) ON [FG_Test_Id_01]
  2. GO

知识点3:创建聚集索引,其实这句话里面包含了几个动作,在E:\FG_TestUnique_Id_02_data.ndf文件上分配页面,并把aa表的数据

放进去E:\FG_TestUnique_Id_02_data.ndf文件,其实这里聚集索引成为了移动数据的中介,我在

SQLSERVER聚集索引与非聚集索引的再次研究(上)文章写到:聚集索引叶子节点就是数据,我们把聚集索引(一定要是聚集索引,非聚集索引不是)

建立在E:\FG_TestUnique_Id_02_data.ndf文件上实际上就是把数据页面聚集索引页面移动到E:\FG_TestUnique_Id_02_data.ndf文件里

因为SQLSERVER是没有 ALTER TABLE aa(id INT ,cname NVARCHAR(4000)) ON [FG_Test_Id_01] 这种语法上

就是说你一旦建表并且表中已经有数据之后,如果你要移动表数据,只能通过聚集索引这个中介来移动表数据

  1. CREATE CLUSTERED INDEX PK_ID ON [dbo].[aa]([id]) WITH(ONLINE=ON) ON [FG_Test_Id_02]
  2. GO

从下图可以看出数据都已经移动到E:\FG_TestUnique_Id_02_data.ndf文件上

知识点4:为什麽要加上WITH(ONLINE=ON)??如果你的应用是不能停机的话,加上WITH(ONLINE=ON)就可以在线的创建索引

详情参考:CREATE INDEX (Transact-SQL)

  1. CREATE CLUSTERED INDEX PK_ID ON [dbo].[aa]([id]) WITH(ONLINE=ON) ON [FG_Test_Id_02]
  2. GO

知识点5:为什麽要收缩[FG_Test_Id_01]文件组文件E:\FG_TestUnique_Id_01_data.ndf

因为数据已经移动到E:\FG_TestUnique_Id_02_data.ndf文件上了,既然已经移到E:\FG_TestUnique_Id_02_data.ndf文件上

为什麽E:\FG_TestUnique_Id_01_data.ndf文件还显示5MB大小??

大家可以看一下这篇文章:聚集索引表插入数据和删除数据的方式是怎样的

我delete了数据,SQLSERVER却没有完全释放空间,其实这里移动数据到别的文件/文件组相当于delete了数据了

但是SQLSERVER并没有释放这些空间,所以我需要收缩一下FG_Test_Id_01文件组文件

  1. --9.收缩一下FG_Test_Id_01文件组文件
  2. DBCC SHRINKFILE(FG_TestUnique_Id_01_data,1)

知识点6:DBCC SHRINKFILE(FG_TestUnique_Id_01_data,1)

SHRINKFILE的单位是MB,上面的语句就是收缩到1MB大小,其实这里如果E:\FG_TestUnique_Id_01_data.ndf文件有数据的话

并且需要占用2MB大小的空间,那么您使用DBCC SHRINKFILE(FG_TestUnique_Id_01_data,1)这句话只能收缩到2MB大小

并不会收缩到1MB大小的,因为这些数据需要占用空间,你怎麽收缩都收缩不了的,不信的话您们可以测试一下

相关文章:

DBCC SHRINKFILE (Transact-SQL)

user database的initial size和dbcc shrinkfile


总结

虽然移动数据的动作比较简单,但是知识点挺多的,有些人只知道怎麽做,不知道为什么我觉得这样不好

寻根问底是我的特性o(∩_∩)o 哈哈

注意:移动数据只能一张表一张表的移动,如果表里预先已经有聚集索引,需要先drop掉

移动之前查一下表中的数据在哪个文件组中

  1. USE [Northwind]
  2. GO
  3. EXEC [sys].[sp_help] @objname = N'[dbo].[Categories]' -- nvarchar(776)

如有不对的地方,欢迎大家拍砖o(∩_∩)o

2014-1-19 补充:

有表分区的方法

大概有三种

先创建新的数据文件,文件组,分区方案,分区函数

例如创建三个新的数据文件和文件组,分区方案和分区函数对于这三个新的数据文件和文件组

旧表:无论已经分区或者还没有分区都适用下面三种方法:

方法一:建立中间表(新表),中间表建立在新的分区方案上,然后用insert into 新表 select * from 旧表的方法,插入完毕之后

drop掉旧表

方法二:建立中间表,中间表建立在新的分区方案上,然后用switch to,因为新分区方案有三个分区

那么,switch to只能够将表的全部数据切换到其中一个分区

  1. USE Sales
  2. GO
  3. ALTER TABLE 旧表 SWITCH PARTITION 1 TO 新表 PARTITION 1
  4. GO

或者

  1. USE Sales
  2. GO
  3. ALTER TABLE 旧表 SWITCH PARTITION 1 TO 新表 PARTITION 2
  4. GO

或者

  1. USE Sales
  2. GO
  3. ALTER TABLE 旧表 SWITCH PARTITION 1 TO 新表 PARTITION 3
  4. GO

上面的三条语句就会把旧表的所有数据移动到新表的某个分区,至于移动到哪个分区由最后那个数字来指定PARTITION 要移动到的分区的数字

完成后drop掉旧表

方法三:drop掉旧表的聚集索引(如果有),然后在旧表上创建一个聚集索引,创建聚集索引的时候指定新的分区方案

这样就会把表数据移动到新分区方案,即新的数据文件里,这个方法跟没有表分区的数据移动是一样的

参考文章:http://blog.csdn.net/smallfools/article/details/4930810

删除文件和删除文件组

删除失效文件组 (SQL Server)

  1. --Transact-SQL
  2. USE master;
  3. GO
  4. ALTER DATABASE AdventureWorks2012
  5. REMOVE FILE test1dat3 ;
  6. ALTER DATABASE AdventureWorks2012
  7. REMOVE FILE test1dat4 ;
  8. GO
  9.  
  10. --Transact-SQL
  11. USE master;
  12. GO
  13. ALTER DATABASE AdventureWorks2012
  14. REMOVE FILEGROUP Test1FG1 ;
  15. GO

移数据

  1. -- 备份数据库
  2. backup database xx to disk =xx
  3.  
  4. DBCC SHRINKFILE(BSDF2017,EMPTYFILE )
  5.  
  6. ALTER DATABASE AdventureWorks2012
  7. REMOVE FILE BSDF2017 ;

SQLSERVER将一个文件组的数据移动到另一个文件组的更多相关文章

  1. SQLSERVER将数据移到另一个文件组之后清空文件组并删除文件组

    SQLSERVER将数据移到另一个文件组之后清空文件组并删除文件组 之前写过一篇文章:SQLSERVER将一个文件组的数据移动到另一个文件组 每个物理文件(数据文件)对应一个文件组的情况(一对一) 如 ...

  2. VLOOKUP函数将一个excel表格的数据匹配到另一个表中

    sklearn实战-乳腺癌细胞数据挖掘(博主亲自录制视频) https://study.163.com/course/introduction.htm?courseId=1005269003& ...

  3. sql将一个表中的数据插入到另一个表中

    sql将一个表中的数据插入到另一个表中 列名不一定要相同,只要你在HH中列出要插入列的列表跟select   from   mm表中的选择的列的列表一一对应就可以了,当然两边的数据类型应该是兼容的.  ...

  4. shell脚本就是由Shell命令组成的执行文件,将一些命令整合到一个文件中,进行处理业务逻辑,脚本不用编译即可运行。它通过解释器解释运行,所以速度相对来说比较慢。

    shell脚本?在说什么是shell脚本之前,先说说什么是shell. shell是外壳的意思,就是操作系统的外壳.我们可以通过shell命令来操作和控制操作系统,比如Linux中的Shell命令就包 ...

  5. 在C#中如何确定一个文件是不是文本文件,以及如何确定一个文件的类型

    博客搬到了fresky.github.io - Dawei XU,请各位看官挪步.最新的一篇是:在C#中如何确定一个文件是不是文本文件,以及如何确定一个文件的类型.

  6. 工具:从一个文件夹中复制jar到另一个文件夹中

    工具类:从一个文件夹中复制jar到另一个文件夹中 需要的小伙伴可以试一试,很爽哦,有时候真的很需要! 需求:当我们拿到一个maven项目时,而maven项目的jar包都是通过pom.xml文件管理的, ...

  7. sql-server的添加数据库文件(日志数据)以及收缩数据库文件(日志数据)

    环境: SSMS sql-server2016 一.为数据库添加数据文件 添加日志数据文件 以下是添加数据文件和日志文件的代码 ALTER DATABASE [joinbest] ADD FILE ( ...

  8. 如何将一个excel表格的数据匹配到另一个表中

    我们在操作excel表的时,有时需要将一个excel表中的数据匹配到另一个表中,那么就需要用到VLOOKUP函数,VLOOKUP函数是Excel中的一个纵向查找函数,VLOOKUP是按列查找,最终返回 ...

  9. 【VS开发】使用WinPcap编程(4)——把网络数据包存储到一个文件中

    这里用到的数据结构是pcap_dumper_t,这也是一个相当于文件描述符的东西,我们在用的时候先指定pcap_dumper_t *dumpfp; 使用两个函数来存储网络数据,一个是pcap_dump ...

随机推荐

  1. .NetCore MVC中的路由(2)在路由中使用约束

    p { margin-bottom: 0.25cm; direction: ltr; color: #000000; line-height: 120%; orphans: 2; widows: 2 ...

  2. PHP源码分析-变量

    1. 变量的三要素变量名称,变量类型,变量值 那么在PHP用户态下变量类型都有哪些,如下: // Zend/zend.h #define IS_NULL 0 #define IS_LONG 1 #de ...

  3. Electron使用与学习--(页面间的通信)

    目录结构: index.js是主进程js. const electron = require('electron') const app = electron.app const BrowserWin ...

  4. [自翻]fasthttp中文文档(持续更新)

    fasthttp是由valyala编写的声称快于Go官方标准库net/http包十倍的快速HTTP实现.从各方的性能测试结果来看(评测一.评测二),fasthttp作为当下最快的http Go语言包当 ...

  5. EF上下文对象线程内唯一性与优化

    在一次请求中,即一个线程内,若是用到EF数据上下文对象,就创建一个,这也加是很多人的代码中习惯在使用上下文对象时,习惯将对象建立在using中,也是为了尽早释放上下文对象, 但是如果有一个业务逻辑调用 ...

  6. python 数据类型 --- 集合

    1. 注意列表和集合的区别 set 列表表现形式: list_1 = [1,3,4];  集合表现形式:set_1= set() list_1 = [1,2,3,4,23,4,2] print(lis ...

  7. 用Kotlin创建第一个Android项目(KAD 01)

    原文标题:Create your first Android project using Kotlin (KAD 01) 作者:Antonio Leiva 时间:Nov 21, 2016 原文链接:h ...

  8. ORACLE中STATUS为INACTIVE但是SERVER为SHARED状态的会话浅析

    我们知道当ORACLE数据库启用共享服务器模式时,通过共享服务器模式连接到数据库的会话是有一些特征的.在v$session里面,其SERVER的状态一般为SHARED和NONE, 为SHARED时,表 ...

  9. JavaMail发送邮件

    发送邮件包含的内容有: from字段  --用于指明发件人 to字段      --用于指明收件人 subject字段  --用于说明邮件主题 cc字段     -- 抄送,将邮件发送给收件人的同时抄 ...

  10. Java中的进程与线程(总结篇)

    详细文档: Java中的进程与线程.rar 474KB 1/7/2017 6:21:15 PM 概述: 几乎任何的操作系统都支持运行多个任务,通常一个任务就是一个程序,而一个程序就是一个进程.当一个进 ...