创build与SmsMessage.createFromPdu()(GSM 3gpp)一起使用的Android的PDU

目标:(注意:所选的答案会生成CDMA(3gpp2)的GSM(3gpp)PDU )请参阅此处

创build一个可以传入SmsMessage.createFromPdu(byte[] pdu) 。 我是“广播一个意图”我的BroadcastReciever收听,听SMS消息。

一个BroadcastReciever

使用android.provider.Telephony.SMS_RECEIVED “真正的”短信

对这些新的“应用程序SMS”使用自定义intent-filter操作。

 @Override public void onReceive(Context context, Intent intent) { Bundle bundle = intent.getExtras(); if (bundle != null) { Object[] pdusObj = (Object[]) bundle.get("pdus"); SmsMessage[] messages = new SmsMessage[pdusObj.length]; // getting SMS information from Pdu. for (int i = 0; i < pdusObj.length; i++) { messages[i] = SmsMessage.createFromPdu((byte[]) pdusObj[i]); } for (SmsMessage currentMessage : messages) { //the currentMessage.getDisplayOriginatingAddress() //or .getDisplayMessageBody() is null if I Broadcast a fake sms Log.i("BB", "address:"+currentMessage.getDisplayOriginatingAddress()+" message:"+currentMessage.getDisplayMessageBody()); ... 

所以我希望我的BroadcastReciever能够处理这两种types的消息,而无需添加额外的代码

(是的,我知道我可以为不同的intent-filter行动有一个不同的BroadcastReciever intent-filter但我想实际上把它取消,因为我知道它可以做到,我固执)

研究:

我一整天都在做研究。 尽pipe我对math和转换非常糟糕,并且创build了一个合适的algorithm,但我已经尝试了写自己的作品。 我已经查看了PDU上的Stack主题 ,并创build了PDU Android,但链接在答案中被打破。 我甚至看了com.google.android.mms.pdu源代码

到目前为止,我只能使用http://www.wrankl.de/JavaPC/SMSTools.html中的一些代码创build一个没有“ 始发地址 ”的PDU

PDU:

目的地:555消息:helloworld

 "1100038155f50000aa0ae8329bfdbebfe56c32" 

这显然是无效的…

附注:

除了本地使用外,我不打算对PDU做任何事情,我不想在我的代码中使用硬编码的PDU,因为我没有重用PDU。

如果有什么我可以添加到我用来添加一个“始发地址”的代码 ,这将工作。 或者没有人有我不知道的图书馆的信息?

谢谢

更新:

试着

 byte[] by =(byte[])(SmsMessage.getSubmitPdu("12345", "1234", "hello", false).encodedMessage); 

这给了我以下(hex表示)

 "0000100200000000000000000000000004010203040000000e000320ec400107102e8cbb366f00" 

没有工作

也许这个片段没有像你想要的许多细节领域,但为了我的简单目的,它可以像另一个短信一样调用通知。

  private static void createFakeSms(Context context, String sender, String body) { byte[] pdu = null; byte[] scBytes = PhoneNumberUtils .networkPortionToCalledPartyBCD("0000000000"); byte[] senderBytes = PhoneNumberUtils .networkPortionToCalledPartyBCD(sender); int lsmcs = scBytes.length; byte[] dateBytes = new byte[7]; Calendar calendar = new GregorianCalendar(); dateBytes[0] = reverseByte((byte) (calendar.get(Calendar.YEAR))); dateBytes[1] = reverseByte((byte) (calendar.get(Calendar.MONTH) + 1)); dateBytes[2] = reverseByte((byte) (calendar.get(Calendar.DAY_OF_MONTH))); dateBytes[3] = reverseByte((byte) (calendar.get(Calendar.HOUR_OF_DAY))); dateBytes[4] = reverseByte((byte) (calendar.get(Calendar.MINUTE))); dateBytes[5] = reverseByte((byte) (calendar.get(Calendar.SECOND))); dateBytes[6] = reverseByte((byte) ((calendar.get(Calendar.ZONE_OFFSET) + calendar .get(Calendar.DST_OFFSET)) / (60 * 1000 * 15))); try { ByteArrayOutputStream bo = new ByteArrayOutputStream(); bo.write(lsmcs); bo.write(scBytes); bo.write(0x04); bo.write((byte) sender.length()); bo.write(senderBytes); bo.write(0x00); bo.write(0x00); // encoding: 0 for default 7bit bo.write(dateBytes); try { String sReflectedClassName = "com.android.internal.telephony.GsmAlphabet"; Class cReflectedNFCExtras = Class.forName(sReflectedClassName); Method stringToGsm7BitPacked = cReflectedNFCExtras.getMethod( "stringToGsm7BitPacked", new Class[] { String.class }); stringToGsm7BitPacked.setAccessible(true); byte[] bodybytes = (byte[]) stringToGsm7BitPacked.invoke(null, body); bo.write(bodybytes); } catch (Exception e) { } pdu = bo.toByteArray(); } catch (IOException e) { } Intent intent = new Intent(); intent.setClassName("com.android.mms", "com.android.mms.transaction.SmsReceiverService"); intent.setAction("android.provider.Telephony.SMS_RECEIVED"); intent.putExtra("pdus", new Object[] { pdu }); intent.putExtra("format", "3gpp"); context.startService(intent); } private static byte reverseByte(byte b) { return (byte) ((b & 0xF0) >> 4 | (b & 0x0F) << 4); } 

希望你会find有用的东西

更新:

  public static final SmsMessage[] getMessagesFromIntent( Intent intent) { Object[] messages = (Object[]) intent.getSerializableExtra("pdus"); byte[][] pduObjs = new byte[messages.length][]; for (int i = 0; i < messages.length; i++) { pduObjs[i] = (byte[]) messages[i]; } byte[][] pdus = new byte[pduObjs.length][]; int pduCount = pdus.length; SmsMessage[] msgs = new SmsMessage[pduCount]; for (int i = 0; i < pduCount; i++) { pdus[i] = pduObjs[i]; msgs[i] = SmsMessage.createFromPdu(pdus[i]); } return msgs; } 

自从我做了任何PDU的直接纠缠以来,这是一个很长的时间,但是当我这么做的时候,我很快就放弃了,并使用了SMSLib :PDU工具,它非常适合通过诺基亚手机发送(通过串行链接)。 我的假设(这可能是错误的)是,他们也会为Android工作,假设界面实际上符合规范。

在console.c中检查这个代码 。 这就是android模拟器创buildpdu和RIL.java的地方,其中CMT消息被转换为SmsMessage 。 你可以使用SmsMessage.getPdu来获得pdu。 但SmsMessage.newFromCmt看起来是一个内部API。 所以它可能不可靠。

此外,这只是为GSM,CDMA有一个完全不同的代码stream,因为RIL.java和调制解调器是完全制造商特定的,这可能只适用于模拟器。

通常GSM代码在Android上更可靠,所以不妨在GSM手机上工作。 请尝试一下。