如何将S/4HANA系统存储的图片文件用Java程序保存到本地

我在S/4HANA的事务码MM02里为Material维护图片文件作为附件:

通过如下简单的ABAP代码即可将图片文件的二进制内容读取出来:

REPORT zgos_api.

DATA ls_appl_object     TYPE gos_s_obj.
DATA lo_gos_api         TYPE REF TO cl_gos_api.
DATA lt_attachment_list TYPE gos_t_atta.
DATA lt_role_filter     TYPE gos_t_rol.

DATA: lv_id  TYPE matnr VALUE '16',
      lt_att TYPE TABLE OF sibflporb.

CALL FUNCTION 'CONVERSION_EXIT_MATN1_INPUT'
  EXPORTING
    input  = lv_id
  IMPORTING
    output = lv_id.
ls_appl_object-typeid = 'BUS1001006'.
ls_appl_object-instid = lv_id.
ls_appl_object-catid  = 'BO'.

START-OF-SELECTION.

  TRY.
      lo_gos_api = cl_gos_api=>create_instance( ls_appl_object ).
      APPEND cl_gos_api=>c_attachment TO lt_role_filter.
      lt_attachment_list = lo_gos_api->get_atta_list( lt_role_filter ).
    CATCH cx_gos_api INTO DATA(error).
      WRITE:/ error->get_text( ).
      RETURN.
  ENDTRY.

  DATA: ls_key TYPE gos_s_attkey,
        lv_base64 type string.

  LOOP AT lt_attachment_list ASSIGNING FIELD-SYMBOL(<id>).
    ls_key-atta_id = <id>-atta_id.
    ls_key-atta_cat = <id>-atta_cat.
    DATA(ls) = lo_gos_api->get_al_item( ls_key ).
    BREAK-POINT.
  ENDLOOP.

然后我把这段代码封装到一个Function moduleZDIS_GET_MATERIAL_IMAGES里,在Java代码里消费这个function module,把S/4HANA返回的二进制内容存成本地图片文件。在ABAP系统里图片二进制文件内容的类型为RAWSTRING,如何在Java里使用它?

使用JCO在Java里消费ABAP function module的代码:

static private void getProductImageTest(){
    	JCoDestination destination = null;
		try {
			destination = JCoDestinationManager.getDestination(DESTINATION_NAME);
			JCoRepository repo = destination.getRepository();
    		JCoFunction stfcConnection = repo.getFunction("ZDIS_GET_MATERIAL_IMAGES");

    		JCoParameterList imports = stfcConnection.getImportParameterList();
        
    		String materialID = "16";

    		imports.setValue("IV_MATERIAL_ID", materialID);

    		stfcConnection.execute(destination);
        
    		JCoParameterList exports = stfcConnection.getExportParameterList();
    		
    	    int abapDuration = exports.getInt("EV_DURATION");
    	    
    	    StringBuilder sb = new StringBuilder();
    	    sb.append("{ \"" + ABAP_DURATION + "\": " + abapDuration + ",");
    	    
    	    sb.append("\"" + UPSELL_PRODUCT + "\":[");
    	    
    	    JCoTable codes = exports.getTable("ET_IMAGES");
    	    
    	    int row = codes.getNumRows();
    	    System.out.println("Total rows: " + row);
    	    
    	    System.out.println("ABAP duration: " + abapDuration);
    	    
    	    for( int i = 0; i < row; i++){
    	    	codes.setRow(i);
                sb.append("{\"" + FILE_ID + "\":" + codes.getString("FILEID") + ","
                		+ "\"" + FILE_OWNER + "\":\"" + codes.getString("OWNER") + "\"" + ",");
                sb.append("{\"" + FILE_CDATE + "\":" + codes.getString("CREATION_DATE") + ","
                		+ "\"" + FILE_NAME + "\":\"" + codes.getString("FILENAME") + "\""); 
                
                storeLocalFile(codes);
                if( i < row - 1){
                	sb.append("},");
                }
                else{
                	sb.append("}");
                }
    	    }
    	    sb.append("]}");
    	    
    	    System.out.println("Final json: " + sb.toString());
    	    
		} catch (JCoException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
    }

其中把ABAP的类型为RAWSTRING的字段保存成本地文件的代码封装于方法storeLocalFile里:

static private void storeLocalFile(JCoTable codes){
    	InputStream is = codes.getBinaryStream("FILECONTENT");
    	try {
    		File file = new File("c:\\temp\\" + codes.getString("FILENAME"));

    		byte[] bytes = new byte[is.available()];
    		is.read(bytes);
    		
	        OutputStream output = new FileOutputStream(file);

		BufferedOutputStream bufferedOutput = new BufferedOutputStream(output);

		bufferedOutput.write(bytes);
			
		bufferedOutput.close();
		is.close();
		} catch (IOException e) {
			e.printStackTrace();
		}
     }

执行Java代码后,在本地C盘temp目录下能够看到S/4HANA里Material ID为16的两个附件图片:

要获取更多Jerry的原创技术文章,请关注公众号"汪子熙"或者扫描下面二维码:

原创声明,本文系作者授权云+社区发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏谈补锅

复杂sql分组查询 ( pivot)

      一个数据表里面字段有年、月、日、金额、支付方式等字段,然后现在想写个sql语句,把每一天的每种支付方式金额(支付方式有多重)排在同一行,

1.2K3
来自专栏编程札记

Lucene构建个人搜索引擎解析

简单来说,Lucene提供了一套完整的工具来帮助开发者构建自己的搜索引擎,开发者只需要import Lucene对应的package即可快速地开发构建自己的业务...

2212
来自专栏搜云库

Spring Boot 中使用 Java API 调用 lucene

Lucene是apache软件基金会4 jakarta项目组的一个子项目,是一个开放源代码的全文检索引擎工具包,但它不是一个完整的全文检索引擎,而是一个全文检索...

4685
来自专栏C++

Windows核心编程:第3章 内核对象

662
来自专栏杨建荣的学习笔记

关于db_files和maxdatafiles的问题(r4笔记第31天)

昨天在做生产监控的时候发现有个库的表空间不够了,就发邮件给客户的dba去处理,但是得到的反馈是尝试添加的时候发现已经超过了数据文件的最大数限制。这个错误毫无疑问...

3586
来自专栏腾讯云流计算

Apache Calcite 功能简析及在 Flink 的应用

• Apache Calcite 是一个动态数据的管理框架,可以用来构建数据库系统的语法解析模块

1.5K7
来自专栏Python

Django---ORM操作大全

前言 Django框架功能齐全自带数据库操作功能,本文主要介绍Django的ORM框架 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞:...

1K10
来自专栏跟着阿笨一起玩NET

C#常用工具类——Excel操作类

1571
来自专栏牛肉圆粉不加葱

Spark SQL Limit 介绍及优化

全局限制,最多返回 limitExpr 对应条 records。总是通过 IntegerLiteral#unapply(limitExpr: Expressio...

4272
来自专栏技术碎碎念

windows API 开发飞机订票系统 图形化界面 (四)

接下来的是录入航班、修改航班信息功能的实现: 1 //录入航班 2 BOOL EntryFlight(HWND hEntryDlg){ 3 4 ...

3245

扫码关注云+社区

领取腾讯云代金券