我有一个设置,其中有 JMS 生产者和 JMS 接收器。发送者应用程序发送如下消息:
source text ⟨е, ё, и, ю, я⟩ abcdefg
JMS 接收器收到消息后,使用纯 IBM MQ API 类将其放入 IBM MQ 队列。
将此消息发送到 MQ 时,我收到以下异常:
INFO | 2020-09-17 09:45:19 | [main] mimq.MQReceiver (MQReceiver.java:211) - IO Exception Occurred: Input length = 1
java.nio.charset.UnmappableCharacterException: Input length = 1
at java.nio.charset.CoderResult.throwException(CoderResult.java:282)
at java.nio.charset.CharsetEncoder.encode(CharsetEncoder.java:816)
at com.ibm.mq.jmqi.system.JmqiCodepage.stringToBytes(JmqiCodepage.java:923)
at com.ibm.mq.MQMessage.writeString(MQMessage.java:2848)
at com.ibm.mimq.MQReceiver.sendToAnotherQueue(MQReceiver.java:192)
at com.ibm.mimq.MQReceiver.main(MQReceiver.java:113)
下面是我的MQ PUT code :
public static void sendToLocalQueue(String msg) {
int port = 1414;
String host = "some-host";
String channel = "some-channel";
String manager = "some-QM";
String user = "user";
String passwd = "passwd";
String qname = "TEST";
String qmname = "some-QM";
MQQueueManager qMgr;
MQQueue inputQ;
try {
Hashtable<String, String> h = new Hashtable<String, String>();
h.put(MQC.TRANSPORT_PROPERTY, MQC.TRANSPORT_MQSERIES_CLIENT);
MQEnvironment.properties = h;
MQEnvironment.hostname = host;
MQEnvironment.port = port;
MQEnvironment.channel = channel;
MQEnvironment.userID = user;
MQEnvironment.password = passwd;
MQEnvironment.disableTracing();
MQException.log = null;
qMgr = new MQQueueManager(manager);
MQMessage m = new MQMessage();
m.applicationOriginData = "AMPS";
m.messageType = MQC.MQMT_DATAGRAM;
m.format = MQC.MQFMT_STRING;
m.encoding = MQC.MQENC_NATIVE;
m.priority = 4;
m.persistence = MQC.MQPER_PERSISTENT;
m.characterSet = MQC.MQCCSI_Q_MGR;
//m.characterSet = 1208;
m.expiry = MQC.MQEI_UNLIMITED;
m.writeString(msg);
MQPutMessageOptions putOptions = new MQPutMessageOptions();
putOptions.options = MQC.MQPMO_SYNCPOINT | MQC.MQPMO_FAIL_IF_QUIESCING;
logger.info("Putting message to LAN MQ (TEST queue)....");
qMgr.put(qname, qmname, m, putOptions);
qMgr.commit();
} catch(MQException me) {
logger.info("Error Code : " +me.getErrorCode());
logger.info("LocalizedMessage : " +me.getLocalizedMessage());
logger.info("Message : " +me.getMessage());
logger.info("Reason : " +me.getReason());
me.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
logger.info("IO Exception Occurred : " +e.getLocalizedMessage());
e.printStackTrace();
}
}
由于不可映射字符,无法将消息放入队列。编码在队列管理器级别设置为 UTF-8。
但是,当我替换以下行时:m.characterSet = MQC.MQCCSI_Q_MGR;
与行:m.characterSet = 1208;
问题已经不存在了。
我的问题是为什么这种转换没有在 MQ 级别完成。我需要检查哪些设置以确保正确转换。我尝试过以下技术,但不起作用:
Setting java parameter as : -Dfile.encoding=UTF-8
My 环境 :
Server : Linux
MQ : 9.0 or 7.5
Java : 1.8
还要提一下,迁移后相同的设置适用于 7.5,但不适用于 MQ 9.0。我知道通过上面的一行代码更改我可以传递消息。但我想了解 MQ 级别是否遗漏了一些配置。任何建议将不胜感激。
谢谢。
UPDATE
我将消息发送到 MQ 的客户端计算机具有 CCSID :MQMD.CodedCharSetId = 1208
我连接并发送消息的 MQ 服务器具有以下内容:
getDefaultProperty(Object) returns [819(0x333)] Integer
setCCSID(int) setter [819(0x333)]
因此,当我在代码中显式设置 1208 时,它就可以工作。如果不是,转换就会失败。
UPDATE-2
价值MQC.MQCCSI_Q_MGR is Zero
正如我在罐子里看到的那样。因此,代码是这样设计的,如果该值为零,它将从 Jar 中获取默认值,该值设置为819。当我打开 MQ 跟踪时我了解到了这一点。代码是这样的:
getDefaultProperty(Object) returns [819(0x333)] Integer
setCCSID(int) setter [819(0x333)]
此代码存在于 jar 内。因此,我们需要显式设置消息的字符集值。就我而言,它是 1208。