ABAP非Unicode系统中字符串拼接(CONCATENATE)时吃字符问题
2024-10-06 07:16:41
系统是老R3,非Unicdoe系统,某些表字段是从外界系统过来的,由于接口设计的固定长度,外界系统传超长字符串过来后,就可能从最后一个中文字符中间截断,这问题到还没什么,只不过显示时最后一个字符显示成乱码而已,但是,如果将这些表字段捞出来与其它分隔符(如竖线) CONCATENATE时,可能会将这个分隔符吃掉,导致这些数据抛到对方系统后,无法再分隔,还原成一个个字段
如下面运行过程中:c3本身是一个 司 字后面跟半个中文字符,当使用 竖线分隔符与0字符CONCATENATE后,发现竖线没有了,它与前面的?问号(注:不是真正的问号,而是由于编码B5在GBK字符集里找不到,ABAP编辑器以问号显示而已)合成 祙 字了,即竖线被前面的半个汉字编码给吃掉了:
下面是经过frm_trunct_last 方法将半个中文字符截断后,再CONCATENATE拼接时,分隔符竖线就不会被吃掉了:
DATA: str TYPE string.
DATA: x3(3) TYPE x.
DATA: c3(3) TYPE c.
FIELD-SYMBOLS:<c3> TYPE x.
START-OF-SELECTION.
ASSIGN c3 TO <c3> CASTING.
"CBBE为“司”的编码,B5为前半个汉字编码,如“祙、单”等字的前半个就是B5
<c3> = 'CBBEB5'."模拟半个中文
WRITE:/ c3.
str = c3.
CONCATENATE str `0` INTO str SEPARATED BY `|`.
WRITE:/ str.
str = c3.
PERFORM frm_trunct_last USING str .
CONCATENATE str `0` INTO str SEPARATED BY `|`.
WRITE:/ str.
*&---------------------------------------------------------------------*
*& Form frm_trunct_last
*&---------------------------------------------------------------------*
* 如果字符串最后一个是半个字符,则截掉
*----------------------------------------------------------------------*
* -->STR 如果不会吃字符,则返回原字符串
*----------------------------------------------------------------------*
FORM frm_trunct_last USING str TYPE string .
DATA: xstr TYPE xstring.
DATA: l_codepage(4) TYPE n .
DATA: l_encoding(20).
DATA: truc_str TYPE string.
DATA: off_index TYPE i.
DATA: trunct_str TYPE string.
DATA: last_char TYPE string.
DATA: last_char_code TYPE xstring.
DATA: len TYPE i.
DATA: x7f TYPE x VALUE '7F'.
DATA: convout TYPE REF TO cl_abap_conv_out_ce.
DATA: convin TYPE REF TO cl_abap_conv_in_ce.
CALL FUNCTION 'SCP_CODEPAGE_BY_EXTERNAL_NAME'
EXPORTING
external_name = 'UTF-16BE'
IMPORTING
sap_codepage = l_codepage.
l_encoding = l_codepage.
convout = cl_abap_conv_out_ce=>create( encoding = l_encoding ).
convout->write( data = str ).
xstr = convout->get_buffer( ).
truc_str = xstr.
len = STRLEN( truc_str ).
len = len / 4 .
off_index = 4 * ( len - 1 ).
xstr = truc_str+0(off_index).
convin = cl_abap_conv_in_ce=>create( encoding = l_encoding input = xstr ).
CALL METHOD convin->read
IMPORTING
data = trunct_str.
off_index = 4 * ( len - 1 ).
xstr = truc_str+off_index(4).
convin = cl_abap_conv_in_ce=>create( encoding = l_encoding input = xstr ).
CALL METHOD convin->read
IMPORTING
data = last_char.
CALL FUNCTION 'SCP_CODEPAGE_BY_EXTERNAL_NAME'
EXPORTING
external_name = 'GBK'
IMPORTING
sap_codepage = l_codepage.
l_encoding = l_codepage.
convout = cl_abap_conv_out_ce=>create( encoding = l_encoding ).
convout->write( data = last_char ).
last_char_code = convout->get_buffer( ).
len = XSTRLEN( last_char_code ).
IF len = 1 AND last_char_code > x7f.
str = trunct_str.
ENDIF.
ENDFORM. "frm_trunct_last
注:分隔符是否被吃掉了,在屏幕输出上是看不出来的,如下面是输出结果:
但从上面的调试过程可以看出是被吃掉了,经测试传到其他系统后,也是会被吃掉的
最新文章
- sql语句查询经纬度范围(转载,源链接失效)
- CentOS 6.5下Redis安装记录
- play 源码分析
- CSS hack方式一览【转】
- 修改Esxi克隆的CentOS的IP地址
- Javascript高级程序设计——面向对象之理解对象
- [翻译] java NIO Channel
- android绘画折线图二
- 黑客群体的露面说明互联网公司开始回馈IT行业了,
- 微信公众平台开发localStorage数据总是被清空
- Android:解决client从server上获取数据乱码的方法
- 【IOS】 遍历info 所有内容 &;amp;&;amp; 唯一的节能设备UUID
- Memory and Scores
- python 写入数据
- 这可能是史上最全的css布局教程
- sql语言 含有包含关系的查询 (含mysql 和sql sever)
- python调用虹软2.0(全网首发)-更新中
- jenkins配置ssh
- apache-jmeter-3.1的简单压力测试使用方法(下载和安装)
- ubuntu 显示隐藏文件
热门文章
- sparkStreaming消费kafka-1.0.1方式:direct方式(存储offset到zookeeper)-- 2
- tasksetCPU亲和力&;docke容器资源限制
- 如何确定系统上的CPU插槽数量
- gitlab之六: gitlab 备份恢复
- 删除ELK中的数据。。
- oracle数据入库出现空格问题
- LeetCode 234. 回文链表
- radio按钮单选效果
- SpringMVC(三十) 实例:SpringMVC_RESTRUL_CRUD_显示所有员工信息
- Python 面向对象的补充