
当我们把一个SQL语句提交完后,Oracle绑定变量会检查一下在共享缓冲池(shared pool)中有无与之完全相同的相关语句,如果有的话只须执行软分析即可,否则就得进行硬分析。以下就是文章的具体内容介绍。而唯一使得oracle 能够重复利用执行计划的方法就是采用绑定变量。绑定变量的实质就是用于替代sql语句中的常量的替代变量。绑定变量能够使得每次提交的sql语句都完全一样。
在字段(包括字段集)建有索引,且字段(集)的集的势非常大(也就是有个值在字段中出现的比例特别的大)的情况下,使用绑定变量可能会导致查询计划错误,因而会使查询效率非常低。这种情况最好不要使用绑定变量。
但是并不是任何情况下都需要使用绑定变量,下面是两种例外情况:
1.对于隔相当一段时间才执行一次的SQL语句,这是利用绑定变量的好处会被不能有效利用优化器而抵消
2.数据仓库的情况下。
绑定变量不能当作嵌入的字符串来使用,只能当作语句中的变量来用。不能用Oracle绑定变量来代替表名、过程名、字段名等.
从效率来看,由于Oracle10G放弃了RBO,全面引入CBO,因此,在10G中使用绑定变量效率的提升比9i中更为明显。
举例:
普通sql语句:
SELECT fname, lname, pcode FROM cust WHERE id = 674;
SELECT fname, lname, pcode FROM cust WHERE id = 234;
SELECT fname, lname, pcode FROM cust WHERE id = 332;

含绑定变量的sql 语句:
SELECT fname, lname, pcode FROM cust WHERE id = :cust_no;
Sql*plus 中使用Oracle绑定变量:

SQL> set timing on
SQL> variable x number;
SQL> exec 😡 :=8
PL/SQL 过程已成功完成。
已用时间: 00: 00: 00.03
SQL> select * from A;
已用时间: 00: 00: 00.06
SQL> insert into A values(:x);
已创建 1 行。
已用时间: 00: 00: 00.01
SQL> select * from A;
已用时间: 00: 00: 00.01
PL/SQL很多时候都会自动绑定变量而无需编程人员操心,即很多你写得sql语句都会自动利用Oracle绑定变量,如下例所示:
SQL> Set timing on
SQL> declare
4 for I IN 1..1000 LOOP
5 INSERT INTO A VALUES(I);
PL/SQL 过程已成功完成。
已用时间: 00: 00: 00.12
这段代码是不需要使用Oracle绑定变量的方法来提高效率的,Oracle会自动将其中的变量绑定。
SQL> create table D ( id varchar(10));
表已创建。
已用时间: 00: 00: 00.50
SQL> declare
3 sqlstr varchar(2000);
5 for i in 1..1000 loop
6 sqlstr :=’ insert into d values(‘||to_char(i)||’)’;
7 execute immediate sqlstr;
PL/SQL 过程已成功完成。
已用时间: 00: 00: 00.68
这段代码同样是执行了1000条insert语句,但是每一条语句都是不同的,因此Oracle会把每条语句硬解析一次,其效率就比前面那段就低得多了。如果要提高效率,不妨使用绑定变量将循环中的语句改为
SQL> declarev
3 sqlstr varchar(2000);
5 for i in 1..1000 loop
6 sqlstr :=’ insert into d values(:i)’;
7 execute immediate sqlstr using i;
PL/SQL 过程已成功完成。
已用时间: 00: 00: 00.18
这样执行的效率就高得多了。
在PL/SQL中,引用变量即是引用绑定变量。但是在pl/sql中动态sql并不是这样。在vb,java以及其他应用程序中都得显式地利用Oracle绑定变量。对于绑定变量的支持不仅仅限于Oracle,其他RDBMS向SQL SERVER也支持这一特性。
【编辑推荐】
Oracle中表建立联合索引后,其中的一个字段是否可以用来进行索引。
1、表的主键、外键必须有索引;2、数据量超过300的表应该有索引;3、经常与其他表进行连接的表,在连接字段上应该建立索引;4、经常出现在Where子句中的字段,特别是大表的字段,应该建立索引;5、索引应该建在选择性高的字段上;6、索引应该建在小字段上,对于大的文本字段甚至超长字段,不要建索引;7、复合索引的建立需要进行仔细分析;尽量考虑用单字段索引代替:A、正确选择复合索引中的主列字段,一般是选择性较好的字段;B、复合索引的几个字段是否经常同时以AND方式出现在Where子句中?单字段查询是否极少甚至没有?如果是,则可以建立复合索引;否则考虑单字段索引;C、如果复合索引中包含的字段经常单独出现在Where子句中,则分解为多个单字段索引;D、如果复合索引所包含的字段超过3个,那么仔细考虑其必要性,考虑减少复合的字段;E、如果既有单字段索引,又有这几个字段上的复合索引,一般可以删除复合索引;8、频繁进行数据操作的表,不要建立太多的索引;9、删除无用的索引,避免对执行计划造成负面影响;以上是一些普遍的建立索引时的判断依据。 一言以蔽之,索引的建立必须慎重,对每个索引的必要性都应该经过仔细分析,要有建立的依据。 因为太多的索引与不充分、不正确的索引对性能都毫无益处:在表上建立的每个索引都会增加存储开销,索引对于插入、删除、更新操作也会增加处理上的开销。 另外,过多的复合索引,在有单字段索引的情况下,一般都是没有存在价值的;相反,还会降低数据增加删除时的性能,特别是对频繁更新的表来说,负面影响更大。 看看这个你就明白了
PL/SQL里 execute immediate的用法 谁给解释下
在ORACLE的PL/SQL里:EXECUTE IMMEDIATE 代替了以前Oracle8i中DBMS_SQL package包.它解析并马上执行动态的SQL语句或非运行时创建的PL/SQL块.动态创建和执行SQL语句性能超前,EXECUTE IMMEDIATE的目标在于减小企业费用并获得较高的性能,较之以前它相当容易编码.尽管DBMS_SQL仍然可用,但是推荐使用EXECUTE IMMEDIATE,因为它获的收益在包之上。 -- 使用技巧1. EXECUTE IMMEDIATE将不会提交一个DML事务执行,应该显式提交如果通过EXECUTE IMMEDIATE处理DML命令,那么在完成以前需要显式提交或者作为EXECUTE IMMEDIATE自己的一部分.如果通过EXECUTE IMMEDIATE处理DDL命令,它提交所有以前改变的数据2. 不支持返回多行的查询,这种交互将用临时表来存储记录(参照例子如下)或者用REF cursors.3. 当执行SQL语句时,不要用分号,当执行PL/SQL块时,在其尾部用分号.4. 在Oracle手册中,未详细覆盖这些功能。 下面的例子展示了所有用到Execute immediate的可能方面.希望能给你带来方便.5. 对于Forms开发者,当在PL/SQL 8.0.6.3.版本中,Forms 6i不能使用此功能 IMMEDIATE -- 用法例子1. 在PL/SQL运行DDL语句beginexecute immediate set role all;end;2. 给动态语句传值(USING 子句)declarel_depnam varchar2(20) := testing;l_loc varchar2(10) := Dubai;beginexecute immediate insert into dept values (:1, :2, :3)using 50, l_depnam, l_loc;commit;end;3. 从动态语句检索值(INTO子句)declarel_cnt varchar2(20);beginexecute immediate select count(1) from empinto l_cnt;dbms__line(l_cnt);end;4. 动态调用例程.例程中用到的绑定变量参数必须指定参数类型.黓认为IN类型,其它类型必须显式指定declarel_routinvarchar2(100) := _rowcnt;l_tblnamvarchar2(20) := emp;l_cnt number;l_statusvarchar2(200);beginexecute immediate begin || l_routin || (:2, :3, :4); end;using in l_tblnam, out l_cnt, in out l_status; if l_status != OK then dbms__line(error);end if;end;5. 将返回值传递到PL/SQL记录类型;同样也可用%rowtype变量declaretype empdtlrec is record (empno number(4), ename varchar2(20), deptno number(2));empdtl empdtlrec;beginexecute immediate select empno, ename, deptno || from emp where empno = 7934into empdtl;end;6. 传递并检索值子句用在USING子句前declarel_dept pls_integer := 20;l_namvarchar2(20);l_locvarchar2(20);beginexecute immediate select dname, loc from dept where deptno = :1into l_nam, l_locusing l_dept ;end;7. 多行查询选项.对此选项用insert语句填充临时表,用临时表进行进一步的处理,也可以用REF cursors纠正此缺憾_salpls_integer := 2000;beginexecute immediate insert into temp(empno, ename) || select empno, ename from emp || where sal > :1using l_sal;commit;end;对于处理动态语句,EXECUTE IMMEDIATE 比以前可能用到的更容易并且更高效.当意图执行动态语句时,适当地处理异常更加重要.应该关注于捕获所有可能的异常.
怎样在hibernate的HQL语句中使用mysql 的自定义函数
这时就需要重新定义hibernate的方言(dialect),将你需要用到的进行注册; 这里提到一个方言(dialect)的概念:Hibernate底层依然使用SQL语句来执行数据库操作,虽然所有关系型数据库都支持使用标准SQL语句,但所有数据库都对标准SQL进行了一些扩展,所以在语法细节上存在一些差异,因此Hibernate需要根据数据库来识别这些差异。 而方言(dialect)就是用来区分hibernate对应哪种数据库 通常在使用Spring整合hibernate的时候,会有这样一段配置=5InnoDBDialect _sql=true =none _sql=true _class= _mode=auto =1 =false _query_cache=false _second_level_cache=false _size=25 _size=50 其中红色的配置语句就是指明使用mysql方言,对于其他不同的数据库都有对应不同的方言,详情点击这里。 首先创建一个mysql函数,我们命名为my_change_local(number bigint) --这里是使用navicat工具创建的函数 BEGIN DECLARE stemp varchar(255); set stemp=; select into stemp from sms where =number; RETURN stemp; END 1.重写方言类,注册函数 public class MyLocalDialect extends MySQL5InnoDBDialect { public MyLocalDialect() { super(); registerFunction(my_change_local,new SQLFunctionTemplate(,my_change_local(?1))); } } 2.修改配置文件,将数据库方言指向自己定义的方言类 将 =5InnoDBDialect 改为 = 3.之后就可以在hql中使用自己定义的函数了
发表评论