SAP-ABAP:变量、常量、结构与内表声明(10篇博客合集) 第三篇:STRUCT结构声明全指南:嵌套结构与深层结构的差异与选型
变量、常量、结构与内表声明10篇博客合集第三篇STRUCT结构声明全指南嵌套结构与深层结构的差异与选型在ABAP开发中结构体Structure是将多个字段组合成一个逻辑单元的核心手段。从简单的地址组合到复杂的订单抬头-行项目嵌套结构体的合理设计直接影响代码的可读性和数据处理的复杂度。本文将系统讲解从扁平结构到多层嵌套的声明方法对比TYPES与DATA定义结构的适用场景并分享结构组件的继承、重命名技巧帮助你在复杂业务场景中游刃有余。一、什么是结构体结构体Structure是将多个不同类型的字段组件聚合为一个复合数据对象。它本身不直接对应数据库表而是作为程序内部的数据模板或工作区。典型应用场景作为内表的行类型。作为数据库表的工作区INTO CORRESPONDING FIELDS OF。在函数或方法之间传递一组相关数据。二、结构体的声明方式TYPESvsDATA2.1 使用TYPES定义结构类型抽象模板TYPES定义的是数据类型不占用内存。可以多次基于它创建数据对象。TYPES: BEGIN OF ty_address, street TYPE c LENGTH 30, city TYPE c LENGTH 20, zip TYPE c LENGTH 10, END OF ty_address. DATA: ls_addr1 TYPE ty_address, ls_addr2 TYPE ty_address.适用场景需要在多个地方重复使用相同的结构模板。2.2 使用DATA直接定义结构对象实例DATA定义的是数据对象立即分配内存且只能在此处使用。DATA: BEGIN OF ls_order, order_id TYPE vbeln, order_date TYPE erdat, END OF ls_order.适用场景一次性使用、局部临时结构无需复用类型。2.3 对比总结特性TYPES定义结构类型DATA直接定义结构本质抽象数据类型具体数据对象实例内存占用不占内存立即分配内存重用性可基于它声明多个变量无法重用需重复编写适用范围全局或局部类型局部变量可读性更清晰推荐作为公共类型适合简单一次性结构最佳实践优先使用TYPES定义结构类型即使只在一个地方使用也能保持代码一致性和扩展性。三、扁平结构 vs 嵌套结构3.1 扁平结构Flat Structure所有组件都是基础类型或简单类型没有嵌套其他结构或内表。TYPES: BEGIN OF ty_person, name TYPE c LENGTH 30, age TYPE i, weight TYPE p LENGTH 5 DECIMALS 2, END OF ty_person.特点内存布局紧凑访问速度快赋值操作简单。3.2 嵌套结构Nested Structure结构体的某个组件本身又是一个结构体或其他复杂类型。TYPES: BEGIN OF ty_address, street TYPE c LENGTH 30, city TYPE c LENGTH 20, END OF ty_address. TYPES: BEGIN OF ty_employee, emp_id TYPE i, address TYPE ty_address, 嵌套另一个结构 END OF ty_employee.访问嵌套组件DATA ls_emp TYPE ty_employee. ls_emp-address-city Shanghai.特点适合表达层次化数据如订单抬头-行项目但赋值和比较需要逐层处理。3.3 深层嵌套与性能考虑嵌套层数过深例如 5 层以上会导致代码冗长ls_a-b-c-d-e-f且维护困难。建议将深层嵌套拆分为多个独立结构通过组合而非层层包裹。使用INCLUDE STRUCTURE实现结构复用减少嵌套深度见第五部分。四、多层嵌套的声明方法与访问技巧4.1 直接内嵌结构TYPES: BEGIN OF ty_order_item, item_no TYPE i, material TYPE matnr, quantity TYPE menge_d, END OF ty_order_item. TYPES: BEGIN OF ty_order_header, order_no TYPE vbeln, order_date TYPE erdat, items TYPE STANDARD TABLE OF ty_order_item WITH DEFAULT KEY, 内表嵌套 END OF ty_order_header.此时items是内表不是结构。若需要嵌套结构而非内表去掉TABLE OF即可。4.2 使用INCLUDE STRUCTURE实现扁平化嵌套INCLUDE STRUCTURE可以将另一个结构的所有组件直接“拷贝”到当前结构中从而避免多层点号访问。TYPES: BEGIN OF ty_address, street TYPE c LENGTH 30, city TYPE c LENGTH 20, END OF ty_address. TYPES: BEGIN OF ty_customer, cust_id TYPE i, INCLUDE STRUCTURE ty_address. 将 address 的字段扁平展开 END OF ty_customer.访问方式ls_cust-city而不是ls_cust-address-city。注意INCLUDE STRUCTURE在 7.40 之后推荐使用INCLUDE TYPE或INCLUDE STRUCTURE。如果被包含的结构有修改当前结构自动同步需重新激活。4.3 结构组件重命名使用RENAME或定义新类型时指定新名称。TYPES: BEGIN OF ty_person, name TYPE c LENGTH 30, END OF ty_person. TYPES: BEGIN OF ty_employee, emp_name TYPE c LENGTH 30, 重命名但类型一致 END OF ty_employee.更好的方式是使用INCLUDE配合RENAMING WITH SUFFIXTYPES: BEGIN OF ty_address, street TYPE c LENGTH 30, city TYPE c LENGTH 20, END OF ty_address. TYPES: BEGIN OF ty_company, name TYPE c LENGTH 50, INCLUDE STRUCTURE ty_address RENAMING WITH SUFFIX _addr, END OF ty_company.访问ls_company-street_addr、ls_company-city_addr避免字段名冲突。五、结构组件的继承与扩展技巧5.1 扩展已有结构子结构通过INCLUDE基础结构再添加新字段。TYPES: BEGIN OF ty_base, id TYPE i, name TYPE c LENGTH 30, END OF ty_base. TYPES: BEGIN OF ty_extended, INCLUDE STRUCTURE ty_base, extra_field TYPE string, END OF ty_extended.5.2 使用APPEND STRUCTURE用于数据库表增强在数据字典中可以为表定义追加结构APPEND STRUCTURE但在程序层面直接使用INCLUDE模拟即可。六、结构体的赋值与比较6.1 逐字段赋值 vs 整体赋值ls_person-name 张三. ls_person-age 28. 整体赋值结构必须兼容 ls_person2 ls_person. 深拷贝所有字段复制6.2 使用MOVE-CORRESPONDING或CORRESPONDING对于字段名不完全相同的结构可以使用对应赋值。DATA: ls_source TYPE ty_source, ls_target TYPE ty_target. ls_target CORRESPONDING #( ls_source ). 只赋值名称相同的字段6.3 结构体比较IF ls_person1 ls_person2. 逐字段比较所有字段必须相等注意包含内表的结构体比较会进行深层比较内表内容逐一比较性能消耗大。七、结构体声明的最佳实践与误区7.1 最佳实践为所有结构类型命名使用TY_或TY_模块_前缀如TY_MM_ORDER_HEADER。扁平化优于深层嵌套避免超过 3 层的点号访问。优先使用TYPES声明类型即使只用一个实例。利用INCLUDE实现复用减少重复定义。7.2 常见误区误区1在同一个程序中对同一结构既用TYPES又用DATA重复定义。误区2忽视INCLUDE字段名冲突。不同结构包含相同字段名会导致激活错误需用RENAMING。误区3在嵌套结构中频繁使用MOVE-CORRESPONDING可能意外覆盖不需要的字段。误区4将超大结构体作为函数/方法的参数传递导致栈内存复制开销大。改用传值VALUE或传引用REF TO。八、完整示例订单抬头行项目结构的声明与使用 1. 定义行项目结构 TYPES: BEGIN OF ty_item, posnr TYPE posnr, 行项目号 matnr TYPE matnr, 物料号 menge TYPE menge_d, 数量 netwr TYPE netwr, 净价 END OF ty_item. 2. 定义订单抬头结构包含行项目内表 TYPES: BEGIN OF ty_order, vbeln TYPE vbeln, 订单号 erdat TYPE erdat, 创建日期 ernam TYPE ernam, 创建人 items TYPE STANDARD TABLE OF ty_item WITH DEFAULT KEY, END OF ty_order. 3. 声明变量并填充 DATA: ls_order TYPE ty_order, ls_item TYPE ty_item. ls_order-vbeln 4500000001. ls_order-erdat sy-datum. ls_item-posnr 0001. ls_item-matnr MAT-100. ls_item-menge 10. ls_item-netwr 99.99. APPEND ls_item TO ls_order-items. 4. 输出行项目 LOOP AT ls_order-items INTO ls_item. WRITE: / ls_item-posnr, ls_item-matnr, ls_item-menge. ENDLOOP.九、总结特性扁平结构嵌套结构含INCLUDE复杂度低中高访问路径-单级-多级或扁平的INCLUDE适用场景简单数据聚合层次化业务对象赋值性能快较慢需递归代码可读性高中等需规范命名掌握结构体的声明技巧能够让你在处理复杂业务数据时保持代码清晰、高效。下一篇将进入内表声明的核心方式对比STANDARD、SORTED、HASHED表的性能差异与选型。下篇预告内表声明的3种核心方式STANDARD/SORTED/HASHED表性能对比作者你的ABAP学习伙伴版本记录2026年5月 你曾因深层嵌套导致代码难以阅读吗有没有更好的设计模式欢迎留言分享。