SAP CDS+AMDP获取订单状态文本(替代函数STATUS_TEXT_EDIT)

背景:

客户反映某本报表运行效率极慢,需要进行优化,简单看了代码并且使用SAT分析后,定位了两个主要的问题,最主要的问题是合计数量的时候在LOOP循环中使用了SELECT SUM(*) FROM internal_table,而且还有很多个,导致当LOOP的本地内表条目比较大的时候,效率将会受到非常大的影响;其次是在循环中调用函数STATUS_TEXT_EDIT来获取订单文本,这个其实影响不算很大,但为了最大化提升效率,针对这点也做了优化。

优化前:time out dump。

优化后:10秒运行完成。


问题点1:

最大的元凶就是下面这段代码,LOOP中多个聚合函数导致随着数据量的增加,效率会异常低下。

解决方式:

sort table + collect 


问题点2:

循环中调用函数获取订单状态文本,对性能影响较小,不过本着能优则优的心态,顺手也对其进行了优化。

解决方式: 

table function + cds view 

1.首先定义table function:ZPRPP_ORDER_STATUS_TF

@ClientDependent: false
@AccessControl.authorizationCheck: #CHECK
@EndUserText.label: 'Table function for get PP order status'
define table function ZPRPP_ORDER_STATUS_TF
returns
{
  objnr          :j_objnr;
  status_profile :j_stsma;
  @EndUserText.label: 'System status'
  system_status  :char60;
}
implemented by method
  zprcl_st_text=>get_status_text;

2.然后创建实现类: zprcl_st_text

CLASS zprcl_st_text DEFINITION
  PUBLIC
  FINAL
  CREATE PUBLIC .

  PUBLIC SECTION.
    INTERFACES if_amdp_marker_hdb.


    CLASS-METHODS get_status_text FOR TABLE FUNCTION zprpp_order_status_tf.
  PROTECTED SECTION.
  PRIVATE SECTION.
ENDCLASS.



CLASS zprcl_st_text IMPLEMENTATION.
  METHOD get_status_text BY DATABASE FUNCTION
                               FOR HDB
                               LANGUAGE SQLSCRIPT
                               OPTIONS READ-ONLY
                               USING jest jsto tj02t tj02 tj04.

    declare lv_99 "$ABAP.type( numc2 )";

    lv_99 = 99;

    gt_status = select distinct
                       a.objnr,
                       b.stsma as status_profile,
                       d.txt04 as system_status,
                       E.nodis as no_display,

                       CASE
                         when f.linep is not null or f.linep <> 00 then
                           f.linep
                         else
                           lv_99
                       end as position,

                       CASE
                         when f.statp is not null or f.statp <> 00 then
                           f.statp
                         else
                           lv_99
                       end as priority

                  from jest as a
                 inner join jsto as b
                    on b.objnr = a.objnr
                  left outer join tj02t as d
                    on d.istat = a.stat
                 inner join tj02 as e on e.istat = a.stat
                  left outer join tj04 as f
                    on f.obtyp = b.obtyp
                   and f.istat = a.stat
                 where a.inact <> 'X'
                   and e.nodis <> 'X'
                   and a.mandt = session_context('CLIENT')
                   and b.mandt = session_context('CLIENT')
                   and d.spras = session_context('LOCALE_SAP')
                         order by POSITION desc,
                         priority desc,
                         system_status desc;

    RETURN SELECT objnr,
                  status_profile,
                  STRING_AGG( system_status, char( 32 ) ORDER BY position asc, priority asc, system_status asc ) as system_status
             FROM :gt_status
             GROUP BY objnr, status_profile;
  ENDMETHOD.
ENDCLASS.

3.最后创建cds view:

@AbapCatalog.sqlViewName: 'ZPRPPORDSTTXT'
@AbapCatalog.compiler.compareFilter: true
@AbapCatalog.preserveKey: true
@AccessControl.authorizationCheck: #CHECK
@EndUserText.label: 'CDS View for get PP order status text'
define view ZPRPP_ORDER_STATUS_V
  as select from ZPRPP_ORDER_STATUS_TF
{
  key objnr,
      status_profile,
      @EndUserText.label: 'System status'
      cast(system_status as abap.char(60)) as system_status
}

 4.在循环外直接根据AFKO-OBJNR关联该cds对应的sql view,一条sql即可获取订单状态。


注意点:

AMDP用的并不是特别熟练,有几个小的知识点也记录一下,方便我后续查找吧。

变量声明

        declare lv_99 "$ABAP.type( numc2 )";

系统变量:

  • session_context('CLIENT')  等同于 SY-MANDT
  • session_context('APPLICATIONUSER')  等同于 SY-UNAME
  • session_context('LOCALE_SAP') 等同于 SY-LANGU
  • session_context('SAP_SYSTEM_DATE') 等同于 SY-DATUM

空格:

        char( 32 )  代表空格

空格问题参考链接:The ABAPVARCHARMODE: Blanks and Empty Strings in ABAP and SQLScript (brandeis.de)

排序拼接: 

        基于函数STATUS_TEXT_EDIT的输出规则,在行转列拼接时要按照指定的顺序拼接展示,在string_agg方法中使用order by即可指定拼接顺序,asc为升序,desc为降序。


总结:

尽可能的避免在循环中多次出现数据库操作语句,程序性能优化是一个慢慢积累的过程,能跑出来结果永远是最低标准。

以上。

<think>好的,我现在需要解决用户关于SAP CDS视图和AMDP实现或故障排除的问题。首先,我得回忆一下CDS视图和AMDP的基础知识。 根据引用[1],AMDPABAP托管的数据库过程,主要用于在SAP HANA上执行高性能操作。CDS视图则是数据定义的工具,用于建模数据库层的数据结构。用户可能想知道如何将两者结合使用,或者在实现过程中遇到问题该如何排查。 接下来,我应该整理实现步骤。根据引用[4],创建AMDP需要定义一个类并实现IF_AMDP_MARKER_HDB接口,方法要用BY DATABASE PROCEDURE等语法。同时,CDS视图的创建需要遵循特定的注解,如@AbapCatalog.sqlViewName,这在引用[3]中有示例。 用户可能遇到的问题包括语法错误、权限问题或性能不佳。例如,SQLSCRIPT的语法与ABAP不同,容易出错。权限方面,可能需要检查是否有访问相关表或执行存储过程的权限。性能问题可能需要优化SQL语句,比如避免不必要的循环。 还要考虑如何结合CDS视图和AMDP。比如在AMDP中调用CDS视图,或者在CDS中使用AMDP函数。引用[3]中的例子展示了CDS如何调用AMDP函数获取订单状态,这可能是一个常见的应用场景。 故障排除部分,需要检查ABAP和HANA端的日志,使用ST22查看ABAP转储,HANA Studio检查执行计划。此外,注意AMDP方法的参数传递是否正确,数据类型是否匹配。 最后,应该生成相关问题,帮助用户深入理解,比如性能优化方法或调试技巧。</think>### SAP CDS视图与AMDP实现及故障排查指南 #### 一、CDS视图与AMDP基础概念 1. **CDS视图** Core Data Services (CDS)SAP HANA中用于定义数据模型的领域特定语言,支持在数据库层创建复杂的视图逻辑。例如: ```abap @AbapCatalog.sqlViewName: 'ZMATERIAL' define view ZMaterialView as select from mara { key matnr as Material, mbrsh as IndustrySector } ``` 该视图从表`MARA`中提取物料编号和行业领域字段[^3]。 2. **AMDP技术** AMDPABAP Managed Database Procedures)允许在ABAP类中直接编写HANA原生SQLScript代码,通过以下步骤实现: - 创建实现接口`IF_AMDP_MARKER_HDB`的ABAP类; - 在方法中使用`BY DATABASE PROCEDURE`声明存储过程[^4]。 #### 二、CDS视图与AMDP集成实现步骤 1. **场景示例:在CDS中调用AMDP函数** *目标*:通过AMDP生成订单状态文本,并在CDS中引用。 **步骤1:创建AMDP类** ```abap CLASS zcl_order_status DEFINITION PUBLIC. PUBLIC SECTION. INTERFACES if_amdp_marker_hdb. CLASS-METHODS get_status_text IMPORTING VALUE(iv_objnr) TYPE objnr EXPORTING VALUE(ev_text) TYPE string FOR DATABASE PROCEDURE LANGUAGE SQLSCRIPT OPTIONS READ-ONLY USING jest. ENDCLASS. CLASS zcl_order_status IMPLEMENTATION. METHOD get_status_text BY DATABASE PROCEDURE FOR HDB LANGUAGE SQLSCRIPT
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DeveloperMrMeng

觉得有用的佛系投币哦

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值