• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
西门吹雪
非我伤春春伤我,风一阵,雨一阵,好景不在哀残痕。 非我厌世世厌我,闲言来,碎语去,红颜憔悴心头冷。
博客园    首页    新随笔    联系   管理    订阅  订阅

从SQL SERVER 向ORACLE 8迁移的技术实现方案(1)

数据库端SQL语法的迁移   以下为常用的SQL语法迁移,包括数据类型、ID列向SEQUENCE迁移、表(主键、外键、CHECK、UNIQUE、DEFAULT、INDEX)、游标、存储过程、函数、触发器、常用SQL语法与函数几个方面,考虑SQL SERVER的实际情况,没有涉及ORACLE特有的PACKAGE、EXCEPTION等。在以下的描述中,将SQL SERVER的TRANSACT-SQL简称为T-SQL。在ORACLE中,其语法集称为PL/SQL。   <一> 数据类型的迁移   <1>、ORACLE端语法说明   在ORACLE中,分析其数据类型,大致可分为数字、字符、日期时间和特殊四大类。其中,数字类型有NUMBER;字符类型有CHAR与VARCHAR2;日期时间类型只有DATE一种;除此之外,LONG、RAW、LONG RAW、BLOB、CLOB和BFILE等数据类型都可视为特殊数据类型。   <2>、SQL SERVER端语法说明   在SQL SERVER中,参照上面对ORACLE的划分,数据类型也大致可分为数字、字符、日期时间和特殊四大类。数字类型又可分为精确数值、近似数值、整数、二进制数、货币等几类,其中,精确数值有DECIMAL[(P[, S])]与NUMERIC[(P[, S])];近似数值有FLOAT[(N)];整数有INT、SMALLINT、TINYINT;二进制数有BINARY[(N)]、VARBINARY[(N)];货币有MONEY、SMALLMONEY。字符类型有CHAR[(N)]与VARCHAR[(N)]。日期时间类型有DATETIME、SMALLDATETIME。除此之外,BIT、TIMESTAMP、TEXT和IMAGE、BINARY VARING等数据类型都可视为特殊数据类型。   <3>、从SQL SERVER向ORACLE的迁移方案   比较ORACLE与SQL SERVER在数据类型上的不同,当从SQL SERVER向ORACLE迁移时,可以做如下调整:   SQL SERVER   ORACLE   数字类型 DECIMAL[(P[, S])] NUMBER[(P[, S])] NUMERIC[(P[, S])] NUMBER[(P[, S])] FLOAT[(N)] NUMBER[(N)] INT NUMBER SMALLINT NUMBER TINYINT NUMBER MONEY NUMBER[19,4] SMALLMONEY NUMBER[19,4] 字符类型 CHAR[(N)] CHAR[(N)] VARCHAR[(N)] VARCHAR2[(N)] 日期时间类型 DATETIME DATE SMALLDATETIME DATE 其它 TEXT CLOB IMAGE BLOB BIT NUMBER(1) 方法:   公司原系统中的Money 用于金额时转换用number(14,2);用于单价时用 number(10,4)代替;   <二> ID列向SEQUENCE迁移   <1>、SQL SERVER端语法说明   在SQL SERVER中,可以将数据库中的某一字段定义为IDENTITY列以做主键识别,如:   jlbh numeric(12,0) identity(1,1) /*记录编号字段*/   CONSTRAINT PK_tbl_example PRIMARY KEY nonclustered (jlbh) /*主键约束*/   在这里,jlbh是一个ID列,在向具有该列的表插入记录时,系统将从1开始以1的步长自动对jlbh的值进行维护。   <2>、ORACLE端语法说明   但在ORACLE中,没有这样的ID列定义,而是采用另一种方法,即创建SEQUENCE。   如: /*--1、创建各使用地区编码表--*/ drop table LT_AREA; create table LT_AREA ( area_id number(5,0) NOT NULL, /*地区编码*/ area_name varchar2(20) NOT NULL, /*地区名称*/ constraint PK_LT_AREA PRIMARY KEY(area_id) ); /*--2、创建SEQUENCE,将列area_id 类ID化--*/ drop sequence SEQ_LT_AREA; create sequence SEQ_LT_AREA increment by 1 /*该SEQUENCE以1的步长递增*/ start with 1 maxvalue 99999; /*从1开始,最大增长到99999*/ /*--3、实际操作时引用SEQUENCE的下一个值--*/ insert into LT_AREA(area_id, area_name) values(SEQ_LT_AREA.NEXTVAL, '深圳'); insert into LT_AREA(area_id, area_name) values(SEQ_LT_AREA.NEXTVAL, '广州'); insert into LT_AREA(area_id, area_name) values(SEQ_LT_AREA.NEXTVAL, '北京'); /*--4、新插入连续三条记录后,下一条语句运行后,‘上海’地区的area_id为4--*/ insert into LT_AREA(area_id, area_name) values(SEQ_LT_AREA.NEXTVAL, '上海'); <3>、从SQL SERVER向ORACLE的迁移方案   根据以上分析,当从SQL SERVER向ORACLE迁移时,可以做如下调整:   1、去掉建表语句中有关ID列的identity声明关键字;   2、创建SEQUENCE,将此SEQUENCE与需类ID化的列对应;   3、在INSERT语句中对相应列引用其SEQUENCE值:SEQUENCENAME.NEXTVAL   实际上,处理以上情况在ORACLE中采用的方法为对有自动增长字段的表增加一插入前触发器(具体资料见后“触发器”一节),如下: CREATE OR REPLACE TRIGGER GenaerateAreaID BEFORE INSERT ON LT_AREA FOR EACH ROW Select SEQ_LT_AREA.NEXTVAL INTO :NEW.ID FROM DUAL; BEGIN END GenaerateAreaID;   GenaerateAreaID实际上修改了伪记录:new的area_id值。 :new最有用的一个特性----当该语句真正被执行时,:new中的存储内容就会被使用。所以系统每次都能自动生成新的号码。   <三> 表(主键、外键、CHECK、UNIQUE、DEFAULT、INDEX)   <1>、SQL SERVER端语法说明   有如下SQL SERVER语句: /* ------------------------ 创建employee 表------------------------ */ IF EXISTS(SELECT 1 FROM SYSOBJECTS WHERE NAME = ‘employee’ AND TYPE = ‘U’) DROP TABLE employee GO CREATE TABLE employee ( emp_id empid /*empid为用户自定义数据类型*/ /*创建自命名主键约束*/ CONSTRAINT PK_employee PRIMARY KEY NONCLUSTERED /*创建自命名CHECK约束*/ CONSTRAINT CK_emp_id CHECK (emp_id LIKE '[A-Z][A-Z][A-Z][1-9][0-9][0-9][0-9][0-9][FM]' or emp_id LIKE '[A-Z]-[A-Z][1-9][0-9][0-9][0-9][0-9][FM]'), /* CHECK约束说明:Each employee ID consists of three characters that represent the employee's initials, followed by a five digit number ranging from 10000 to 99999 and then the employee's gender (M or F). A (hyphen) - is acceptable for the middle initial. */ fname varchar(20) NOT NULL, minit char(1) NULL, lname varchar(30) NOT NULL, ss_id varchar(9) UNIQUE, /*创建唯一性约束*/ job_id smallint NOT NULL DEFAULT 1, /*设定DEFAULT值*/ job_lvl tinyint DEFAULT 10, /*设定DEFAULT值*/ /* Entry job_lvl for new hires. */ pub_id char(4) NOT NULL DEFAULT ('9952') /*设定DEFAULT值*/ REFERENCES publishers(pub_id), /*创建系统命名外键约束*/ /* By default, the Parent Company Publisher is the company to whom each employee reports. */ hire_date datetime NOT NULL DEFAULT (getdate()), /*设定DEFAULT值*/ /* By default, the current system date will be entered. */ CONSTRAINT FK_employee_job FOREIGN KEY (job_id) REFERENCES jobs(job_id) /*创建自命名外键约束*/ ) GO /* --------------------- 创建employee表上的index --------------------- */ IF EXISTS (SELECT 1 FROM sysindexes WHERE name = 'emp_pub_id_ind') DROP INDEX employee. emp_pub_id_ind GO CREATE INDEX emp_pub_id_ind ON employee(pub_id) GO <2>、ORACLE端语法说明   在ORACLE端的语法如下: /* ---------------------- 创建employee 表---------------------- */ DROP TABLE employee; CREATE TABLE employee ( emp_id varchar2(9) /*根据用户自定义数据类型的定义调整为varchar2(9)*/ /*创建自命名主键约束*/ CONSTRAINT PK_employee PRIMARY KEY NONCLUSTERED /*创建自命名CHECK约束*/ CONSTRAINT CK_emp_id CHECK (emp_id LIKE '[A-Z][A-Z][A-Z][1-9][0-9][0-9][0-9][0-9][FM]' or emp_id LIKE '[A-Z]-[A-Z][1-9][0-9][0-9][0-9][0-9][FM]'), /* CHECK约束说明:Each employee ID consists of three characters that represent the employee's initials, followed by a five digit number ranging from 10000 to 99999 and then the employee's gender (M or F). A (hyphen) - is acceptable for the middle initial. */ fname varchar2(20) NOT NULL, minit varchar2(1) NULL, lname varchar2(30) NOT NULL, ss_id varchar2(9) UNIQUE, /*创建唯一性约束*/ job_id number(5,0) NOT NULL /*这里考虑了SMALLINT的长度,也可调整为number*/ DEFAULT 1, /*设定DEFAULT值*/ job_lvl number(3,0) /*这里考虑了TINYINT的长度,也可调整为number*/ DEFAULT 10, /*设定DEFAULT值*/ /* Entry job_lvl for new hires. */ pub_id varchar2(4) NOT NULL DEFAULT ('9952') /*设定DEFAULT值*/ REFERENCES publishers(pub_id), /*创建系统命名外键约束*/ /* By default, the Parent Company Publisher is the company to whom each employee reports. */ hire_date date NOT NULL DEFAULT SYSDATE, /*设定DEFAULT值*/ /*这里,SQL SERVER的getdate()调整为ORACLE的SYSDATE*/ /* By default, the current system date will be entered. */ CONSTRAINT FK_employee_job FOREIGN KEY (job_id) REFERENCES jobs(job_id) /*创建自命名外键约束*/ ); /* -------------------- 创建employee表上的index -------------------- */ DROP INDEX employee. emp_pub_id_ind; CREATE INDEX emp_pub_id_ind ON employee(pub_id);
posted @ 2009-03-12 09:36  西门吹雪  阅读(318)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3