在Oracle数据库体系中,有一种极为实用的特性——Database Link(数据库链接),它能够将物理上分散于多台服务器的多个数据库,逻辑上整合为一个统一的整体。用户只需通过网络即可直接访问远程数据库中的数据,而背后服务器之间的协同、数据的分布策略以及网络细节的处理,对前端应用而言完全透明。开发人员无需关心网络连接如何建立、数据位于哪个节点、服务器之间如何“握手”——所有这些底层复杂性都由Database Link自动完成。
要创建一个Database Link,首先需要在发起链接的数据库上配置链接字符串,简单来说,就是为远程数据库在本地定义一个“网络服务名”。具体的配置方法可参考客户端连接服务器的相关内容,此处不再赘述。
数据库全局名称可以通过以下命令查询:
SELECT * FROM GLOBAL_NAME;
如需修改该参数的值,可执行:
ALTER SYSTEM SET GLOBAL_NAME = TRUE/FALSE;
接下来分两种场景进行说明。
场景一:当参数global_names = false时
在此设置下,Database Link的名称不必与远端数据库的名称保持一致。Oracle数据库之间可通过以下语法建立连接:
CREATE [PUBLIC] DATABASE LINK link
CONNECT TO username IDENTIFIED BY password
USING 'connectstring'
需要特别注意的是,将global_names设为false后,必须重启数据库才能生效,或者使用sys用户执行。此外,创建Database Link的账户必须拥有CREATE DATABASE LINK或CREATE PUBLIC DATABASE LINK系统权限,而用于登录远程数据库的账号则需具备CREATE SESSION权限。这两类权限都已包含在CONNECT角色中(CREATE PUBLIC DATABASE LINK权限则在DBA角色内)。
公用数据库链接对数据库中的所有用户开放,而私有链接则仅限创建者使用。用户无法将私有数据库链接授权给其他用户——它要么是公用的,要么是私有的,没有中间状态。
另外,创建链接时还可以采用“缺省登录”方式,即不指定远程用户名和密码:
create public database link zrhs_link using 'zrhs';
在这种方式下,Oracle会使用当前用户的用户名和口令去登录远程数据库。USING后面紧随的是链接字符串,也就是远程数据库的网络服务名,该服务名保存在TNSNAMES.ORA文件中,文件中定义了协议、主机名、端口和数据库名。
接下来举一个实际操作的例子。假设有两台数据库服务器db_A和db_B,db_A下的用户user_a希望借助Database Link访问db_B上user_b的数据,具体流程如下:
1. 首先,user_b这个用户必须拥有SELECT权限。
2. 在db_B的tnsnames.ora中新建一个连接字符串tnsname_BtoA,使db_B能够连接到db_A。
3. 在db_A的tnsnames.ora中新建一个连接字符串tnsname_AtoB,使db_A能够连接到db_B。
4. 然后在db_A上创建Database Link:
create database link link_AtoB_name
connect to user_b identified by user_b的password
using 'tnsname_AtoB';
此处link_AtoB_name必须与远程数据库的全局名(SID)保持一致,tnsname_AtoB即指向远程数据库的连接字符串。
5. 测试连接是否成功:select * from dual@link_AtoB_name;如果返回结果DUMMY X,说明配置无误。
删除数据库链接的语句非常简单:
DROP [PUBLIC] DATABASE LINK zrhs_link;
引用数据库链接
通常情况下,引用Database Link时直接将其放在表名或视图名之后,中间用“@”分隔:
SELECT * FROM worker@zrhs_link;
如果某个链接频繁使用,可以创建一个本地同义词,避免每次都要写一长串:
CREATE SYNONYM worker_syn FOR worker@zrhs_link;
或者创建一个本地远程视图:
CREATE VIEW worker AS SELECT * FROM worker@zrhs_link where ...;
这样一来,该视图可以像本地普通视图一样被对待,也可以授权给其他用户。但需要注意,被授权的用户必须拥有访问该Database Link的权限。
另一种常见情况是:要访问的表并不在Database Link指定的远程账号下,但该账号拥有访问这张表的权限。此时,表名前面需要加上用户名:
SELECT * FROM camel.worker@zrhs_link;
场景二:当参数global_names = true时
这种情况下要求非常严格——Database Link的名称必须与远端数据库的全局名称完全相同。数据库全局名称同样可以通过SELECT * FROM GLOBAL_NAME;查询。
创建Database Link的标准语法为:
sql>create database link 数据库链路名
connect to 用户名 identified by 口令
using '主机字符串名';
例如:
sql>create database link ora9i.us.oracle.com
connect to scott identified by tiger
using 'sun';
这里有几点需要特别注意:
1. Database Link的名称必须与远程数据库的全局数据库名(global_name)完全一致。
2. 用户名和口令必须是远程数据库上合法的账户信息。
3. 主机字符串即为本机tnsnames.ora文件中定义的连接串。
4. 两个同名的数据库之间无法建立Database Link。
随后,你可以像下面这样访问远程数据库:
sql>select * from 表名@ora9i.us.oracle.com;
通过Database Link还可以创建快照(snapshot),实现远程数据的自动传输。
最后,如果想查看数据库中所有的Database Link,可以在系统管理员权限下运行:
SQL>select owner,object_name from dba_objects where object_type='DATABASE LINK';
