当前位置:   article > 正文

设置SQLServer数据库中某些表为只读的多种方法_sqlserver 数据表设置为只读

sqlserver 数据表设置为只读

翻译自:http://www.mssqltips.com/sqlservertip/2711/different-ways-to-make-a-table-read-only-in-a-sql-server-database/?utm_source=dailynewsletter&utm_medium=email&utm_content=headline&utm_campaign=2012614

在某些情况下需要把SQLServer的表设为只读,下面举出几种方法:

一般情况下会有几种情况需要你把数据库设为只读:

1.        Insert,Update,Delete 触发器

2.        Check 约束 和 Delete 触发器

3.        设置数据库为只读

4.        把表放到只读文件组中

5.        拒绝对象级别权限

6.        创建视图

在开始之前,先创建一个数据库及表作为示例:

  1. create database MyDB
  2. create table tblEvents
  3. (
  4. id int,
  5. logEvent varchar(1000)
  6. )
  7. insert into tblEvents
  8. values (1, 'Password Changed'), (2, 'User Dropped'), (3, 'Finance Data Changed')

nsert/Update/Delete触发器:

请注意这里使用的是INSTEADOF trigger,因为如果你使用了AFTER trigger,会在执行DELETE, UPDATE和INSERT语句时请求锁,会对写事务日志和回滚操作造成性能上的影响

  1. CREATE TRIGGER trReadOnly_tblEvents ON tblEvents
  2. INSTEAD OF INSERT,
  3. UPDATE,
  4. DELETE
  5. AS
  6. BEGIN
  7. RAISERROR( 'tblEvents table is read only.', 16, 1 )
  8. ROLLBACK TRANSACTION
  9. END

当用户执行insert/update/delete时,将提示以下错误:

Msg 50000, Level 16, State 1, Procedure trReadOnly_tblEvents, Line 7
tblEvents table is read only.
Msg 3609, Level 16, State 1, Line 1
The transaction ended in the trigger. The batch has been aborted.

使用 Check 约束和Delete 触发器:

现在先在表中添加一个check 约束“1=0”,意味着总是失败。它禁止你在任何行执行INSERT或者Delete操作。

首先,先禁用在上一步创建的触发器:disable trigger trReadOnly_tblEvents on tblevents
然后,添加约束:ALTER TABLE tblEvents WITH NOCHECK ADD CONSTRAINT chk_read_only_tblEvent CHECK( 1 = 0 )

执行以后,无论你执行任何一个INSERT/UPDATE语句,都将提示以下错误信息:

Msg 547, Level 16, State 0, Line 1
The UPDATE statement conflicted with the CHECKconstraint "chk_read_only_tblEvent". The conflict occurred indatabase "MyDB", table "dbo.tblEvents".
The statement has been terminated.

但是,该约束不会对DELETE操作造成影响,为此,需要再创建一个DDL触发器:

  1. CREATE TRIGGER trReadOnlyDel_tblEvents ON tblEvents
  2. INSTEAD OF
  3. DELETE
  4. AS
  5. BEGIN
  6. RAISERROR( 'tblEvents table is read only.', 16, 1 )
  7. ROLLBACK TRANSACTION
  8. END

设置数据库为只读:

你可以设置数据库为只读,这样就禁止对整个数据库的DDL/DML操作。可以使用以下语句:

  1. USE [master]
  2. GO
  3. ALTER DATABASE [MyDB] SET READ_ONLY WITH NO_WAIT
  4. GO

把表放到只读文件组:

可以在一个只读文件组中创建一个表:

  1. USE [master]
  2. GO
  3. ALTER DATABASE [MyDB] ADD FILEGROUP [READ_ONLY_TBLS]
  4. GO
  5. ALTER DATABASE [MyDB] ADD FILE ( NAME = N'mydb_readonly_tables', FILENAME = N'C:\JSPACE\myDBReadOnly.ndf' , SIZE = 2048KB , FILEGROWTH = 1024KB ) TO FILEGROUP [READ_ONLY_TBLS]
  6. GO
  7. DROP table tblEvents
  8. create table tblEvents
  9. (
  10. id int,
  11. logEvent varchar(1000)
  12. )
  13. ON [READ_ONLY_TBLS]
  14. ALTER DATABASE [MyDB] MODIFY FILEGROUP [READ_ONLY_TBLS] READONLY
  15. 任何对表的DML操作都会被拒绝,并返回以下错误信息:
  16. Msg 652, Level 16, State 1, Line 1
  17. The index "" for table "dbo.tblEvents" (RowsetId 72057594038845440) resides on a read-only filegroup ("READ_ONLY_TBLS"), which cannot be modified.

拒绝对象级别权限

可以通过DCL命令控制用户权限,但此步无法限制高级权限用户(如system admin,DatabaseOwner):

  1. DENY INSERT, UPDATE, DELETE ON tblEvents TO Jugal
  2. DENY INSERT, UPDATE, DELETE ON tblEvents TO Public

创建视图

为了替代直接访问表,可以使用视图:

  1. create view vwtblEvents
  2. as
  3. select ID, Logevent from tblEvents
  4. union all
  5. select 0, '0' where 1=0

在这个视图中,使用了UNION,只有在你确保有对应数量的列时才使用。在这个例子中,表有两列,所以使用两个输出列。同时,你也应该确保数据类型也一致。

当一个用户尝试通过INSERT/UPDATE/DELETE操作数据时,将得到以下错误信息:

Msg 4406, Level 16, State 1, Line 1
Update or insert of view or function 'vwtblEvents1' failed because it contains a derived or constant field.
Msg 4426, Level 16, State 1, Line 1
View'vwtblEvents1' is not updatable because the definition contains a UNIONoperator.


最后一步:

确认是否有必要用这些步骤来设置表为只读。

如果一个表总是只读,那么你应该放到只读文件组中。


声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/盐析白兔/article/detail/619186
推荐阅读
相关标签
  

闽ICP备14008679号