如何理解sql中的schema

本文通过生动的比喻,深入浅出地介绍了数据库对象与架构的概念。数据库对象包括表、视图等,它们需要被组织在架构(Schema)中,而架构则如同文件夹一样,用于存放这些对象。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

我相信很多人接触这些概念的时候一头雾水。要把这些概念理清楚真不是件容易的事,哪像原始社会,只要能分清楚什么能吃什么不能吃就行了。

   但是我始终坚信,每一个概念的产生必然是因为碰到了无法解决的问题。换句话说,如果没有它,必然会导致某些问题难以解决。所以我想从这个角度切入,希望能把这几个复杂而暧昧的多角关系从最实用的角度来阐述清楚。

   数据库对象。首先,数据库对象是比较容易懂的。所有的表,视图,存储过程,触发器都称为数据库对象。

   我们可以拿一个网站来做类比。一个网站包含很多的网页,图片,脚本文件,我们姑且称它为网站对象。

   显然,我们不可能把所有的网站对象都放到一个文件夹下面,同样道理,数据库对象也不可能象煮饺子一样就在数据库里这么一锅出。对于网站,我们通常会把不同模块的文件放在不同的子文件夹下,那么谁是存放数据库对象的文件夹呢?答案就是:架构(Schema.

   架构(Schema)。微软的官方说明(MSDN): "数据库架构是一个独立于数据库用户的非重复命名空间,您可以将架构视为对象的容器",详细参考http://technet.microsoft.com/zh-cn/library/ms190387.aspx.我们知道,在JAVA中,命名空间名其实就是文件夹名。因此我们非常明确一点:一个对象只能属于一个架构,就像一个文件只能存放于一个文件夹中一样。与文件夹不同的是,架构是不能嵌套的,如此而已。因此,我们要访问一个数据库对象的时候,通常应该是引用它的全名"架构名. 对象名",这点非常类似C#

   问:为什么有的时候写select * from tablename也可以执行呢?

   :这是因为default schema.当只写tablename时,Sql Server会自动加上当前登录用户的default schema


  如果此表不属于当前登录用户的default schema,将会提示无效的对象名。


  加上shcema以后成功。


  不过我们也可以更改当前用户的default schema,这时就可以不用加前缀了。

ALTER USER dbo WITH DEFAULT_SCHEMA =emdbuser;
当然,我们也可以改变此表的schema,相当于把这个表放到另一个文件夹,emdbuser放到dbo中。


Code
alterschema dbo TRANSFER emdbuser.Borrower
以上两种作法在真实项目中都不应该作为解决方案,因为它改变了原来的设置。我们最希望的是,即使我们以dbo登陆,我们也可以伪装成emdbuser来操作数据库对象,伪装完了还能切换回来。在Sql Server,刚好有这样的语句实现这个功能。


Code
SELECTCURRENT_USER AS 'emdbuser'
这种机制被称为上下文切换,操作完以后,可以实用REVERT命令切换回来。

详细解释参照MSDNhttp://msdn.microsoft.com/zh-cn/library/bb153640(SQL.90).aspx

问:如何根据表名获取一个表的Schema呢?

答:可以使用以下SQL语句从sys.objects视图和sys.schemas视图中获取。

结论:架构就是数据库对象的容器。数据库对象是饮料,架构就是杯子,谁拿杯子喝水呢?当然是用户,那么是不是一个用户只能用一个杯子,一个杯子是不是从一而终,只能给一个人用呢?。


### SQL Server Schema 使用指南 #### 创建架构 在 SQL Server 中,创建新架构可以使用 `CREATE SCHEMA` 命令。此命令允许定义新的命名空间来组织对象,如表、视图和存储过程。 ```sql CREATE SCHEMA UserSchema1; ``` 通过上述语句创建了一个名为 `UserSchema1` 的架构[^1]。 #### 更改架构所有权 更改架构的所有权可以通过 `ALTER AUTHORIZATION` 语句实现。这使得能够指定哪个数据库主体拥有特定的架构。 ```sql ALTER AUTHORIZATION ON SCHEMA::[UserSchema1] TO [dbo]; ``` 这条指令将 `UserSchema1` 架构的所有权转移给 `dbo` 用户。 #### 查询现有架构列表 为了查看当前数据库中存在的所有架构,可执行如下查询: ```sql SELECT name AS 'Schema Name' FROM sys.schemas; ``` 该查询返回一个包含所有已定义架构名称的结果集[^2]。 #### 删除架构 删除不再需要使用的架构应谨慎操作,因为会同时移除属于该架构下的所有对象。使用 `DROP SCHEMA` 来完成这一任务之前要确认没有任何依赖关系存在。 ```sql DROP SCHEMA IF EXISTS OldSchemaName; ``` 这里展示了如何安全地尝试删除名为 `OldSchemaName` 的架构;如果它不存在,则不会抛出错误而是继续执行后续代码。 #### 设置默认架构 当为登录名设置默认架构时,在未显式指明的情况下,默认会在该用户的上下文中查找对象。这对于简化跨多个模式的工作流程非常有用。 ```sql ALTER USER [UserName] WITH DEFAULT_SCHEMA = [DefaultSchemaName]; ``` 这段脚本修改了 `[UserName]` 登录者的默认架构至 `[DefaultSchemaName]`。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值