目录
oracle查询要求." >借助内存表处理复杂的oracle查询要求.
, 创建测试用表." >, 创建测试用表.
, 方法1 , 使用动态语句拼凑实现." >, 方法1 , 使用动态语句拼凑实现.
, 方法2 使用like查询" >, 方法2 使用like查询
, 方法3 使用instr函数处理" >, 方法3 使用instr函数处理
, 方法4 使用内存表处理" >, 方法4 使用内存表处理
首页 数据库 mysql教程 借助内存表处理复杂的oracle查询要求

借助内存表处理复杂的oracle查询要求

Jun 07, 2016 pm 03:50 PM
oracle 内存 处理 复杂 查询 要求

借助内存表处理复杂的 oracle 查询要求 . 在日常业务处理过程中 , 我们经常会碰到一些非常规的查询需求 , 这些需求我们或者可以借助动态语句 , 或者其他现有的 oracle 函数完成查询结果 , 但效率往往差强人意 . 假设我们有一个客户订单业务表 { 订单号 , 订

借助内存表处理复杂的oracle查询要求.

在日常业务处理过程中,我们经常会碰到一些非常规的查询需求, 这些需求我们或者可以借助动态语句,或者其他现有的oracle函数完成查询结果, 但效率往往差强人意.

假设我们有一个客户订单业务表{订单号, 订单客户, 订单日期, 数量, 金额}存储了订单的往来明细数据,订单表中保存最近3个月的往来明细共1000w条记录, 其中客户总量约500000. 并假定在订单表上有针对日期和客户的单独索引.

现在要求提供对任意集合的多个客户的某段时间的订单明细数据.

Select 订单号, 订单客户, 订单日期, 数量, 金额

From 订单业务表

Where 订单日期 between 开始日期 and 结束日期

  And 订单客户 in (客户1, 客户2, 客户3…)

面对这种需求, 我们可以要求前台程序传回三个参数, 开始日期, 结束日期, 客户列表(类似于客户1, 客户2, 客户3, 客户4…)

, 创建测试用表.

Create table t_order_cust(

O_id varchar2(20),

O_customer varchar2(20),

O_date date,

O_qty numeric(18,2),

O_amount numeric(18,2)

);

 

Create index ind_t_order_cust_01 on t_order_cust(o_customer);

Create index ind_t_order_cust_02 on t_order_cust(o_date);

 

, 方法1 , 使用动态语句拼凑实现.

针对上述查询, 我们可以拼凑动态语句实现, 如下代码所示.

Declare

  V_beg_date date := trunc(sysdate,’month’);

  V_end_date date := trunk(sysdate);

  V_cust_str varchar2(1000) := ‘’’客户1’’, ’’客户2’’, ’’客户3’’…’;

  V_sql_str varchar2(2000) ;

Begin

  V_sql_str := ‘select * from t_order_cust

Where o_date between ’ || v_beg_date || ‘ and ’ || v_end_date ||’

And o_customer in (’||v_cust_str||’)’;

  Execute immediate v_sql_str;

End;

根据表明细数据的特点我们知道, 客户索引的选择性为 1000w/50w= 20, 而日期索引的选择性为 1000w/(3*30) = 10w, 明显使用日期索引效率极差, 我们只能选择使用客户上的索引,使用这种处理方法的优势是可以用到客户上的索引, in使用索引的效率相对较差, 并且这种处理方式下, oracle每次执行查询都需要重新建立查询执行树, 也是需要一定的额外开销.

, 方法2 使用like查询

除了上面的拼凑动态执行语句的方法之外, 我们可以设想的到的第二种方法就是借助于oracle提供的like功能. 如下代码所示.这种处理方式下对客户列表字符串的要求跟方法一少有区别.

Declare

  V_beg_date date := trunc(sysdate,’month’);

  V_end_date date := trunk(sysdate);

  V_cust_str varchar2(1000) := ‘客户1, 客户2, 客户3…’;

Begin

  select * from t_order_cust

Where o_date between v_beg_date  and  v_end_date

And v_cust_str like ‘%’||o_customer||’%’;

End;

这种处理方式的优点在于代码书写简单, 但由于对客户索引所在字段o_customer做了拼接处理||, 所以将导致客户索引ind_t_order_cust_01无效, 而只能使用效率较差的日期索引. 在数据量较小, 对效率无法造成影响时这种方法可以接受, 但数据量较大时, 这种方法的缺点将是致命的.

, 方法3 使用instr函数处理

Declare

  V_beg_date date := trunc(sysdate,’month’);

  V_end_date date := trunk(sysdate);

  V_cust_str varchar2(1000) := ‘客户1, 客户2, 客户3…’;

Begin

  select * from t_order_cust

Where o_date between v_beg_date  and  v_end_date

And instr(v_cust_str like ,o_custome) >0

End;

这种处理方式的优缺点跟使用like相似, 同样由于对o_customer使用了函数, 导致该索引不可用, 函数索引同样也不适用于这种情况.

, 方法4 使用内存表处理

我们知道, oracle, sqlserver等关系数据库最善于处理的数据类型是集合, 而不是单独的记录. 同样的100条记录, 如果逐条循环处理和批量处理其效率的差别将是几何单位的.

所以, 为了提高查询效率, 我们这里考虑将给定的客户字符串转变为一个集合或者临时表来处理. Oracle使用全局临时表和复杂数据类型集合来支持这一点.

这里我们介绍一下使用复杂数据类型集合来处理的方式.

首先我们定义一个复杂类型.

create or replace type ctl.type_jax_varc2tab is table of varchar2(2000);

然后定义一个函数实现将给定的字符串转换为嵌套内存表.

CREATEORREPLACEFUNCTION f_jax_str2tab(p_str INVARCHAR2,
p_sep
varchar2default','
)
RETURN ctl.type_jax_varc2tab IS
 
/******************************************************************
  Ver1.0 Created by jaxzhang on 2009-06-08
 
把字符串(1*2*3*4*5)转换为内存表形式
  create or replace type type_jax_varc2tab is table of varchar2(2000);
 
测试用例:SELECT * FROM TABLE(f_jax_str2tab('1*2*3*4*5','*'));
  ******************************************************************/

  v_str
varchar2(2000);
  v_cnt
NUMBER ;
  v_numtab type_jax_varc2tab := type_jax_varc2tab();
--返回内存表
BEGIN
 
select decode(substr(p_str,-1),p_sep,p_str,p_str || p_sep) into v_str from dual;
 
select  length(v_str) - length(REPLACE(v_str, p_sep)) into v_cnt from dual;

 
FOR i IN1 .. v_cnt LOOP
    v_numtab.
EXTEND;
    v_numtab(i) := substr(v_str,
1, instr(v_str, p_sep) - 1);
    v_str := substr(v_str, instr(v_str,p_sep) +
1);
 
ENDLOOP;

 
RETURN v_numtab;
EXCEPTION
 
WHENOTHERSTHEN
    v_numtab.
DELETE;
END;
上述函数的功能就是要将类似于客户1,客户2,客户3’的字符串转换为如下形式.

SELECT * FROMTABLE(f_jax_str2tab('客户1,客户2,客户3',','));

COLUMN_VALUE

客户1

客户2

客户3

得到上述的内存表之后, 我们就可以使用类似于一个表或者视图的方式来与正式表t_order_cust关联得到我们需要的查询结果.

Select /*+ ordered use_nl(a b)*/

  From TABLE(f_jax_str2tab('客户1,客户2,客户3',',')) a,

        T_order_cust b

   Where b.o_customer = a.column_value

 

 

 

 

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

oracle如何查表空间大小 oracle如何查表空间大小 Apr 11, 2025 pm 08:15 PM

要查询 Oracle 表空间大小,请遵循以下步骤:确定表空间名称,方法是运行查询:SELECT tablespace_name FROM dba_tablespaces;查询表空间大小,方法是运行查询:SELECT sum(bytes) AS total_size, sum(bytes_free) AS available_space, sum(bytes) - sum(bytes_free) AS used_space FROM dba_data_files WHERE tablespace_

oracle数据库如何导入 oracle数据库如何导入 Apr 11, 2025 pm 08:06 PM

数据导入方法:1. 使用 SQLLoader 实用程序:准备数据文件、创建控制文件、运行 SQLLoader;2. 使用 IMP/EXP 工具:导出数据、导入数据。提示:1. 大数据集推荐 SQL*Loader;2. 目标表应存在,列定义匹配;3. 导入后需验证数据完整性。

Oracle安装失败如何卸载 Oracle安装失败如何卸载 Apr 11, 2025 pm 08:24 PM

Oracle 安装失败的卸载方法:关闭 Oracle 服务,删除 Oracle 程序文件和注册表项,卸载 Oracle 环境变量,重新启动计算机。若卸载失败,可使用 Oracle 通用卸载工具手动卸载。

如何在oracle中创建表 如何在oracle中创建表 Apr 11, 2025 pm 08:00 PM

创建 Oracle 表涉及以下步骤:使用 CREATE TABLE 语法指定表名、列名、数据类型、约束和默认值。表名应简洁、描述性,且不超过 30 个字符。列名应描述性,数据类型指定列中存储的数据类型。NOT NULL 约束确保列中不允许使用空值,DEFAULT 子句可指定列的默认值。PRIMARY KEY 约束标识表的唯一记录。FOREIGN KEY 约束指定表中的列引用另一个表中的主键。请参见示例表 students 的创建,其中包含主键、唯一约束和默认值。

oracle如何去重查询 oracle如何去重查询 Apr 11, 2025 pm 07:33 PM

Oracle 提供多种去重查询方法:DISTINCT 关键字返回每列的唯一值。GROUP BY 子句对结果分组并返回每个分组的非重复值。UNIQUE 关键字用于创建仅包含唯一行的索引,查询该索引将自动去重。ROW_NUMBER() 函数分配唯一数字并过滤出仅包含第 1 行的结果。MIN() 或 MAX() 函数可返回数字列的非重复值。INTERSECT 运算符返回两个结果集的公共值(无重复项)。

oracle如何查看实例名 oracle如何查看实例名 Apr 11, 2025 pm 08:18 PM

在 Oracle 中查看实例名的方法有三种:命令行中使用 "sqlplus" 和 "select instance_name from v$instance;" 命令。在 SQL*Plus 中使用 "show instance_name;" 命令。通过操作系统的任务管理器、Oracle Enterprise Manager 或检查环境变量 (Linux 上的 ORACLE_SID)。

oracle视图如何加密 oracle视图如何加密 Apr 11, 2025 pm 08:30 PM

Oracle 视图加密允许您加密视图中的数据,从而增强敏感信息安全性。步骤包括:1) 创建主加密密钥 (MEk);2) 创建加密视图,指定要加密的视图和 MEk;3) 授权用户访问加密视图。加密视图工作原理:当用户查询加密视图时,Oracle 使用 MEk 解密数据,确保只有授权用户可以访问可读数据。

oracle中乱码怎么解决 oracle中乱码怎么解决 Apr 11, 2025 pm 10:09 PM

Oracle 乱码问题可以通过以下步骤解决:检查数据库字符集以确保与数据相匹配。设置客户端字符集以与数据库相匹配。转换数据或修改列字符集以匹配数据库字符集。使用 Unicode 字符集,并避免多字节字符集。检查数据库和客户端的语言设置是否正确。

See all articles