博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
动态魔术使用DBMS_SQL
阅读量:5925 次
发布时间:2019-06-19

本文共 7681 字,大约阅读时间需要 25 分钟。

动态SQL / PLSQL被认为是Oracle PL / SQL中最高级的主题之一。尽管它根本不是一个新主题,但它仍然是消化和掌握最具挑战性的概念之一。

动态SQL只是SQL代码的任何部分,在编译之前未明确写入。因此,例如,如果您不知道SQL查询中的“WHERE”谓词,则必须使用动态SQL。Oracle为我们提供了两种不同的方法来构建动态SQL / plsql代码,第一种是NativeDynamic SQL(NDS)。使用优雅,易于使用EXECUTE IMMEDIATE和OPEN.. FOR 语句,这种方法是在场景中超过90%绰绰有余。但剩下的10%呢?这是它变得更复杂的地方。

当面对动态SQL方法四,你需要你的代码执行包含不同数量的绑定变量的动态SQL语句时,NDS是不够的,因为使用NDS,你需要知道列数和数据类型你要检索它们(列类型)。Oracle提供了一种替代(实际上更老)的方法来处理代码需要超级动态的最极端情况,那就是DBMS_SQL包。

DBMS_SQL能够处理您可能遇到的所有未知数。使用DBMS_SQL时,使用DBMS_SQL游标时只使用特殊的游标类型(不是我们知道的常规游标,并且总是使用它),并且只获取唯一的游标ID。然后,我们获得完整的表描述,包括列的类型和属性。使用所有这些信息,我们可以将数据提取到正确类型的变量中。该过程通常如下进行:

定义一个游标; 只是一个ID

建立你的查询; 您将要执行的查询
使用您的查询解析光标并检查查询中的任何错误。
绑定变量(如果有)。
获取表格描述; 列数和类型信息。
定义列并将它们链接到正确的变量。
执行游标和查询。
从列中获取实际数据。
欣赏你的代码!
现在,让我们谈谈我们想象中的情景。出于某种原因,我们需要创建一个可用作“数据库”的表来存储来自任何其他表的任何类型的数据,并将其保存为加密形式。我们必须这样做,只有将数据删除到“保险库”的人才能解密并检索它。

为此,我们需要许多部分,例如一些自定义数据类型和对象,但更重要的是,我们必须构建一个能够从任何表中收集数据并加密然后存储在Vault中的函数。

请注意,该函数的编写方式使得可以在不知道任何表的情况下处理任何表,那么我们唯一的工具是什么呢?很简单,这是DBMS_SQL包!

使用Data Vault虚构函数,我将尝试演示和解释依赖于DBMS_SQL的基于动态SQL的函数的最重要,实用的组件。

最后一点,这是一个教程,旨在演示如何使用DBMS_SQL,因此它不是100%生产就绪,许多东西可以增强,甚至可以用其他方式完成。还有一些安全问题,但同样,它是关于DBMS_SQL的。

安装程序

1.我构建了一个原始类型的嵌套表来存储将存储在Vault中的加密数据。

CREAT OR REPLACE TYPE raw_tab AS NESTED TABLE OF RAW(2000)

2.为了存储列信息,我创建了一个模仿DBMS_SQL_DESC类型的对象,我还创建了一个该对象的嵌套表来保存有关所有表列的信息。

艾哈迈德创建或 替换 类型。“COL_INFO_OBJ” 是 对象

column_type NUMBER,column_name VARCHAR2(32),column_name_len NUMBER,column_schema_name VARCHAR2(32),column_precision NUMBER,column_scale NUMBER

);

/
CREATE OR REPLACE TYPE COL_INFO_OBJ_T AS TABLE OF col_info_obj;

/

我选择创建两张桌子; 一个是将要存储加密数据的保险库本身,另一个是保存有关插入操作的信息的主记录,在构建恢复/解密功能或过程时非常重要。为了演示NDS与DBMS_SQL一起使用,如果其中任何一个表不存在,该函数将创建所需的表。

功能 dynamic_vault_ins(table_name IN VARCHAR2,cond IN VARCHAR2 DEFAULT'1 = 1',

enc_key IN  VARCHAR2  DEFAULT  NULL,del_flag VARCHAR2  DEFAULT  '否')        返回 号码 是encrypt_typ       NUMBER:= DBMS_CRYPTO.encrypt_aes256 + DBMS_CRYPTO.chain_cbc + DBMS_CRYPTO.pad_pkcs5;enc_key_var       VARCHAR2(256);raw_col_v         RAW(2000);encrypted_col_v   RAW(2000);encr_row raw_tab:= raw_tab();sql_query         VARCHAR2(2000);col_count         NUMBER ;col_desc DBMS_SQL.desc_tab:= DBMS_SQL.desc_tab();cur_id            NUMBER ;row_counter       NUMBER:= 0 ;tab_exist         NUMBER ;num_var           NUMBER ;char_var          VARCHAR2(2000);date_var          日期 ;c_var             CHAR(4);row_id_var        NUMBER:= 1 ;opr_id_var        NUMBER:= 1 ;temp_var          NUMBER ;col_details col_info_obj_t:= col_info_obj_t();bad_cond          EXCEPTION ;开始    IF REGEXP_COUNT(UPPER(cond),'; | DELETE | DROP | CREATE | INSERT | UPDATE | GRANT | TRUNCATE')> 0  那么        提升 bad_cond;    结束 如果 ;    IF enc_key IS  NULL  或 LENGTH(enc_key)!= 32  那么        enc_key_var:= 'ahmed-rony-yousef-29102006-12-11' ;    其他        enc_key_var:= enc_key;    结束 如果 ;    SELECT  COUNT(tname)INTO tab_exist FROM  tab  WHERE tname = 'DATA_VAULT' ;    IF tab_exist = 0  那么        EXECUTE  IMMEDIATE'CREATE  TABLE data_vault(row_id number,opr_id NUMBER,            ins_date timestamp,table_name VARCHAR2(32),enc_data raw_tab)            NESTED TABLE enc_data STORE AS enc_data' ;    其他        EXECUTE  IMMEDIATE'SELECT  nvl(MAX(row_id),0)+ 1,nvl(MAX(opr_id),0)+ 1 FROM data_vault'        INTO row_id_var,opr_id_var;    结束 如果 ;    SELECT  COUNT(tname)INTO tab_exist FROM  tab  WHERE tname = 'VAULT_COL_INFO2' ;    IF tab_exist = 0  那么        EXECUTE  IMMEDIATE'CREATE  TABLE vault_col_info2(        opr_id NUMBER,inst_time TIMESTAMP,no_of_rows NUMBER,org_query varchar(2000),                 col_data col_info_obj_t)        嵌套表col_data存储为col_data2' ;    结束 如果 ;    / *构建查询* /    sql_query:= 'select * from' || DBMS_ASSERT.sql_object_name(table_name)|| 'where' || cond;    / *打开光标* /    cur_id:= DBMS_SQL.open_cursor;    / *解析光标* /    DBMS_SQL.PARSE(cur_id,sql_query,DBMS_SQL。天然);    / *构建列的描述并将结果存储在特殊的中       记录集合类型'dbms_sql.desc_tab'并获取列和的计数       存储在变量中(在这种情况下,它是col_count * /    DBMS_SQL.describe_columns(cur_id,col_count,col_desc);    / *扩展嵌套表,然后存储colums详细信息       objetcs的自定义嵌套表* /    col_details.EXTEND(col_count);    FOR i IN  1 ..col_count LOOP        col_details(i):= col_info_obj(            col_desc(ⅰ).col_type,            col_desc(ⅰ).col_name,            col_desc(ⅰ).col_name_len,            col_desc(ⅰ).col_schema_name,            col_desc(ⅰ).col_precision,            col_desc(ⅰ).col_scale);    结束 循环 ;    / *定义列类型       如果你要使用FETCH,你必须定义这些colun       你会得到一个错误* /    FOR i IN  1 ..col_count LOOP        case col_desc(i).col_type        当 2   那么                DBMS_SQL.define_column(cur_id,i,num_var);        当 12  那么                DBMS_SQL.define_column(cur_id,i,date_var);        当 1   那么                DBMS_SQL.define_column(cur_id,i,char_var,200);        当 96  那么                DBMS_SQL.define_column(cur_id,i,c_var,12);        END  情况 ;    结束 循环 ;    / *执行光标* /    temp_var:= DBMS_SQL。执行(cur_id);    encr_row.EXTEND(col_count);    / *从行中获取数据并对其进行加密* /    当 DBMS_SQL.fetch_rows(cur_id)> 0  LOOP时        FOR i IN  1 ..col_count LOOP            case col_desc(i).col_type            当 2   那么                DBMS_SQL.COLUMN_VALUE(cur_id,i,num_var);                / *将数据转换为RAW * /                raw_col_v:= UTL_I18N.string_to_raw(TO_CHAR(NVL(num_var,0)),'AL32UTF8');                - 加密数据                encrypted_col_v:= DBMS_CRYPTO.encrypt(src => raw_col_v,                    typ => encrypt_typ,key => UTL_I18N.string_to_raw(enc_key_var));                encr_row(i):= encrypted_col_v;
  • 当 12  那么            DBMS_SQL.COLUMN_VALUE(cur_id,i,date_var);            raw_col_v:= UTL_I18N.string_to_raw(TO_CHAR(NVL(date_var,TO_DATE('1999/07/15','YYYY / MM / DD'))),'AL32UTF8');            encrypted_col_v:= DBMS_CRYPTO.encrypt(src => raw_col_v,                typ => encrypt_typ,key => UTL_I18N.string_to_raw(enc_key_var));            encr_row(i):= encrypted_col_v;        当 1   那么            DBMS_SQL.COLUMN_VALUE(cur_id,i,char_var);            raw_col_v:= UTL_I18N.string_to_raw(NVL(char_var,' - '),'AL32UTF8');            encrypted_col_v:= DBMS_CRYPTO.encrypt(src => raw_col_v,                typ => encrypt_typ,key => UTL_I18N.string_to_raw(enc_key_var));            encr_row(i):= encrypted_col_v;        当 96  那么            DBMS_SQL.COLUMN_VALUE(cur_id,i,c_var);            raw_col_v:= UTL_I18N.string_to_raw(TO_CHAR(c_var),'AL32UTF8');            encrypted_col_v:= DBMS_CRYPTO.encrypt(src => raw_col_v,                typ => encrypt_typ,key => UTL_I18N.string_to_raw(enc_key_var));            encr_row(i):= encrypted_col_v;        其他            DBMS_SQL.COLUMN_VALUE(cur_id,i,char_var);            raw_col_v:= UTL_I18N.string_to_raw(char_var,'AL32UTF8');            encrypted_col_v:= DBMS_CRYPTO.encrypt(src => raw_col_v,                typ => encrypt_typ,key => UTL_I18N.string_to_raw(enc_key_var));            encr_row(i):= encrypted_col_v;        END  情况 ;    结束 循环 ;    / *这里你必须使用动态sql,因为如果表仍然不存在         编译器会引发错误'表或视图不存在'* /    EXECUTE  IMMEDIATE  '插入data_vault值(:1,:2,:3,:4,:5)'        使用 row_id_var,opr_id_var,SYSTIMESTAMP,table_name,encr_row;    row_id_var:= row_id_var + 1 ;    row_counter:= row_counter + 1 ;结束 循环 ;/ *这里你必须使用动态sql,因为如果表仍然不存在         编译器会引发错误'表或视图不存在'* /EXECUTE  IMMEDIATE  '插入vault_col_info2值(:1,:2,:3,:4,:5)'    使用 opr_id_var,SYSTIMESTAMP,row_counter,sql_query,col_details;IF del_flag = 'DELETE'  或 del_flag = 'YES'  那么    EXECUTE  IMMEDIATE  '删除' || table_name || 'where' || cond;    DBMS_OUTPUT.put_line(删除的总oreginal行数为:' || SQL%ROWCOUNT);结束 如果 ;RETURN row_counter;COMMIT ;DBMS_SQL.close_cursor(cur_id);例外WHEN bad_cond THEN    raise_application_error( - 20001,'Bad or dangerous condetions');    return row_counter;

    WHEN OTHERS THEN

    raise_application_error( - 20002,'发生了错误!==>' || SQLERRM);    return row_counter;

    END dynamic_vault_ins;

转载地址:http://hmavx.baihongyu.com/

你可能感兴趣的文章
hdu 2579 BFS
查看>>
[转]让Linux的tty界面支持中文
查看>>
Introduction to Change Data Capture (CDC) in SQL Server 2008[转]
查看>>
Vertex Texture Fetch(VTF) && Fragment Texture Fetch ( FTF )
查看>>
文件用户如何将一个有界面的正常app和一个或多个越狱插件.deb同时安装到手机上...
查看>>
Linux学习之CentOS(二十三)--Linux软件管理之源代码以及RPM软件包管理
查看>>
次优二叉树
查看>>
SQL Server CONVERT() 函数
查看>>
我看电商(作者近三十年从事零售及电子商务管理的总结和分享)
查看>>
Form身份验证
查看>>
【Android开发】Android应用程序目录结构
查看>>
腾讯大湘网某处csrf(city.hn.qq.com)可投诉刷留言
查看>>
也来谈谈这致命的手机充电器
查看>>
zendframework配置篇
查看>>
C++ 函数映射使用讲解
查看>>
域名的MX设置及校验方法
查看>>
从零开始学C++之STL(七):剩下5种算法代码分析与使用示例(remove 、rotate 、sort、lower_bound、accumulate)...
查看>>
飘逸的python - hack输出流便于调试
查看>>
2013年7月27日杂记
查看>>
Android应用程序之间共享文字和图片(一)
查看>>