PL-SQL
✍ dations ◷ 2025-07-02 12:31:05 #PL-SQL
PL/SQL(Procedural Language/SQL)是甲骨文公司专有的SQL扩展语言,应用在甲骨文公司的Oracle数据库系统。一些的SQL数据库管理系统也提供了类似的扩展SQL语言。PL/SQL的的语法非常类似于Ada,而且像1980年代的Ada编译器一样,PL/SQL的运作系统使用Diana作为中介语言。
重要的是PL/SQL紧密的结合在Oracle数据库里面。
PL/SQL是Oracle数据库使用的三种语言的其中之一,另外两个是SQL和Java。
Oracle连接操作(left join ,right join,full join)的语法确与SQL标准完全不同,没有左连接与右连接的概念,也不支持全外连接。Oracle语法如下所示:
select * from t1,t2 where t1.id=t2.id(+)
采用(+)来表示外连接,在Oracle中它相当于左连接。Oracle9i中增加了标准外连接的语法支持,但使用不广。
物化视图(materialized view)或译为实体化视图。与普通视图关系不同,物化视图更像是一个表,保存了实实在在的数据,并且可以与表一样定义存储参数,可以与表一样使用select,insert,update,delete。在其它数据库中也有和物化视图相似的解决方案,DB2叫物化查询表(materialized query table),sqlserver有索引视图,但是索引视图仅是起优化作用,与oracle的物化视图还不太一样。
采用本地的number类型做指数或对数运算,与标准的浮点数性能可能会相差50倍。好在Oracle10g中增加了高效的浮点类型binary_float,binary_double,从而弥补了浮点数性能的问题。
variable_name number)] = 0;
- NUMBER可选指定precision (P)与scale (S)。精度表示十进制有效数字的个数,最多不能超过38个有效数字(实际支持39-40位十进制数字)。Scale的范围为。Scale为正数时,表示从小数点到最不重要的十进制有效数字的个数;为负数时,其绝对值表示从最不重要的十进制有效数字到小数点的位数。如果没有指定精度,precision与scale默认为最大的取值区间。如果指定了精度,没有指定scale,scale默认为0。内部存储格式是变长阿拉伯数字的字节数组:
- 首字节为长度值,最大22;如果为NULL,则该字节值为255(0xFF)
- 第二字节是符号和指数字节(sign bit/exponent),其最高比特为符号位,1表示正数,0表示负数;其余7比特构成基为100的指数值,取值范围,NUMBER数据类型的取值范围为
- 两个字节255与101表示正无穷
- 单字节0表示负无穷
INTEGER是NUMBER的子类型,它等同于NUMBER(38,0),用来存储整数。若插入、更新的数值有小数,则会被四舍五入。FLOAT类型也是NUMBER的子类型。Float(n),数 n 指示位的精度,可以存储的值的数目。n 值的范围可以从 1 到 126。若要从二进制转换为十进制的精度,请将 n 乘以 0.30103。要从十进制转换为二进制的精度,请用 3.32193 乘小数精度。126 位二进制精度的最大值是大约相当于 38 位小数精度。BINARY_FLOAT 是 32 位、 单精度浮点数字数据类型。可以支持至少6位精度,每个 BINARY_FLOAT 的值需要 5 个字节,包括长度字节。BINARY_DOUBLE 是为 64 位,双精度浮点数字数据类型。每个 BINARY_DOUBLE 的值需要 9 个字节,包括长度字节。其它数值类型:dec, decimal, double precision, int, numeric, real, smallint, binary_integer.
字符串数据类型依据存储空间分为两种:
串的开头存储了串的长度。如果串的长度小于或等于250(0x01~0xFA), Oracle 会使用1 个字节来表示长度。对于所有长度超过250 的串,都会在一个标志字节0xFE 后跟有两个字节来表示长度。
chr(0)表示的不可见字符,即我们通常所说的
-- 字段translated_name是NCHAR类型,则需要如下书写:SELECT translated_description FROM product_descriptions WHERE translated_name = N'LCD Monitor 11/PM'; variable_name varchar2(20) = 'Text';-- e.g.: address varchar2(20) := 'lake view road';
日期类型
variable_name date = to_date('01-01-2005 14:20:23', 'DD-MM-YYYY hh24:mi:ss');
- Date类型可以表示日期与时间。精度到秒。日期范围可以是公元前4712年1月1日至公元9999年12月31日。占用7个字节的存储空间。第1字节:世纪+100;第2字节:年; 第3字节:月; 第4字节:天; 第5字节:小时+1; 第6字节:分+1;第7字节:秒+1。其中时间可以忽略。但无法只表示时间而忽略日期。Oracle Datatypes(页面存档备份,存于互联网档案馆)
- TIMESTAMP类型:7字节或11字节的定宽日期/时间数据类型。可以包含小数秒,小数位数可以指定为0-9,默认为6位,所以最高精度可以到ns(纳秒).如果精度为0,则用7字节存储,与date类型功能相同,如果精度大于0则用11字节存储。
- TIMESTAMP WITH TIME ZONE类型:TIMESTAMP类型的变种,它包含了时区偏移量的值
- TIMESTAMP WITH LOCAL TIME ZONE类型:
- INTERVAL YEAR TO MOTH:
- INTERVAL DAY TO SECOND:
函数TO_DATE
把字符串转换为日期值。
to_date('31-12-2004', 'dd-mm-yyyy') to_date ('31-Dec-2004', 'dd-mon-yyyy', 'NLS_DATE_LANGUAGE = American')
函数TO_CHAR (date_string, format_string)
把日期值转换为字符串。
PL/SQL支持使用ANSI日期与时间间隔值 The following clause gives an 18-month range:
示例:
WHERE dateField BETWEEN DATE '2004-12-30' - INTERVAL '1-6' YEAR TO MONTH AND DATE '2004-12-30'
create table T( C1 DATE, C2 TIMESTAMP(9)); insert into t(c1,c2) values(date'2010-2-12',timestamp'2010-2-12 13:24:52.234123211');insert into t(c1,c2) values( to_date('2010-2-12 10:20:30','YYYY-MM-DD HH24:MI:SS'), to_timestamp('2010-2-12 13:24:52.123456','YYYY-MM-DD HH24:MI:SS.FF6')); SQL> select c1,dump(c1) c1_d,c2,dump(c2) c2_d from t;
- sysdate--返回当前系统日期和时间,精确到秒
- systimestamp--返回当前系统日期和时间,精确到毫秒
日期型数据可以与数值加减得到新的日期,加减数值单位为天
内置的LOB数据类型包括BLOB、CLOB、NCLOB、BFILE(外部存储)的大型化和非结构化数据,如文本、图像、视屏、空间数据存储。
均为较老的数据类型,Oracle不建议使用。
在数据库中的每一行都有一个地址。然而,一些表行的地址不是物理或永久的,或者不是ORACLE数据库生成的。例如,索引组织表行地址存储在索引的叶子,可以移动。例如,外部表的ROWID(如通过网关访问DB2表)不是标准的ORACLE的rowid。
ORACLE使用通用的ROWID(UROWIDs)的存储地址的索引组织表和外表。10个字节,格式为: ********.****.****,*为0或1。NROWID类型为二进制数据表中记录的唯一行号,最大长度4000字节
索引组织表有逻辑urowids的,和外表的外urowids。UROWID这两种类型的存储在ROWID(堆组织的表的物理行id)。
创建基于逻辑的rowid在表中的主键。逻辑的rowid不会改变,只要主键不改变。索引组织表的ROWID伪UROWID数据类型。你可以访问这个伪列,你会堆组织表的ROWID伪(即使用一个SELECT ...ROWID语句)。如果你想存储的rowid索引组织表,那么你就可以定义一列的表型UROWID到列检索值的ROWID伪。
定义一个变量,其类型是指定表的指定列的数据类型:
Variable_name Table_name.Column_name%type;
自定义类型
程序员自定义类型:
type data_type is record (field_1 type_1 = xyz, field_2 type_2 := xyz, ..., field_n type_n = xyz);
例如:
declare type t_address is record ( name address.name%type, street address.street%type, street_number address.street_number%type, postcode address.postcode%type); v_address t_address;begin select name, street, street_number, postcode into v_address from address where rownum = 1;end;
可以使用点表示(dot-notation)获取结构中的域:
v_address.street = 'High Street';"
自增长数据类型
Oracle的数据类型里没有自增长(auto-incremental)字段类型,Oracle的官方解决方案是采用sequence实现。insert的时候需要用sequence.nextval。需要增加一张专用表来保存自增长字段的表和字段名,每次新增记录时都把这个记录值加1再取出使用。
以下的代码展示了IF-THEN-ELSIF结构。ELSIF和ELSE部分是可选的,从而可以创建更简单的IF-THEN或者IF-THEN-ELSE结构。
IF x = 1 THEN sequence_of_statements_1;ELSIF x = 2 THEN sequence_of_statements_2;ELSIF x = 3 THEN sequence_of_statements_3;ELSIF x = 4 THEN sequence_of_statements_4;ELSIF x = 5 THEN sequence_of_statements_5;ELSE sequence_of_statements_N;END IF;
CASE语句简化了一些大的IF-THEN-ELSE结构。
CASE WHEN x = 1 THEN sequence_of_statements_1; WHEN x = 2 THEN sequence_of_statements_2; WHEN x = 3 THEN sequence_of_statements_3; WHEN x = 4 THEN sequence_of_statements_4; WHEN x = 5 THEN sequence_of_statements_5; ELSE sequence_of_statements_N;END CASE;
CASE语句可以使用预定义的选择符:
CASE x WHEN 1 THEN sequence_of_statements_1; WHEN 2 THEN sequence_of_statements_2; WHEN 3 THEN sequence_of_statements_3; WHEN 4 THEN sequence_of_statements_4; WHEN 5 THEN sequence_of_statements_5; ELSE sequence_of_statements_N;END CASE;
阵列
循环
比较常用的有三种形式。
For ... Loop ... End Loop;
loop循环是最简单的循环语句,用于无限制的循环执行语句:
Loop ... End Loop;
while...loop;
如果退出循环,则必须使用exit语句终止循环。exit语句分两种格式:
exit:该格式的语句用于无条件强迫终止循环。exit...when:该格式用于有条件终止循环,首先检测when的条件是否满足。
例1:
loop if...then exit; end if;end loop;
例2:
loop exit when;end loop;
while...loop循环在执行语句时,首先检测条件的值。例如:
declare div_name varchar2(20); div_num integer=1;begin while div_num<10 loop select name into div_name from div_tab where div_author ='A000'||to_char(div_num); div_num=div_num+1; end loop;end;
for...loop循环可以限定循环的次数例如:
declare div_name varchar2(20); div_num integer:=1;begin for div_num in 1..9 loop select name into div_name from div_tab where div_author='A000'||to_char(div_num); end loop;end;
例如:
declare div_name varchar2(20); div_num integer=1;begin for div_num in 1..9 loop EXIT WHEN div_num >7; DBMS_OUTPUT.put_line(div_num); end loop;end;
类似的语言
功能近似PL/SQL的程序语言和其他关系型数据库:
相关
- 何塞·玛利亚·奥拉萨巴尔何塞·马里亚·奥拉萨瓦尔(西班牙语:José María Olazábal,1966年2月5日-),西班牙职业高尔夫球选手。他赢得过两个大满贯赛事冠军。分别是1994年的美国名人赛和1999年的美国名人
- 嫦娥二号嫦娥二号是中国的第二颗绕月人造卫星。它是建基于探月工程一期的嫦娥一号备份星进行技术改进,作为二期工程的先导星,且命名为嫦娥二号。嫦娥二号主要是用作试验、验证部分新技
- 沈世宏沈世宏(1949年6月19日-)籍贯江西省大庾县,中华民国政治人物,迁台外省人第二代,曾任台北市环保局长。曾任中华民国环保署署长达六年,为迄今在任最久的署长。2009年1月9日,高雄县潮老
- 古罗马与葡萄酒古罗马在葡萄酒的历史中有着重要的地位,意大利半岛最早出现葡萄酒是在古希腊殖民地和伊特鲁里亚,随着罗马帝国的兴起,新的酿酒技术蔓延到帝国的各地。在今天的主要的酿酒地区的
- 武光玡武光玡(越南语:Vũ Quang Nhạ/.mw-parser-output .han-nom{font-family:"Nom Na Tong","Han-Nom Gothic","Han-Nom Ming","HAN NOM A","HAN NOM B","Ming-Lt-HKSCS-UNI-H","Mi
- 索菲亚·迈尔斯索菲亚·迈尔斯(英语: Sophia Jane Myles ,1980年3月18日-)出生于英国英格兰的伦敦,为英国知名女演员。迈尔斯出生于伦敦市,她的母亲简(尼勒·阿伦)在一家教育出版社工作,他的父亲彼得
- 光之路者光之路者(波斯语:راهیان نور)是伊朗针对一个大型旅行团的称呼。该旅行团推出的是朝圣旅游,他们参观伊朗与伊拉克曾发生战争的战区。该朝圣旅游者参观的战区包括西部
- 马恩山坐标:47°24′44″N 11°30′28″E / 47.41229°N 11.50786°E / 47.41229; 11.50786马恩山(德语:Mahnkopf),是奥地利的山峰,位于该国西部,由蒂罗尔州负责管辖,属于卡尔文德尔山脉的
- 费舍尔·安妮费舍尔·安妮(匈牙利语:Fischer Annie,1914年7月5日-1995年4月10日),匈牙利钢琴家。费舍尔出生于布达佩斯,就读于同城的弗朗茨·李斯特音乐学院,师从多赫南伊·埃尔诺。1933年她在出
- 隋新梅隋新梅(1965年1月29日-),山东招远人,中国女子铅球运动员。她曾在亚特兰大的1996年夏季奥林匹克运动会上赢得银牌。她也赢得1991年世界室内田径锦标赛、1990年亚洲运动会、1994年亚洲运动会和1995年亚洲锦标赛。她在1997年东亚运动会上赢得银牌。她的个人最好成绩是21.66米,于1990年在北京取得,使她成为铅球总成绩第10名。