下面案例中采用了sql硬编码方式,当方法中的参数变量改变时,导致不能使用数据库的共享缓存,频繁对sql语句进行语法分析,浪费数据库资源,改为绑定变量方式可以解决此类问题,不用重复对sql语句进行语法分析。
典型案例:
将以上java代码更改为如下形式:
mkrule.sample文件内容:
select nvl(max(PQ_GAIN_LOSS), 0) from M_SUPL_DMD_AF t where t.af_info_id in (select AF_INFO_ID from M_AF x where x.org_no = ? and extract(year from bgn_date) = ?and extract(month from bgn_date) = ?and x.anal_type_code = '3' and x.anal_fore_flag = '2' and x.freq_code = '4')
为了说明上面的案例,先来了解一下共享sql语句概念。为了不重复解析相同的SQL语句(因为解析操作比较费资源,会导致性能下降),在第一次解析之后,ORACLE将SQL语句及解析后得到的执行计划存放在内存中。这块位于系统全局区域SGA(system global area)的共享池(shared buffer pool)中的内存可以被所有的数据库用户共享。因此,当你执行一个SQL语句(有时被称为一个游标)时,如果该语句和之前的执行过的某一语句完全相同,并且之前执行的该语句与其执行计划仍然在内存中存在,则ORACLE就不需要再进行分析,直接得到该语句的执行路径。ORACLE的这个功能大大地提高了SQL的执行性能并大大节省了内存的使用。使用这个功能的关键是将执行过的语句尽可能放到内存中,所以这要求有大的共享池(通过设置shared buffer pool参数值)和尽可能的使用绑定变量的方法执行SQL语句。
当你向ORACLE提交一个SQL语句,ORACLE会首先在共享内存中查找是否有相同的语句。这里需要注明的是,ORACLE对两者采取的是一种严格匹配,要达成共享,SQL语句必须完全相同(包括空格,换行等)。
下面是判断SQL语句是否与共享内存中某一SQL相同的步骤:
1) 对所发出语句的文本串进行hashed。如果hash值与已在共享池中SQL语句的hash值相同,则进行第2步:
2) 将所发出语句的文本串(包括大小写、空白和注释)与在第1步中识别的所有
已存在的SQL语句相比较。
例如:
SELECT * FROM emp WHERE empno = 1000;
和下列每一个都不同
SELECT * from emp WHERE empno = 1000;
SELECT * FROM EMP WHERE empno = 1000;
SELECT * FROM emp WHERE empno = 2000;
在上面的语句中列值都是直接SQL语句中的,我们将这类sql成为硬编码SQL。
使用绑定变量的SQL语句中必须使用相同的名字的绑定变量(bind variables) ,
例如:
a、该2个sql语句被认为相同
select pin , name from people where pin = :blk1.pin;
select pin , name from people where pin = :blk1.pin;
b、该2个sql语句被认为不相同
select pin , name from people where pin = :blk1.ot_ind;
select pin , name from people where pin = :blk1.ov_ind;
我们将上面的这类语句称为绑定变量SQL。
3) 将所发出语句中涉及的对象与第2步中识别的已存在语句所涉及对象相比较。
例如:
如用户user1与用户user2下都有EMP表,则
用户user1发出的语句:SELECT * FROM EMP; 与
用户user2发出的语句:SELECT * FROM EMP; 被认为是不相同的语句,
因为两个语句中引用的EMP不是指同一个表。
4) 在SQL语句中使用的捆绑变量的捆绑类型必须一致。
如果语句与当前在共享池中的另一个语句是等同的话,Oracle并不对它进行语法分析。而直接执行该语句,提高了执行效率,因为语法分析比较耗费资源。
Labels parameters
典型案例:
public long querySample(String orgNo,String strYear,String strMonth)
{
StringBuffer queryString = new StringBuffer();
queryString.append("select nvl(max(PQ_GAIN_LOSS), 0) from M_SUPL_DMD_AF t where t.af_info_id in (select AF_INFO_ID from M_AF x where x.org_no = '");
queryString.append(orgNo +"'");
queryString.append("and extract(year from bgn_date) = '");
queryString.append(strYear+"' ");
queryString.append("and extract(month from bgn_date) = '");
queryString.append(strMonth+"' ");
queryString.append("and x.anal_type_code = ? and x.anal_fore_flag = '2' and x.freq_code = '4')");
query.setStatement("{"+queryString.toString()+"}");
query.addParameter("3");
将以上java代码更改为如下形式:
public long querySample(String orgNo,String strYear,String strMonth)
{
protected static final String ST_MKRULE_SAMPLE = "mkrule.sample";
query.setStatement(ST_MKRULE_SAMPLE);
query.addParameter(orgNo);
query.addParameter(strYear);
query.addParameter(strMonth);
mkrule.sample文件内容:
select nvl(max(PQ_GAIN_LOSS), 0) from M_SUPL_DMD_AF t where t.af_info_id in (select AF_INFO_ID from M_AF x where x.org_no = ? and extract(year from bgn_date) = ?and extract(month from bgn_date) = ?and x.anal_type_code = '3' and x.anal_fore_flag = '2' and x.freq_code = '4')
为了说明上面的案例,先来了解一下共享sql语句概念。为了不重复解析相同的SQL语句(因为解析操作比较费资源,会导致性能下降),在第一次解析之后,ORACLE将SQL语句及解析后得到的执行计划存放在内存中。这块位于系统全局区域SGA(system global area)的共享池(shared buffer pool)中的内存可以被所有的数据库用户共享。因此,当你执行一个SQL语句(有时被称为一个游标)时,如果该语句和之前的执行过的某一语句完全相同,并且之前执行的该语句与其执行计划仍然在内存中存在,则ORACLE就不需要再进行分析,直接得到该语句的执行路径。ORACLE的这个功能大大地提高了SQL的执行性能并大大节省了内存的使用。使用这个功能的关键是将执行过的语句尽可能放到内存中,所以这要求有大的共享池(通过设置shared buffer pool参数值)和尽可能的使用绑定变量的方法执行SQL语句。
当你向ORACLE提交一个SQL语句,ORACLE会首先在共享内存中查找是否有相同的语句。这里需要注明的是,ORACLE对两者采取的是一种严格匹配,要达成共享,SQL语句必须完全相同(包括空格,换行等)。
下面是判断SQL语句是否与共享内存中某一SQL相同的步骤:
1) 对所发出语句的文本串进行hashed。如果hash值与已在共享池中SQL语句的hash值相同,则进行第2步:
2) 将所发出语句的文本串(包括大小写、空白和注释)与在第1步中识别的所有
已存在的SQL语句相比较。
例如:
SELECT * FROM emp WHERE empno = 1000;
和下列每一个都不同
SELECT * from emp WHERE empno = 1000;
SELECT * FROM EMP WHERE empno = 1000;
SELECT * FROM emp WHERE empno = 2000;
在上面的语句中列值都是直接SQL语句中的,我们将这类sql成为硬编码SQL。
使用绑定变量的SQL语句中必须使用相同的名字的绑定变量(bind variables) ,
例如:
a、该2个sql语句被认为相同
select pin , name from people where pin = :blk1.pin;
select pin , name from people where pin = :blk1.pin;
b、该2个sql语句被认为不相同
select pin , name from people where pin = :blk1.ot_ind;
select pin , name from people where pin = :blk1.ov_ind;
我们将上面的这类语句称为绑定变量SQL。
3) 将所发出语句中涉及的对象与第2步中识别的已存在语句所涉及对象相比较。
例如:
如用户user1与用户user2下都有EMP表,则
用户user1发出的语句:SELECT * FROM EMP; 与
用户user2发出的语句:SELECT * FROM EMP; 被认为是不相同的语句,
因为两个语句中引用的EMP不是指同一个表。
4) 在SQL语句中使用的捆绑变量的捆绑类型必须一致。
如果语句与当前在共享池中的另一个语句是等同的话,Oracle并不对它进行语法分析。而直接执行该语句,提高了执行效率,因为语法分析比较耗费资源。
Labels parameters