SAP ABAP基础知识 访问外部数据库-开发篇

前言

本文主要介绍通过ABAP语言访问外部数据库的几种方式

一、外部数据库配置

本文示例中的代码访问了两个外部数据库

MTD : 外部oracle数据库,其中示例表 ZTTEMP 字段( ZZTNO,WERKS)

S4Q : 外部HANA数据库(开发系统访问测试系统的数据库), 使用表USR02,ZTTEMP

二、ABAP访问外部数据库

通过ABAP访问外部数据库有四种方式.根据不同的情况,可以选择不同的方法.

OPEN SQL访问
NATIVE SQL 访问
ADBC(ABAP Database Connectivity)
AMDP ABAP Managed Database Procedures ? (未验证通过)


 

三、OPEN SQL直接访问

OPEN SQL 访问的限制条件:必须在ABAP数据字典中存在该表名,并且最好同目标系统表结构一致, 一般情况下,用来访问另外一个同版本的ECC数据库.当然,也可以把ECC的表定义语句在目标系统中创建一个同名同结构的表,然后用该方式访问.

直接访问时,在FROM TABLE 后面添加 CONNECTION s4q .

s4q是DBCO中建立的和另外一个S/4系统的连接

01

报错及处理一

可能的报错及处理方式

下图报错的原因是访问ORACLE数据库必须指定一个SCHEMA. 这个可以配置在连接参数中.

01

报错及处理二

出现下面的报错,表示系统尝试使用MANDT限制数据, 此时需要给OPEN SQL 语句添加CLIENT SPECIFIED 强制OPEN SQL 不要补充MANDT限制

四、NATIVE SQL访问

通过NATIVE SQL 访问外部数据库步骤

打开连接
执行SQL命令
关闭连接
示例代码见文末

01

读取多条记录的方式

游标方式     


非游标方式


非游标方式其实隐式使用了游标.性能比游标方式要差.数据量小的时候看不出来. 大量数据读取就能看出二者的性能差异了. 

标准帮助中提示的优劣比较.

五、ADBC访问

ADBC(ABAP Database Connectivity) 是SAP提供的原生SQL(Native SQL)接口API.可以通过ADBC执行任何数据库的原生SQL语句.

ABAP中有个标准的DEMO程序: ADBC_DEMO 演示了各种SQL语句的调用方式.图四的代码示例给出了SELECT语句的常用写法.

大概需要如下的过程

创建默认数据库的链接对象
创建一个查询对象
基于sql语句创建一个结果对象
定义传递结果集一个数据对象-内表
获取数据内容
关闭连接
赋值数据到内表
示例代码详见文末

六、通过AMDP 访问?

AMDP ABAP Managed Database Procedures

标准ABAP帮助体系中提到访问外部数据库的方法中还有一种AMDP方式.为了完善本文,补充了一个AMDP访问外部数据库表的示例.(验证未通过)

尝试中发现AMDP只适用与HANA数据库. 并且尝试通过DEMO程序 DEMO_AMDP_CONNECTION 连接外部数据库S4Q. 尝试失败. (报错见图四)

感觉AMDP 并不支持连接外部数据库. 

图五中示例代码的连接 是 R/3*开头. 帮助中说S/3*是SERVICE CONNECT. 但是不理解有什么用

处.

 

七、总结

ABAP访问外部数据库的几种方式中. OPEN SQL 最简单,但是有很大局限性. NATIVE方式最简单易懂. 性能也最好. 但是不利于代码动态化. ADBC 可以动态的实现数据的读取及内表的赋值.

前文DB02 SQL编辑器SQL语句自动生成报表 就采用了ADBC访问数据库的方法: 根据语句动态定义选择屏幕,动态定义内表, 读取的数据写入内表呈现.

详见链接无峰,公众号:ABAP开发技巧SAP工具箱之一键生成报表

该工具在付费文章中可以获取.

文中通过AMDP方式连接外部数据库的验证失败. 不推荐使用. 其它三种方式根据实际情况选择使用就好.

示例代码详见文末.

示例代码,

*&---------------------------------------------------------------------*
*& Report ZTS_SQL_DBCO
*&---------------------------------------------------------------------*
*&
*&---------------------------------------------------------------------*
REPORT zts_dbco_opensql.PARAMETERS: p_s4  RADIOBUTTON GROUP ra1,p_ora RADIOBUTTON GROUP ra1.START-OF-SELECTION.CASE 'X'.WHEN p_s4.
*访问另一个S4系统的同名表
*需要注意的是,目标系统CLIENT和当前CLIENT 很可能不一样, 所以需要加上CLIENT SPECIFIED 避免CLIENT不同干扰数据获取SELECT * FROM usr02  CLIENT SPECIFIED  INTO  TABLE @DATA(lt_usr02)  BYPASSING BUFFER CONNECTION R/3*S4QWHERE bname = '00177'.."可以获取数据DATA(lv_subrc) = sy-subrc .cl_demo_output=>write( lv_subrc ).cl_demo_output=>write( lt_usr02 ).cl_demo_output=>display(  ).WHEN p_ora.
*访问另一个其它系统的同名表
*如果ABAP表有MANDT , 目标表没有, 则需要添加CLIENT SPECIFIED 避免系统自动添加MANDT 的限制条件,导致报错:字段MANDT不存在DATA: BEGIN OF ls_temp,zztno(30),werks(4),END OF ls_temp.DATA: lt_temp LIKE TABLE OF ls_temp.SELECT zztno,werks FROM zttemp CLIENT SPECIFIED CONNECTION mtdINTO TABLE @lt_temp.cl_demo_output=>display( lt_temp ).ENDCASE.
*  COMMIT CONNECTION s4q. "在连接中提交.

示例代码 NATIVESQL

*&---------------------------------------------------------------------*
*& Report ZTS_SQL_DBCO
*&---------------------------------------------------------------------*
*&
*&---------------------------------------------------------------------*
REPORT zts_dbco_nativesql.PARAMETERS:p_1 RADIOBUTTON GROUP ra1,p_2 RADIOBUTTON GROUP ra1,p_3 RADIOBUTTON GROUP ra1,p_4 RADIOBUTTON GROUP ra1.DATA: BEGIN OF gs_temp,zztno(30),werks(4),END OF gs_temp.
DATA: gt_temp LIKE TABLE OF gs_temp.
DATA conn TYPE dbcon-con_name.INITIALIZATION.%_p_1_%_app_%-text = 'SELECT方式一:DO循环读取游标,添加内表'.%_p_2_%_app_%-text = 'SELECT方式二:通过例程添加内表'.%_p_3_%_app_%-text = 'UPDATE:更新表内容'.%_p_4_%_app_%-text = 'INSERT:写入表内容'.START-OF-SELECTION.conn = 'MTD'."检查连接是否已经打开EXEC SQL.SET CONNECTION :connENDEXEC.IF sy-subrc <> 0. "如果连接没有打开, 打开连接EXEC SQL.CONNECT TO :connENDEXEC.ENDIF.*两种方式: 方式一性能好于方式二CASE 'X'.WHEN p_1.PERFORM frm_method_1. "SELECT方式一:DO循环读取游标,添加内表'.WHEN p_2.PERFORM frm_method_2. "SELECT方式二:通过例程添加内表'.when p_3.perform frm_update.when p_4.perform frm_insert.ENDCASE."关闭数据库连接EXEC SQL.DISCONNECT :CONNENDEXEC.*输出结果CASE 'X'.WHEN p_1 or p_2.cl_demo_output=>display( gt_temp ).ENDCASE.
*&---------------------------------------------------------------------*
*& Form FRM_METHOD_1
*&---------------------------------------------------------------------*
*& text
*&---------------------------------------------------------------------*
*& -->  p1        text
*& <--  p2        text
*&---------------------------------------------------------------------*
FORM frm_method_1 .DATA: ls_temp LIKE gs_temp,lt_temp LIKE TABLE OF ls_temp."执行SQL语句:通过open dbcur打开游标EXEC SQL.OPEN dbcur FORSELECT zztno,werks FROM zttempENDEXEC."循环通过游标读取记录" 两种赋值方式:" 1.按字段顺序赋值,select 字段与 INTO 字段顺序必须一致"   FETCH NEXT dbcur INTO :ls_TEMP-ZZTNO,:LS_TEMP-WERKS" 2.按结构整体赋值:select 字段必须与结构字段顺序一致,且字段长度一致."   FETCH NEXT dbcur INTO :ls_TEMPDO.EXEC SQL.FETCH NEXT dbcur INTO :ls_TEMP-ZZTNO,:LS_TEMP-WERKSENDEXEC.IF sy-subrc <> 0.EXIT.ELSE.APPEND ls_temp TO lt_temp.ENDIF.ENDDO."关闭游标EXEC SQL.CLOSE dbcurENDEXEC.gt_temp[] = lt_temp[].
ENDFORM.
*&---------------------------------------------------------------------*
*& Form FRM_METHOD_2
*&---------------------------------------------------------------------*
*& text
*&---------------------------------------------------------------------*
*& -->  p1        text
*& <--  p2        text
*&---------------------------------------------------------------------*
FORM frm_method_2 .conn = 'MTD'."检查连接是否已经打开EXEC SQL.SET CONNECTION :connENDEXEC.IF sy-subrc <> 0. "如果连接没有打开, 打开连接EXEC SQL.CONNECT TO :connENDEXEC.ENDIF.
*注意:工作区 gs_temp 内表 gt_temp 必须是全局变量EXEC SQL PERFORMING FRM_FILL_DATA.SELECT zztno,werks FROM zttemp INTO :GS_TEMPENDEXEC.ENDFORM.
FORM frm_fill_data.APPEND gs_temp TO gt_temp.ENDFORM.
*&---------------------------------------------------------------------*
*& Form FRM_UPDATE
*&---------------------------------------------------------------------*
*& text
*&---------------------------------------------------------------------*
*& -->  p1        text
*& <--  p2        text
*&---------------------------------------------------------------------*
FORM frm_update .DATA: lv_werks(4).lv_werks = '1002'.EXEC SQL.UPDATE ZTTEMP SET WERKS = :LV_WERKSWHERE WERKS = '1003'ENDEXEC.IF sy-subrc = 0.
*提交数据更新EXEC SQL.COMMIT WORKENDEXEC.DATA: lv_msg(50).lv_msg = '更新成功记录数:' && sy-dbcnt .cl_demo_output=>display(  lv_msg ).ELSE.cl_demo_output=>display( '更新失败' ).ENDIF.
ENDFORM.
*&---------------------------------------------------------------------*
*& Form FRM_INSERT
*&---------------------------------------------------------------------*
*& text
*&---------------------------------------------------------------------*
*& -->  p1        text
*& <--  p2        text
*&---------------------------------------------------------------------*
FORM frm_insert .DATA: lv_werks(4).lv_werks = '1002'.EXEC SQL.INSERT INTO ZTTEMP VALUES ('4502',:LV_WERKS)ENDEXEC.IF sy-subrc = 0.
*提交数据更新EXEC SQL.COMMIT WORKENDEXEC.DATA: lv_msg(50).lv_msg = '写入成功记录数:' && sy-dbcnt .cl_demo_output=>display(  lv_msg ).ELSE.cl_demo_output=>display( '写入失败' ).ENDIF.
ENDFORM.

*&---------------------------------------------------------------------*
*& Report ZTS_DBCO_ADBC
*&---------------------------------------------------------------------*
*&
*&---------------------------------------------------------------------*
REPORT zts_dbco_adbc.DATA: BEGIN OF gs_temp,zztno(30),werks(4),END OF gs_temp.
DATA: gt_temp LIKE TABLE OF gs_temp.
DATA conn TYPE dbcon-con_name.
DATA: gv_sql TYPE string.START-OF-SELECTION.gv_sql = 'SELECT zztno,werks FROM zttemp'.PERFORM frm_get_data_adbc_simple.
*  PERFORM frm_get_data_adbc.cl_demo_output=>display( gt_temp ).
*&---------------------------------------------------------------------*
*& Form FRM_GET_DATA_ADBC
*&---------------------------------------------------------------------*
*& text
*&---------------------------------------------------------------------*
*& -->  p1        text
*& <--  p2        text
*&---------------------------------------------------------------------*
FORM frm_get_data_adbc .DATA: r_adbc_conn   TYPE REF TO  cl_sql_connection,r_adbc_query  TYPE REF TO  cl_sql_statement,r_metadata    TYPE REF TO  data,it_metadata   TYPE         adbc_rs_metadata_descr_tab,lv_len        TYPE         i,lv_off        TYPE         i,wa_metadata   LIKE LINE OF it_metadata,r_adbc_result TYPE REF TO  cl_sql_result_set,r_tabletype   TYPE REF TO  cl_abap_tabledescr,r_cxadbc      TYPE REF TO  cx_dba_adbc,r_cxsql       TYPE REF TO  cx_sql_exception,tabix_n(4)    TYPE n,column_names  TYPE HASHED TABLE OF adbc_name WITH UNIQUE KEY table_line.DATA:         lv_stmt_type      TYPE string.DATA:ex_structdescr TYPE REF TO  cl_abap_structdescr,ex_result_ref  TYPE REF TO data.
*获取sql语句的类型lv_stmt_type = cl_hdb_sql_executor=>get_statement_type( gv_sql ).
*创建默认数据库的链接对象r_adbc_conn    = cl_db6_con=>get_connection( 'MTD' ).
*创建一个查询对象r_adbc_query   = r_adbc_conn->create_statement( ).
*基于sql语句创建一个结果对象r_adbc_result  = r_adbc_query->execute_query( gv_sql  ).
*获取结果集合的字段名it_metadata = r_adbc_result->get_metadata( ).
*使用结果集合的字段信息,创建一个数据对象-结构r_metadata = r_adbc_result->get_struct_ref( md_tab = it_metadata   p_strict = abap_false ).
*创建一个数据对象-内表ex_structdescr ?= cl_abap_typedescr=>describe_by_data_ref( r_metadata ).r_tabletype     = cl_abap_tabledescr=>create( p_line_type  = ex_structdescrp_table_kind = cl_abap_tabledescr=>tablekind_std ).CREATE DATA ex_result_ref TYPE HANDLE r_tabletype.
*传递结果集一个数据对象-内表r_adbc_result->set_param_table( itab_ref = ex_result_ref ).
*获取数据内容r_adbc_result->next_package( EXPORTING upto = 100 ).
*关闭连接r_adbc_result->close( ).
*赋值数据到内表FIELD-SYMBOLS: <fs_itab> TYPE STANDARD TABLE.ASSIGN ex_result_ref->* TO <fs_itab>.MOVE-CORRESPONDING <fs_itab> TO gt_temp.
ENDFORM.FORM frm_get_data_adbc_simple .DATA: r_adbc_conn   TYPE REF TO  cl_sql_connection,r_adbc_query  TYPE REF TO  cl_sql_statement,r_metadata    TYPE REF TO  data,it_metadata   TYPE         adbc_rs_metadata_descr_tab,lv_len        TYPE         i,lv_off        TYPE         i,wa_metadata   LIKE LINE OF it_metadata,r_adbc_result TYPE REF TO  cl_sql_result_set,r_tabletype   TYPE REF TO  cl_abap_tabledescr,r_cxadbc      TYPE REF TO  cx_dba_adbc,r_cxsql       TYPE REF TO  cx_sql_exception,tabix_n(4)    TYPE n,column_names  TYPE HASHED TABLE OF adbc_name WITH UNIQUE KEY table_line.DATA:         lv_stmt_type      TYPE string.DATA:ex_structdescr TYPE REF TO  cl_abap_structdescr,ex_result_ref  TYPE REF TO data.*创建默认数据库的链接对象r_adbc_conn    = cl_db6_con=>get_connection( 'MTD' ).
*创建一个查询对象r_adbc_query   = r_adbc_conn->create_statement( ).
*基于sql语句创建一个结果对象r_adbc_result  = r_adbc_query->execute_query( gv_sql  ).*定义DATA: lr_ref LIKE REF TO gt_temp.CREATE DATA lr_ref .
*传递结果集一个数据对象-内表r_adbc_result->set_param_table( itab_ref = lr_ref ).
*获取数据内容r_adbc_result->next_package( EXPORTING upto = 100 ).
*关闭连接r_adbc_result->close( ).
*赋值数据到内表gt_temp = lr_ref->*.
ENDFORM.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/108854.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HBS 家庭总线驱动和接收芯片MS1192,应用于电话及相关设备、空调设备、安全设备、AV 装置

MS1192 是适用于 HBS 总线规范&#xff08;日本电子工业协会&#xff09; 的适配器芯片&#xff0c;具备发送、接收数据的功能。在发送接收 单元中&#xff0c;采用 AMI 编码方式&#xff0c;可使用双绞线进行互联&#xff0c;信 号传输采用差分方式。 芯片采用单电源…

MFC-GetAdaptersAddresses获取网卡信息

需要:#pragma comment(lib, "IPHLPAPI.lib") GetAdaptersAddresses函数参数说明 ULONG bufferSize = 0;ULONG result = ::GetAdaptersAddresses(AF_UNSPEC, GAA_FLAG_INCLUDE_PREFIX, nullptr, nullptr, &bufferSize);/*参数1:ULONG Family 网络协议族,此参…

CG MAGIC分享3ds Max卡顿未保存处理方法有哪些?

3ds Max进行建模、渲染这一系列过程中&#xff0c;大家使用中都会遇到各种原因导致软件卡顿或崩溃是很常见的情况。 可以说卡机没关系&#xff0c;可是卡顿发生时&#xff0c;如果之前的工作没有及时保存&#xff0c;可能会导致数据的丢失和时间的浪费。这就是最让人烦躁的了&…

iframe 实现跨域,两页面之间的通信

一、 背景 一个项目为vue2&#xff0c;一个项目为vue3&#xff0c;两个不同的项目实现iframe嵌入&#xff0c;并实现通信 二、方案 iframe跨域时&#xff0c;iframe组件之间常用的通信&#xff0c;主要是H5的possmessage方法 三、案例代码 父页面-vue2&#xff08;端口号为…

《PostgreSQL数据分区:原理与实战》

&#x1f337;&#x1f341; 博主猫头虎&#xff08;&#x1f405;&#x1f43e;&#xff09;带您 Go to New World✨&#x1f341; &#x1f405;&#x1f43e;猫头虎建议程序员必备技术栈一览表&#x1f4d6;&#xff1a; &#x1f6e0;️ 全栈技术 Full Stack: &#x1f4da…

vue基础知识十:Vue中组件和插件有什么区别?

一、组件是什么 回顾以前对组件的定义&#xff1a; 组件就是把图形、非图形的各种逻辑均抽象为一个统一的概念&#xff08;组件&#xff09;来实现开发的模式&#xff0c;在Vue中每一个.vue文件都可以视为一个组件 组件的优势 降低整个系统的耦合度&#xff0c;在保持接口不…

DatasetDM: Synthesizing Data with Perception Annotations Using Diffusion Models

论文作者&#xff1a;Weijia Wu,Yuzhong Zhao,Hao Chen,Yuchao Gu,Rui Zhao,Yefei He,Hong Zhou,Mike Zheng Shou,Chunhua Shen 作者单位&#xff1a;Zhejiang University; University of Chinese Academy of Sciences; National University of Singapore 论文链接&#xff1…

睿趣科技:抖音开店前期需要准备什么

抖音作为全球最受欢迎的短视频平台之一&#xff0c;已经成为了许多年轻人的创业和赚钱的机会。如果你计划在抖音上开店&#xff0c;那么在正式开业之前&#xff0c;有一些重要的准备工作是必不可少的。下面就是抖音开店前期需要准备的关键步骤和注意事项。 确定你的目标和产品&…

2023年浦东新区数字化安全风险智慧管控技能比武初赛-技能题一

目录 二、技能题 2.1 MD5===MD5 三、业*&&&务**&&联&&&*&&系 二、技能题 2.1 MD5===MD5

C++多线程的用法(包含线程池小项目)

一些小tips: 编译命令如下&#xff1a; g 7.thread_pool.cpp -lpthread 查看运行时间&#xff1a; time ./a.out 获得本进程的进程id&#xff1a; this_thread::get_id() 需要引入的库函数有&#xff1a; #include<thread> // 引入线程库 #include<mutex> //…

【C语言】每日一题(半月斩)——day2

目录 一.选择题 1、以下程序段的输出结果是( ) 2、若有以下程序&#xff0c;则运行后的输出结果是&#xff08; &#xff09; 3、如下函数的 f(1) 的值为&#xff08; &#xff09; 4、下面3段程序代码的效果一样吗( ) 5、对于下面的说法&#xff0c;正确的是&#xf…

Marin说PCB之封装设计系列---(02)--异形焊盘的封装设计总结

每天下班回家看电视本来是一件很美好的事情&#xff0c;可是正当我磕着瓜子看着异人之下的时候&#xff0c;手机突然响起来了&#xff0c;我以为是我们组哪个同事找我呢。一接电话居然是我的老朋友陈世美陈总&#xff0c;江湖人称少妇杀手。给我打电话主要是说他最近遇到一个异…