如何对SAP数据库表进行增删改查操作

本文介绍如何对SAP数据库表进行增删改查操作,重点是说明如何为数据的增删改查提供编辑界面。假定现在要对 zemployee 表进行增删改查操作。表的字段如下:

通过ABAP 代码进行 CRUD 操作

INSERT 语句对 database table 的记录进行插入操作。语法如下:

INSERT dbtab from wa/itab. 

如果至少有一笔记录插入成功,sy-subrc = 0,如果至少有一笔记录插入失败, sy-subrc =4,比如可能由于 primary key 重复。

DATA: gs_emp LIKE zemployee.

CLEAR gs_emp.
gs_emp-empid = '003'.
gs_emp-empname = 'WANG'.
gs_emp-empaddr = 'Wuhan'.

INSERT zemployee1 FROM gs_emp.

IF sy-subrc IS INITIAL.
  WRITE sy-dbcnt.
ELSE.
  WRITE 'Error'.
ENDIF.

UPDATE 语句允许对数据库表的记录进行修改。

DATA: gs_emp LIKE zemployee1.

CLEAR gs_emp.
gs_emp-empid = '003'.
gs_emp-empname = 'Stone Wang'.
gs_emp-empaddr = 'Wuhan'.

UPDATE zemployee1 FROM gs_emp.

IF sy-subrc IS INITIAL.
  WRITE sy-dbcnt.
ELSE.
  WRITE 'Error'.
ENDIF.

如果因为不知道要插入行的 Primary key 是否已经存在 ,从而不能确定是否可使用 UPDATE 语句,那么 ,请使用 MODIFY 语句。MODIFY 语句既可以插入,也可以更新。

要从数据库表中删除行 ,请使用 DELETE 语句。DELETE 语句允许删除单行或多行。个人觉得删除记录的操作,没有必要使用工作区,可以直接通过条件来定位要删除的行。

DELETE FROM zemployee1 WHERE empid = '003'.

IF sy-subrc IS INITIAL.
  WRITE sy-dbcnt.
ELSE.
  WRITE 'Error'.
ENDIF.

表维护生成器

为了方便对数据表的数据维护,最好能提供维护的界面。表维护生成器 (Table maintenance generator) 是 SAP 对数据表提供的一种通用方法,在 SE1 界面,通过菜单 Utilities -> Table Maintenance Generator 进行设置。设置界面如下:

image

设置了表维护生成器的 table,可以使用事务码 SM30 进行数据的维护(增删改查)。SM30 界面可以通过事务码 SE93 创建一个事务码来维护表。使用 SE93 并结合 SM30 的功能来创建事务码的方法如下:

使用事务码 SE93 进入界面,在 Transaction Code 字段输入 ZEMP,点击创建按钮。


选择 Transaction with parameters


image

在下一屏幕中,Transaction 输入 SM30,勾上 Skip initial screen:


在最下面的 Default Values 部分,给定 VIEWNAME 和 UPDATE 两个参数:

VIEW_MAINTAINENCE_CALL 函数

如果某个表设置了表维护生成器,可以利用函数 VIEW_MAINTAINENCE_CALL 来对表进维护,界面同 SM30 相同。

CALL FUNCTION 'VIEW_MAINTENANCE_CALL'
  EXPORTING
    action    = 'U'
    view_name = 'ZEMPLOYEE'.

RS_TABLE_LIST_CREATE 函数

RS_TABLE_LIST_CREATE 函数比 VIEW_MAINTENANCE_CALL 更加灵活,即使表没有维护“表维护生成器”,也可以通过代码来进行数据维护操作。

CALL FUNCTION 'RS_TABLE_LIST_CREATE'
  EXPORTING
    table_name               = 'ZEMPLOYEE'
    action                   = 'ANLE'.

action 参数有 ANLE 或 ANZE,将使用不同的界面来进行数据维护。

通过 ALV 控件

对于简单的数据表,可以使用 ALV 表格作为界面。下面介绍通过 ALV 控件来维护表数据的方法。

第一步,编写一个程序,将数据在 ALV 中显示。代码如下:

REPORT  z_table_crud.

TYPE-POOLS: slis.
DATA: gt_fieldcat TYPE slis_t_fieldcat_alv WITH HEADER LINE.

DATA: gt_zemp TYPE STANDARD TABLE OF zemployee,
      gs_zemp LIKE LINE OF gt_zemp.


START-OF-SELECTION.
  PERFORM get_data.
  PERFORM show_data.

FORM get_data.
  SELECT * FROM zemployee
    INTO CORRESPONDING FIELDS OF TABLE gt_zemp.
ENDFORM.                    "get_data

FORM set_fieldcat USING fldname desc editable.
  DATA: ls_fieldcat TYPE slis_fieldcat_alv.

  CLEAR ls_fieldcat.
  ls_fieldcat-fieldname = fldname.
  ls_fieldcat-seltext_m = desc.
  ls_fieldcat-edit = editable.
  APPEND ls_fieldcat TO gt_fieldcat.
ENDFORM.                    "set_fieldcat

FORM show_data.

  PERFORM set_fieldcat USING 'EMPID' 'Employee ID' 'X'.
  PERFORM set_fieldcat USING 'EMPNAME' 'Employee Name' 'X'.
  PERFORM set_fieldcat USING 'EMPADDR' 'Employee Addr' 'X'.

  CALL FUNCTION 'REUSE_ALV_GRID_DISPLAY'
   EXPORTING
     it_fieldcat                       = gt_fieldcat[]
    TABLES
      t_outtab                          = gt_zemp[]    .
ENDFORM.                    "show_data

第二步:ALV 关联到自定义工具栏

ALV 内置的工具栏没有与保存相关的按钮,解决办法是拷贝一个工具栏,让 ALV 控件关联拷贝的工具栏。使用事务码 SE41 从程序 SAPLKKBL 拷贝 STANDARD 工具栏到本程序。

在程序中添加一个子例程,设置 GUI STATUS:

FORM set_pf_status USING rt_extab TYPE slis_t_extab.
  SET PF-STATUS 'ZSTANDARD'.
ENDFORM.

添加另外一个子例程,用于将界面上的变更数据保存到数据表,代码待会再编写。

FORM user_command USING r_ucomm LIKE sy-ucomm
                        rs_selfield TYPE slis_selfield. 
 
ENDFORM.

将 ALV 关联到 ZSTANDARD GUI Status:

CALL FUNCTION 'REUSE_ALV_GRID_DISPLAY'
  EXPORTING
    i_callback_program       = sy-repid
    i_callback_pf_status_set = 'SET_PF_STATUS'
    i_callback_user_command  = 'USER_COMMAND'
    it_fieldcat              = gt_fieldcat[]
  TABLES
    t_outtab                 = gt_zemp[].

在 ZSTANDARD 工具栏上新建一个 SAVE 按钮,Function code 为 SAVE, Function text 为 Save。

第三步,编写代码处理数据从 ALV 到内表,内表到数据表:

FORM user_command USING r_ucomm LIKE sy-ucomm
                        rs_selfield TYPE slis_selfield.
 
  DATA: r_ref_grid TYPE REF TO cl_gui_alv_grid.
 
  IF r_ucomm = 'SAVE'.
 
    " 更新的数据保存到内表
    IF r_ref_grid IS INITIAL.
      CALL FUNCTION 'GET_GLOBALS_FROM_SLVC_FULLSCR'
        IMPORTING
          e_grid = r_ref_grid.
    ENDIF.
 
    IF NOT r_ref_grid IS INITIAL.
      CALL METHOD r_ref_grid->check_changed_data.
    ENDIF.
 
    rs_selfield-refresh = 'X'.
 
    " 更新的数据从内表到数据表
    UPDATE zemployee FROM TABLE gt_zemp.
 
    MESSAGE 'Data was saved successfully.' TYPE 'S'.
  ENDIF. 
ENDFORM.

Table Control

使用表格控件 (table control) 对数据进行 CRUD 操作,界面的友好性好于 ALV 控件。本文介绍向导方式的使用方法。

第一步,创建一个编号为 2000 的 Screen,创建一个名为 STANDARD 的 GUI Status, 在 Function Keys 部分,设置 4 个按钮如下图所示:

为了程序能正常退出,激活 Screen 2000 默认的 PBO 和 PAI 事件,并编写如下代码:

REPORT  z_table_crud.

DATA: ok_code LIKE sy-ucomm,
      save_ok LIKE ok_code.

START-OF-SELECTION.
  CALL SCREEN 2000.

MODULE status_2000 OUTPUT.
  SET PF-STATUS 'STANDARD'.
ENDMODULE.                    "STATUS_2000 OUTPUT

MODULE user_command_2000 INPUT.
  save_ok = ok_code.
  CLEAR ok_code.

  CASE save_ok.
    WHEN 'BACK'.
      LEAVE TO SCREEN 0.
    WHEN 'EXIT' OR 'CANCEL'.
      LEAVE PROGRAM.
  ENDCASE.
ENDMODULE.                    "USER_COMMAND_2000 INPUT

此时,程序能够运行并且退出,但程序显示的是一个空界面。

第二步,在程序中定义两个内表,与要维护的数据表 zemployee 结构一致,但为了方便 table control 的使用,特意增加一列 chk,后面会用到 chk:

DATA: gs_emp1 LIKE zemployee.
DATA: BEGIN OF gt_emp OCCURS 0,
        chk TYPE c.
        INCLUDE STRUCTURE gs_emp1.
DATA: END OF gt_emp.

DATA: gt_emp_save LIKE STANDARD TABLE OF zemployee,
      gs_emp      LIKE LINE OF gt_emp_save.

第三步,利用向导生成数据维护的界面:

将程序所有对象激活之后,进入 Screen 2000 的 Layout 界面。利用向导来生成 zemployee 表的数据维护界面。在向导的步骤中,说明几个要点。

指定 table control 的名称:

指定程序用到的内表为 gt_emp:


选择字段,不要包括 chk:


定义字段属性,按照下图定义:

  • Input control: 表示可以编辑
  • With column headers : 定义默认的字段头
  • Line selection column: 选 CHK,选择字段必须为 C,长度为 1

下一屏,按下图设置:


下一屏,使用向导默认的值:


向导帮我们生成了很多代码,激活后,程序可以运行,但数据表的数据不会加载到界面。

第四步:加载数据

我们只需要将数据加载到内表 gt_emp,数据即可以在界面中显示。定义一个子例程:

FORM get_data.
  SELECT * INTO CORRESPONDING FIELDS OF TABLE gt_emp
    FROM zemployee.
ENDFORM. 

在 start-of-selection 事件中加载:

START-OF-SELECTION.
  PERFORM get_data.
  CALL SCREEN 2000.

第五步:编写数据保存的代码

FORM save_data.
  LOOP AT gt_emp.
    CLEAR gs_emp.
    gs_emp-mandt = '001'.
    gs_emp-empid = gt_emp-empid.
    gs_emp-empname = gt_emp-empname.
    gs_emp-empaddr = gt_emp-empaddr.
    APPEND gs_emp TO gt_emp_save.
  ENDLOOP.

  MODIFY zemployee FROM TABLE gt_emp_save.

  MESSAGE 'Data was saved.' TYPE 'S'.
ENDFORM.      

点击 SAVE 按钮运行 save_data:

MODULE user_command_2000 INPUT.
  save_ok = ok_code.
  CLEAR ok_code.

  CASE save_ok.
    WHEN 'BACK'.
      LEAVE TO SCREEN 0.
    WHEN 'EXIT' OR 'CANCEL'.
      LEAVE PROGRAM.
    WHEN 'SAVE'.
      PERFORM save_data.
  ENDCASE.
ENDMODULE.   

ABAP 提供 Restful Service

利用 ABAP 提供 Restful Service,这样可以使用其它编程语言作为数据维护界面,灵活度较高。ABAP 提供 Restful Service 请参考我之前写的系列博文

参考

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