SAP ABAP收货或者货物移动(MIGO,MB11,MB1A)在保存时候的增强点

本文详细介绍了如何使用MB_DOCUMENT_BADI增强点在MIGO和MB11交易代码下,根据生成的凭证号更新自建表的数据,包括凭证类型、序列号、供应商、合同号等关键信息。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

增强的BADI是:MB_DOCUMENT_BADI,特别要注意的是假如你是想在点击保存按钮的时候根据生成的凭证号(即表示这个增强点处凭证号已经生成),把某些数据更新到你的自建表的话,要用第二个方法MB_DOCUMENT_UPDATE,调试可以在第一个方法MB_DOCUMENT_BEFORE_UPDATE里面调试,特别注意不能在方法MB_DOCUMENT_BEFORE_UPDATE里面写COMMIT WORK,因为两个方法的参数完全一样,所以可以在第一个方法那边调试,然后直接把所有的代码拷贝到第二个,要注意第二个方法打断点是无法进去的,但是他确实有执行,所以说只能在第一个方法那边调试,下面赋一段写的代码吧:


 method if_ex_mb_document_badi~mb_document_update.
  data: gs_mseg type  mseg.
  data: gt_gdfh type table of ztmm_gdfh.
  data: gs_gdfh type ztmm_gdfh.


  data: gt_gdth type table of ztmm_gdth.
  data: gs_gdth type ztmm_gdth.
  data: gt_lxjj type table of ztmm_lxjjgl.
  data: gs_lxjj type ztmm_lxjjgl.
  data: gs_mkpf type mkpf.
  if sy-tcode eq ‘MIGO’ or sy-tcode  eq ‘MB11′.
    read table xmkpf into gs_mkpf index 1.


    loop at xmseg into gs_mseg.


      if   gs_mseg-bwart eq ‘201′ and gs_mseg-sobkz eq ” .
        select single zgngo into gs_lxjj-zlxjjbm from marc where matnr eq gs_mseg-matnr and werks eq gs_mseg-werks.
        if sy-subrc eq 0.
          gs_lxjj-mblnr = gs_mseg-mblnr.
          gs_lxjj-matnr = gs_mseg-matnr.
          gs_lxjj-werks = gs_mseg-werks.
          gs_lxjj-bldat = gs_mkpf-bldat.
          append gs_lxjj to gt_lxjj.
        endif.
      endif.


      if gs_mseg-werks eq ‘SD00′ and ( ( gs_mseg-bwart eq ‘201′ and gs_mseg-sobkz eq ” )  or  ( gs_mseg-bwart eq ‘201′ and gs_mseg-sobkz eq ‘K’ )
      or ( gs_mseg-bwart eq ‘221′ and gs_mseg-sobkz eq ‘Q’ ) or ( gs_mseg-bwart eq ‘251′ and gs_mseg-sobkz eq ” )
     or ( gs_mseg-bwart eq ‘251′ and gs_mseg-sobkz eq ‘K’ ) or ( gs_mseg-bwart eq ‘221′ and gs_mseg-sobkz eq ‘K’ ) ).
*      BREAK ZZWANGLP.


        gs_gdfh-mblnr = gs_mseg-mblnr.
        gs_gdfh-mjahr = gs_mseg-mjahr.
        gs_gdfh-bldat = gs_mkpf-bldat.
        gs_gdfh-budat = gs_mkpf-budat.
        gs_gdfh-cpudt = gs_mkpf-cpudt.
        gs_gdfh-cputm = gs_mkpf-cputm.
        gs_gdfh-mblpo = gs_mseg-zeile.
        gs_gdfh-matnr = gs_mseg-matnr.
        gs_gdfh-menge = gs_mseg-menge.
        gs_gdfh-charg = gs_mseg-charg.
        gs_gdfh-lgort = gs_mseg-lgort.
        gs_gdfh-lgpla = gs_mseg-lgpla.
        gs_gdfh-dmbtr = gs_mseg-dmbtr.
        gs_gdfh-pspel   = gs_mseg-ps_psp_pnr.
        gs_gdfh-zdj   = gs_gdfh-dmbtr / gs_gdfh-menge.“单价=(本位币金额/发放数量)
**序列号:通过MBLNR&MJAHR& MBLPO在表SER03中找到OBKNR,再通过OBKNR在表OBJK找到SERNR作为序列号更新到表ZTMM_GDFH
*        SELECT SINGLE SERNR INTO GS_GDFH-SERNR FROM OBJK
*          JOIN SER03 ON SER03~OBKNR = OBJK~OBKNR
*        WHERE MBLNR EQ GS_MSEG-MBLNR AND MJAHR EQ GS_MSEG-MJAHR AND ZEILE EQ GS_MSEG-ZEILE.


*      供应商:通过CHARG在表MCHA找到LIFNR
        select single lifnr into gs_gdfh-lifnr from mcha where charg eq gs_gdfh-charg and matnr eq gs_gdfh-matnr and werks eq ‘SD00′.
*      合同号:通过CHARG在表EKBE中找到EBELN,再通过EBELN在表EKKO找到SUBMI。
        select single submi into gs_gdfh-submi from ekko
          join ekbe on ekbe~ebeln = ekko~ebeln
        where charg eq gs_gdfh-charg and matnr eq gs_gdfh-matnr and werks eq ‘SD00′.


*       采购订单及行号:通过CHARG在表EKBE中找到EBELN,再通过EBELN在表EKPO找到EKPO-EBELP。
        select single ekbe~ebeln ebelp into (gs_gdfh-ebeln,gs_gdfh-ebelp)
          from ekko
          join ekbe on ekbe~ebeln = ekko~ebeln
          where charg eq gs_gdfh-charg and matnr eq gs_gdfh-matnr and werks eq ‘SD00′.“EBELP EQ GS_MSEG-ZEILE.


*    依据预留编号及预留项目编号作为关键字段,在表ZTMM_GDXX找到工单、作业工种(工序)、工单行号字段
        select single zgd zgx zhxmbh into (gs_gdfh-zgd,gs_gdfh-zgx,gs_gdfh-zgdhh) from ztmm_gdxx where rsnum eq gs_mseg-rsnum and rspos eq gs_mseg-rspos.
        if sy-subrc ne 0.
          clear gs_gdfh.
        else.
          gs_gdfh-zyfh = ‘X’.
          append gs_gdfh to gt_gdfh.
        endif.


      endif.


      if gs_mseg-werks eq ‘SD00′ and ( ( gs_mseg-bwart eq ‘202′ and gs_mseg-sobkz eq ” ) or ( gs_mseg-bwart eq ‘202′ and gs_mseg-sobkz eq ‘K’ )  or  ( gs_mseg-bwart eq ‘222′ and gs_mseg-sobkz eq ‘Q’ )
          or ( gs_mseg-bwart eq ‘252′ and gs_mseg-sobkz eq ” ) or ( gs_mseg-bwart eq ‘222′ and gs_mseg-sobkz eq ‘K’ )
         or ( gs_mseg-bwart eq ‘252′ and gs_mseg-sobkz eq ‘K’ ) ).“OR ( GS_MSEG-BWART EQ ’242′ AND GS_MSEG-SOBKZ EQ ” )dete by libin 20120516 bc songwenbin
*      BREAK ZZWANGLP.


        gs_gdth-mblnr = gs_mseg-mblnr.
        gs_gdth-mjahr = gs_mseg-mjahr.
        gs_gdth-bldat = gs_mkpf-bldat.
        gs_gdth-budat = gs_mkpf-budat.
        gs_gdth-cpudt = gs_mkpf-cpudt.
        gs_gdth-cputm = gs_mkpf-cputm.
        gs_gdth-mblpo = gs_mseg-zeile.
        gs_gdth-matnr = gs_mseg-matnr.
        gs_gdth-menge = gs_mseg-menge.
        gs_gdth-charg = gs_mseg-charg.
        gs_gdth-lgort = gs_mseg-lgort.
        gs_gdth-lgpla = gs_mseg-lgpla.
        gs_gdth-dmbtr = gs_mseg-dmbtr.
        gs_gdth-pspel   = gs_mseg-ps_psp_pnr.
        gs_gdth-zdj   = gs_gdth-dmbtr / gs_gdth-menge.“单价=(本位币金额/发放数量)
**序列号:通过MBLNR&MJAHR& MBLPO在表SER03中找到OBKNR,再通过OBKNR在表OBJK找到SERNR作为序列号更新到表ZTMM_GDFH
*        SELECT SINGLE SERNR INTO GS_GDFH-SERNR FROM OBJK
*          JOIN SER03 ON SER03~OBKNR = OBJK~OBKNR
*        WHERE MBLNR EQ GS_MSEG-MBLNR AND MJAHR EQ GS_MSEG-MJAHR AND ZEILE EQ GS_MSEG-ZEILE.


*    依据预留编号及预留项目编号作为关键字段,在表ZTMM_GDXX找到工单、作业工种(工序)、工单行号字段
        select single zgd zgx zhxmbh into (gs_gdth-zgd,gs_gdth-zgx,gs_gdth-zgdhh) from ztmm_gdxx where rsnum eq gs_mseg-rsnum and rspos eq gs_mseg-rspos.
        if sy-subrc ne 0.
          clear gs_gdfh.
        else.
          gs_gdth-zyfh = ‘X’.
          append gs_gdth to gt_gdth.
        endif.
      endif.
    endloop.
    if gt_gdfh is not initial.
      modify ztmm_gdfh from table gt_gdfh.
    endif.
    if gt_gdth is not initial.
      modify ztmm_gdth from table gt_gdth.
    endif.
    if gt_lxjj is not initial.
      modify ztmm_lxjjgl from table gt_lxjj.
    endif.


  endif.
endmethod.
<think>好的,我需要解决用户在SAP中使用MB1A事务代码时,当检查不符合条件时无法保存的问题。用户希望找到实现禁止保存的方法或配置解决方案。首先,我需要理解MB1A是什么,以及常见的相关问题。 根据提供的引用,特别是引用[2],里面提到了SAP MM部分的常用增强,包括ENPT-LMIGOKD1-MIGO公司间采购收货时序列号校验ENPT-LMIGOKG1-MIGO交货单流水号。这可能涉及到MB1A的增强MB1A通常用于物料移动,比如发货或转移过账,如果在保存时遇到问题,可能涉及到字段校验或业务逻辑检查。 用户提到的问题是当不符合条件时无法保存,但用户希望的是不符合条件时禁止保存。这可能意味着当前系统在不符合条件时允许保存,而用户希望加强控制,阻止这种情况。通常,SAP的标准功能可能没有完全覆盖所有业务校验,因此可能需要通过增强来实现自定义的校验逻辑。 引用[2]中提到的增强项目如ENPT-LKACBF03-ZDQ_MM_RKPF_CHK,这是在创建预留时检查成本中心是否符合规则。类似地,可能在MB1A中使用类似的增强,例如在保存前触发检查逻辑。需要找到MB1A的增强,比如在保存前触发的事件,例如在过账前进行校验。 另外,引用[3]关于委外工序的配置可能涉及物料移动的检查,但可能不直接相关。引用[1]提到安全库存的问题,但用户的问题更多是条件检查,所以可能更关注增强配置。 解决方案通常包括两个方向:配置增强开发。首先检查是否可以通过现有配置实现,比如通过物料主数据中的设置或移动类型的配置。例如,某些移动类型可能需要特定字段,如成本中心、订单号等,如果未输入,则系统阻止保存。但若标准配置无法满足复杂的条件检查,则需要使用增强。 常见的增强方法包括使用用户出口(User Exit)、BADI(Business Add-In)或增强(Enhancement Spot)。例如,MB1A过账时可能触发MB_DOCUMENT_BADI这个BADI,在保存前执行自定义校验。如果检查到不符合条件,则抛出错误消息阻止保存。 具体步骤可能包括:1. 确定校验逻辑的条件,例如特定字段是否填写,或字段值是否符合规则。2. 查找MB1A相关的增强,比如在过账前的检查。3. 编写增强程序,在保存前检查条件,若不符合则显示错误消息并终止过账。 例如,使用BADI MB_DOCUMENT_BADI的IF_EX_MB_DOCUMENT_BADI~CHECK方法,在此方法中添加自定义逻辑。如果发现不符合条件,调用MESSAGE E...类型为E的错误消息,阻止保存。 此外,引用[2]中的ENPT-LMIGOKD1-MIGO可能是一个参考案例,说明如何在MIGO事务中实施类似校验,可以借鉴其方法应用到MB1A中。同时,确保增强程序激活并正确分配,避免影响其他事务。 最后,测试是关键,需要模拟不符合条件的情况,确保系统正确阻止保存,并给出明确的错误提示。同时,考虑权限配置是否会影响增强的执行,确保解决方案的稳定性兼容性。</think>要解决SAPMB1A事务代码在不符合条件时禁止保存的问题,可通过以下步骤实现: ### 一、问题分析 MB1A用于物料移动过账(如发货、转移)。当系统未配置强制校验或业务规则不满足时,可能允许用户保存不符合条件的数据。需通过增强实现自定义校验逻辑。 --- ### 二、解决方案 #### 方法1:通过BADI增强实现校验 **步骤1:确定增强** 使用BADI `MB_DOCUMENT_BADI`,其方法`CHECK`会在过账前触发[^2]。 **步骤2:编写增强逻辑** ```abap METHOD if_ex_mb_document_badi~check. DATA: lv_error_flag TYPE abap_bool VALUE abap_false. "示例:检查成本中心是否以特定前缀开头 LOOP AT im_mseg ASSIGNING FIELD-SYMBOL(<fs_mseg>) WHERE kostl IS NOT INITIAL. IF <fs_mseg>-kostl(2) NE 'CC'. "假设要求成本中心以'CC'开头 lv_error_flag = abap_true. EXIT. ENDIF. ENDLOOP. IF lv_error_flag = abap_true. MESSAGE e888(sabapdocu) WITH '成本中心不符合规则'. "自定义错误消息 ENDIF. ENDMETHOD. ``` #### 方法2:配置移动类型字段必输 1. **事务代码:OMJJ** 进入移动类型配置,选择对应移动类型(如261)。 2. **设置字段状态组** 在字段选择中,将关键字段(如成本中心、订单号)设为必输项。 --- ### 三、关键配置与增强位置 | 类型 | 对象 | 作用 | 参考文档 | |------------|---------------------|---------------------------------------|----------------| | BADI增强 | MB_DOCUMENT_BADI | 过账前自定义逻辑校验 | [^4] | | 配置 | 移动类型(OMJJ) | 强制字段必输 | [^1][^3] | --- ### 四、测试与验证 1. **测试用例** - 场景1:输入不符合条件的成本中心(如`AA1000`),系统应阻止保存。 - 场景2:必输字段留空,系统提示错误。 2. **错误处理** 增强中应使用`MESSAGE E`类型确保事务终止,并明确提示原因。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值