创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手机上工作。 请尝试一下。
- Android 4.1:如何检查通知被禁用的应用程序?
- Android:使用SimpleCursorAdapter从数据库获取数据到ListView
- 我怎样才能从一个活动传递一个位图对象到另一个
- Dalvik VM进程是否释放系统RAM?
- Android – build立通知,TaskStackBuilder.addParentStack不能正常工作
- Android – 使用animation更改左边距
- 如何使用Android的相机或camera2 API来支持旧的和新的API版本,而不需要弃用笔记?
- Android的 – 如何dynamic更改菜单项文字onOptionsItemsSelected或onCreateOptionsMenu之外
- Android的startCamera给我空意图和…它是否会摧毁我的全局variables?