原文: [url]http://xujt82.spaces.live.com/blog/cns!1EB2C64171792704178.entry [/url]
1、为什么要使用绑定变量
(1)SQL语句硬分析(Hard Parse)太多,严重消耗CPU资源,延长了SQL语句总的执行时间
SQL语句的执行过程分几个步骤:语法检查、分析、执行、返回结果。其中分析分为硬分析(Hard Parse)和软分析(Soft Parse)。一条SQL语句通过语法检查后,Oracle 会先去shared pool 中找是否有相同的sql,如果找着了,就叫软分析,然后执行SQL语句。硬分析主要是检查该sql所涉及到的所有对象是否有效以及权限等关系,然后根据RBO或CBO模式生成执行计划,然后才执行SQL语句。
可以看出,硬分析比软分析多了很多动作,而这里面的关键是“在shared pool 中是否有相同的sql”,而这就取决于是否使用绑定变量。
另:oracle9i引入了soft soft parse,先到pga中的session cursor cache list列表中去查找(session cursor cache list的长度是由session_cache_cursor参数决定的),如果没有找到这条sql,这时候才去检查shard_pool. 对于Oltp系统,很多时候硬分析的代价比执行还要高,这个我们可以通过10046事件跟踪得知。
(2)共享池中SQL语句数量太多,重用性极低,加速了SQL语句的老化,导致共享池碎片过多。
共享池中不同的SQL语句数量巨大,根据LRU原则,一些语句逐渐老化,最终被清理出共享池;这样就导致shared_pool_size 里面命中率下降,共享池碎片增多,可用内存空间不足。而为了维护共享池内部结构,需要使用latch,一种内部生命周期很短的lock,这将使用大量的cpu 资源,使得性能急剧下降。
不使用绑定变量违背了oracle 的shared pool 的设计的原则,违背了这个设计用来共享的思想。
2、怎么查看没有使用绑定变量
select * from v$sql or v$sqlarea 查看是否有很多类似的语句,除了变量不一样,其他的都一样
3、如何使用绑定变量?
编写java 程序时,我们习惯都是定义JAVA 的程序变量,放入SQL 语句中,如
String v_id = 'xxxxx';
String v_sql = 'select name from table_a where id = ' + v_id ;
以上代码,看起来是使用了变量v_id ,但这却是java 的程序变量,而不是oracle 的绑定变量,语句传递到数据库后,此java 的程序变量
已经被替换成具体的常量值,变成:
select * from table_a where name = 'xxxxx' ;
假定这个语句第一次执行,会进行硬分析。后来,同一段java 代码中v_id 值发现变化(v_id = 'yyyyyy'),数据库又接收到这样的语句:
select * from table_a where name = 'yyyyyy' ;
ORACLE 并不认为以上两条语句是相同的语句,因此对第二条语句会又做一次硬分析。这两条语句的执行计划可是一样的!
其实,只需将以上java 代码改成以下这样,就使用了oracle 的绑定变量:
String v_id = 'xxxxx';
String v_sql = 'select name from table_a where id = ? '; //嵌入绑定变量
stmt = con.prepareStatement( v_sql );
stmt.setString(1, v_id ); //为绑定变量赋值
stmt.executeQuery();
在Java中,结合使用setXXX 系列方法,可以为不同数据类型的绑定变量进行赋值,从而大大优化了SQL 语句的性能。
4、java中应用绑定变量的例子
PreparedStatement stmt = conn.prepareStatement('select a from b where c = ? ');
stmt.setLong(1,123);
stmt.executeQuery()
……
结论:
绑定变量主要适用在Oltp,运行时间很短的系统。如客服系统,时时地进行insert方面的系统。 数据仓库系统不适用,和数据库仓库系统的一条sql运行时间相比,硬分析的代价显然是微不足道的,通过硬分析去选择正确的执行计划才是关键。
简单一句话,在Oltp系统中应用绑定变量,性能会有质的提高。 而且安全性会好很多,主要是因为如果用 + String的形式把变量插到sql中的话,如果参数中有一些非法字符,比如单引号之类.就会引起sql异常,而且会存在不安全因素,而预编译的方式就会避免这种情况的出现.
分享到:
相关推荐
在Java中,我们可以通过使用PreparedStatement对象来实现Oracle的绑定变量。例如: ```java String v_id = 'xxxxx'; String v_sql = "select name from table_a where id = ?"; // 使用问号作为绑定变量 ...
本文将深入探讨绑定变量的原理、优势以及如何在实际应用中使用。 首先,为什么我们要使用绑定变量?这主要是为了提高应用程序的可伸缩性和性能。Oracle的共享池是一个内存结构,用于存储编译后的SQL语句和PL/SQL块...
如果不能使用强制绑定变量的oracle 参数 cursor_sharing=force ,也不能用触发器修改某个用户的 cursor_sharing=force 会话参数,那么这个转换程序可以帮到你(不能解决使用JAVA等编程语言,因为java使用?...
此外,Oracle提供了如**绑定变量窥探**(bind variable peeking)和**cursor_sharing**参数来调整绑定变量的行为,以适应不同的应用场景。PL/SQL作为一种强类型的语言,使用绑定变量更为自然,而其他如Java、VB等...
Oracle数据库作为当前企业级应用中广泛使用的数据库系统之一,其提供的动态SQL功能允许开发者在程序运行时构造和执行SQL语句,这种灵活性为程序开发提供了极大的便利。本文将探讨Oracle中的动态SQL实现机制,重点...
- **绑定变量**:避免SQL注入,使用绑定变量(`:bind_variable`)执行动态SQL。 遵守这些规范,将有助于编写出高质量、易于维护的PL/SQL代码,提高代码的可读性和团队间的沟通效率。在实际开发中,还应考虑性能优化...
为了进一步提高效率,我们可以使用绑定变量(Bind variable)。绑定变量可以减少重复解析,避免因字符串拼接导致的闩锁竞争,并且降低内存和CPU资源的消耗。此外,通过调整`session_cached_cursors`参数,可以控制...
4. **绑定和获取结果**:使用 `ocilib_statement_bind_variable` 绑定输入/输出变量,`ocilib_resultset_fetch_next` 获取结果集中的行。 5. **游标操作**:如果 SQL 语句包含 SELECT,可以使用游标(cursor)遍历...
避免在SQL语句中直接使用动态值,而是用占位符(如::bind_variable)。 5. **聚合函数与GROUP BY**:合理使用GROUP BY和HAVING,避免不必要的数据聚合。如果可能,尽量在子查询中进行聚合,减少主查询的数据量。 ...
绑定变量是一种在SQL语句中使用的占位符,它们可以在执行时被动态替换为实际值。使用绑定变量有助于减少硬解析的数量,并提高查询性能。例如,在编写PL/SQL代码时,可以使用`:bind_variable`来代替硬编码的值。 ###...
- 绑定变量(Bind Variable)使用 `VAR` 或 `VARIABLE` 声明,并以冒号 `:` 作为前缀。 - 示例: ```sql VAR a NUMBER BEGIN :a := 22; END; PRINT a; -- 输出结果 ``` - **命名块**: - 命名块是指具有...
使用`DBMS_SQL.OPEN_CURSOR`打开一个游标,然后通过`DBMS_SQL.PARSE`解析SQL语句,并用`DBMS_SQL.BIND_VARIABLE`绑定变量,最后调用`DBMS_SQL.EXECUTE`执行删除操作。 #### 示例2:执行DDL语句 ```sql CREATE OR ...
3. **BIND_VARIABLE**:绑定变量到SQL语句,使得在执行时可以传递不同的参数值。 4. **DEFINE_COLUMN**:定义游标中某一列的变量,用于存储查询结果。 5. **EXECUTE**:执行已解析并绑定的SQL语句,但不获取数据。 6...
根据给定文件的信息,我们可以深入探讨Oracle数据库中的PL/SQL(程序化SQL)语言的关键概念与用法。PL/SQL是Oracle数据库环境下的一个过程语言,它结合了SQL语句和传统的过程化语言特性,提供了强大的数据库编程能力...
在处理SQL语句时,如果使用了绑定变量(Bind Variable),Oracle将在Sharedpool中查找相同的语句模板,这大大提高了缓存命中率,减少了解析次数,有利于性能提升。相反,不使用绑定变量会导致SQL语句的动态部分无法...
3. **BIND_VARIABLE**: 将变量绑定到SQL语句中的占位符,这样可以在执行语句时传递实际值。 4. **DEFINE_COLUMN**: 对于SELECT语句,此函数定义了字段变量,这些变量将存储从游标中获取的结果集的列值。 5. **...
它接受一个游标ID、SQL语句和一个可选的绑定变量列表,返回一个游标状态,表示SQL语句是否成功解析。 2. BIND:此过程用于将PL/SQL变量绑定到SQL语句中的位置参数。位置参数由冒号加数字(例如::1, :2)表示。你...
- Bind_Variable过程用于为动态SQL语句绑定变量的值。 - Execute函数用于执行SQL语句。 - Define_Column过程用于定义Select语句查询的列名。 - Fetch的功能是从Cursor中获取查询结果的下一行,如果返回值为0则表示...
动态SQL主要分为两种形式:绑定变量(Bind Variables)和EXECUTE IMMEDIATE。绑定变量主要用于减少解析开销,提高性能,而EXECUTE IMMEDIATE则用于执行那些在编译时未知的SQL语句。 EXECUTE IMMEDIATE语法结构如下...
在Oracle中,DBMS_SQL包提供了处理动态SQL的功能,如解析、绑定变量和执行。以下是一个简单的示例: ```sql DECLARE cursor_var INTEGER; column_value VARCHAR2(50); BEGIN cursor_var := DBMS_SQL.OPEN_CURSOR;...