SAP 如何提供 RESTful Web 服务(2) - ABAP 与 JSON

系列文章索引


SAP 如何提供 RESTful Web 服务?
SAP 如何提供 RESTful Web 服务(2) - ABAP 与 JSON
SAP 如何提供 RESTful Web 服务(3) - Rest 路径处理


上一篇介绍了 SAP ABAP 提供 RESTful web service 从技术层面需要解决的三个问题:


  • SAP 对外提供 Http Service
  • 在 SAP 端解决诸如 authentication,cookie, session 等 Http server 必须解决的问题
  • SAP 端实现对响应 (response)以 json 格式返回给客户端,对客户端 json 格式的请求(request)解析并处理

本篇接着上一篇,介绍如何实现 ABAP 对象 与 JSON 互转。ABAP 与 JSON 互转有多种方法,本篇介绍两个类进行序列化和反序列化的方法:

  • CL_TREX_JSON_SERIALIZER :将 ABAP 内表转换为 json 格式
  • CL_TREX_JSON_DESERIALIZER:json 转换为 ABAP 内表

这两个类有一点点问题,序列化生成的格式 key 没有引号,可能导致外部解析的失败。比如

[
    {
        id: "001",
        name: "Stone"
    },
    {
        id: "002",
        name: "Brown"
    }
]

而我们需要的是下面的格式:

[
    {
        "id": "001",
        "name": "Stone"
    },
    {
        "id": "002",
        "name": "Brown"
    }
]

所以需要对两个类进行改造,以符合自己的需求。首先我们使用事务码 SE24CL_TREX_JSON_SERIALIZER 类拷贝一个新类,另存为 ZCL_TREX_JSON_SERIALIZER,然后将 RECURSE() 方法做两点改变:

填加语句:

注释掉 concatenate 语句,改写为下面的语句。就是在 <abapcomp>-name 两边加上双引号 :

完整的 RECURSE() 方法代码如下:

METHOD recurse.

  DATA:
  l_type  TYPE c ,
  l_comps TYPE i ,
  l_lines TYPE i ,
  l_index TYPE i ,
  l_value TYPE string .
  FIELD-SYMBOLS:
    <itab> TYPE ANY TABLE ,
    <comp> TYPE ANY .

  DESCRIBE FIELD data TYPE l_type COMPONENTS l_comps .

  IF l_type = cl_abap_typedescr=>typekind_table .
*   itab -> array
    APPEND '[' TO me->fragments .
    ASSIGN data TO <itab> .
    l_lines = LINES( <itab> ) .
    LOOP AT <itab> ASSIGNING <comp> .
      ADD 1 TO l_index .
      recurse( <comp> ) .
      IF l_index < l_lines .
        APPEND c_comma TO me->fragments .
      ENDIF .
    ENDLOOP .
    APPEND ']' TO fragments .
  ELSE .
    IF l_comps IS INITIAL .
*     field -> scalar
*     todo: format
      l_value = data .
      REPLACE ALL OCCURRENCES OF '\' IN l_value WITH '\\' .
      REPLACE ALL OCCURRENCES OF '''' IN l_value WITH '\''' .
      REPLACE ALL OCCURRENCES OF '"' IN l_value WITH '\"' .
      REPLACE ALL OCCURRENCES OF '&' IN l_value WITH '\&' .
      REPLACE ALL OCCURRENCES OF cl_abap_char_utilities=>cr_lf IN l_value WITH '\r\n' .
      REPLACE ALL OCCURRENCES OF cl_abap_char_utilities=>newline IN l_value WITH '\n' .
      REPLACE ALL OCCURRENCES OF cl_abap_char_utilities=>horizontal_tab IN l_value WITH '\t' .
      REPLACE ALL OCCURRENCES OF cl_abap_char_utilities=>backspace IN l_value WITH '\b' .
      REPLACE ALL OCCURRENCES OF cl_abap_char_utilities=>form_feed IN l_value WITH '\f' .
      CONDENSE l_value.
      CONCATENATE '"' l_value '"' INTO l_value .
      APPEND l_value TO me->fragments .
    ELSE .
*     structure -> object
      DATA l_typedescr TYPE REF TO cl_abap_structdescr .
      FIELD-SYMBOLS <abapcomp> TYPE abap_compdescr .

      APPEND '{' TO me->fragments .
      l_typedescr ?= cl_abap_typedescr=>describe_by_data( data ) .
      LOOP AT l_typedescr->components ASSIGNING <abapcomp> .
        l_index = sy-tabix .
*        CONCATENATE <abapcomp>-name c_colon INTO l_value .
        CONCATENATE '"' <abapcomp>-name '"' c_colon INTO l_value.
        TRANSLATE l_value TO LOWER CASE .
        APPEND l_value TO me->fragments .
        ASSIGN COMPONENT <abapcomp>-name OF STRUCTURE data TO <comp> .
        recurse( <comp> ) .
        IF l_index < l_comps .
          APPEND c_comma TO me->fragments .
        ENDIF .
      ENDLOOP .
      APPEND '}' TO me->fragments .
    ENDIF .
  ENDIF .

ENDMETHOD.

相对应地, 将 CL_TREX_JSON_DESERIALIZER 拷贝成为 ZCL_TREX_JSON_DESERIALIZER,并且对 deserialize_object 方法做两点变更 (网上有一个做好的图片,我就直接贴图了):

图片来源:https://archive.sap.com/image/665574

完成后,deserialize_object 方法的完整代码如下:

METHOD deserialize_object.
  DATA:
  l_node_type TYPE REF TO cl_abap_typedescr ,
  l_ref TYPE REF TO object .

  ADD 1 TO offset . "skip {

  l_node_type = cl_abap_typedescr=>describe_by_data( node ) .

* prepare for dynamic access
  CASE l_node_type->kind .
    WHEN cl_abap_typedescr=>kind_ref .
      l_ref = node .
    WHEN cl_abap_typedescr=>kind_struct .

    WHEN OTHERS .
      RAISE EXCEPTION TYPE cx_trex_serialization .
  ENDCASE .

  DATA:
    l_done TYPE abap_bool ,
    l_len TYPE i ,
    l_name TYPE string .

* handle each component
  WHILE l_done = abap_false .
    "find next key
    FIND REGEX '"(\w+)\s*":' IN SECTION OFFSET offset OF json
      MATCH OFFSET offset MATCH LENGTH l_len
      SUBMATCHES l_name .
    IF sy-subrc <> 0 .
      RAISE EXCEPTION TYPE cx_trex_serialization .
    ENDIF .
    ADD l_len TO offset .

    FIELD-SYMBOLS <comp> TYPE ANY .

*   dynamic binding to component
    TRANSLATE l_name TO UPPER CASE .
    CASE l_node_type->kind .
      WHEN cl_abap_typedescr=>kind_ref .
        ASSIGN l_ref->(l_name) TO <comp> .
      WHEN cl_abap_typedescr=>kind_struct .
        ASSIGN COMPONENT l_name OF STRUCTURE node TO <comp> .
        IF sy-subrc <> 0.
          CONTINUE.
        ENDIF.

      WHEN OTHERS .
        RAISE EXCEPTION TYPE cx_trex_serialization .
    ENDCASE .

    DATA:
      l_comp_type TYPE REF TO cl_abap_typedescr ,
      l_ref_type TYPE REF TO cl_abap_refdescr .

*   check component type
    l_comp_type = cl_abap_typedescr=>describe_by_data( <comp> ) .
    CASE l_comp_type->kind .
*     create instance if it's an oref
      WHEN cl_abap_typedescr=>kind_ref .
        l_ref_type ?= l_comp_type .
        l_comp_type = l_ref_type->get_referenced_type( ) .
        CREATE OBJECT <comp> TYPE (l_comp_type->absolute_name) .
    ENDCASE .

*   deserialize current component
    deserialize_node(
      EXPORTING
        json = json
      CHANGING
        offset = offset
        node = <comp> ) .

    FIND REGEX ',|\}' IN SECTION OFFSET offset OF json MATCH OFFSET offset .
    IF sy-subrc <> 0 .
      RAISE EXCEPTION TYPE cx_trex_serialization .
    ENDIF .

    IF json+offset(1) = '}' .
      l_done = abap_true .
    ENDIF .
    ADD 1 TO offset .
  ENDWHILE .

ENDMETHOD.

示例

调用 ZCL_TREX_JSON_SERIALIZER 实现序列化以及调用 ZCL_TREX_JSON_DESERIALIZER 实现反序列化的代码如下:

REPORT  zabap_json .

DATA: serializer TYPE REF TO zcl_trex_json_serializer,
      lv_json TYPE string.

DATA: BEGIN OF ls_json,
  id TYPE string,
  name TYPE string,
  END OF ls_json.

DATA: lt_json LIKE STANDARD TABLE OF ls_json.

CLEAR ls_json.
ls_json-id = '001'.
ls_json-name = 'Stone'.

APPEND ls_json TO lt_json.

CLEAR ls_json.
ls_json-id = '002'.
ls_json-name = 'Brown'.

APPEND ls_json TO lt_json.

CREATE OBJECT serializer
    EXPORTING DATA = lt_json[].
CALL METHOD serializer->serialize( ).
lv_json = serializer->get_data( ).

WRITE / lv_json.

* Deserialze

WRITE /.
WRITE: / 'Deseriaze json string to internal table: '.

DATA: deserializer TYPE REF TO zcl_trex_json_deserializer.
CLEAR lt_json.

CREATE OBJECT deserializer.
CALL METHOD deserializer->deserialize(
  EXPORTING json = lv_json
  IMPORTING abap = lt_json[] ).

LOOP AT lt_json INTO ls_json.
  WRITE : / ls_json-id, ls_json-name.
ENDLOOP.

参考

How to deserialize an anonymous JSON array in abap?

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343