Android編程之非調(diào)用系統(tǒng)界面實(shí)現(xiàn)發(fā)送彩信的方法(MMS)
本文實(shí)例講述了Android非調(diào)用系統(tǒng)界面實(shí)現(xiàn)發(fā)送彩信的方法。分享給大家供大家參考,具體如下:
一、問題:
最近有個(gè)需求,不去調(diào)用系統(tǒng)界面發(fā)送彩信功能。做過發(fā)送短信功能的同學(xué)可能第一反應(yīng)是這樣:
不使用 StartActivity,像發(fā)短信那樣,調(diào)用一個(gè)類似于發(fā)短信的方法
SmsManager smsManager = SmsManager.getDefault(); smsManager.sendTextMessage(phoneCode, null, text, null, null);
二、解決方法:
由于android上根本就沒有提供發(fā)送彩信的接口,如果你想發(fā)送彩信,對(duì)不起,請(qǐng)調(diào)用系統(tǒng)彩信app界面,如下:
Intent sendIntent = new Intent(Intent.ACTION_SEND, Uri.parse("mms://")); sendIntent.setType("image/jpeg"); String url = "file://sdcard//tmpPhoto.jpg"; sendIntent.putExtra(Intent.EXTRA_STREAM, Uri.parse(url)); startActivity(Intent.createChooser(sendIntent, "MMS:"));
但是這種方法往往不能滿足我們的需求,能不能不調(diào)用系統(tǒng)界面,自己實(shí)現(xiàn)發(fā)送彩信呢?經(jīng)過幾天的努力,終于找到了解決辦法。
第一步:先構(gòu)造出你要發(fā)送的彩信內(nèi)容,即構(gòu)建一個(gè)pdu,需要用到以下幾個(gè)類,這些類都是從android源碼的MMS應(yīng)用中mms.pdu包中copy出來的。你需要將pdu包中的所有類
都拷貝到你的工程中,然后自己酌情調(diào)通。
final SendReq sendRequest = new SendReq(); final PduBody pduBody = new PduBody(); final PduPart part = new PduPart(); //存放附件,每個(gè)附件是一個(gè)part,如果添加多個(gè)附件,就想body中add多個(gè)part。 pduBody.addPart(partPdu); sendRequest.setBody(pduBody); final PduComposer composer = new PduComposer(ctx, sendRequest); final byte[] bytesToSend = composer.make(); //將彩信的內(nèi)容以及主題等信息轉(zhuǎn)化成byte數(shù)組,準(zhǔn)備通過http協(xié)議 //發(fā)送到 ”http://mmsc.monternet.com”;
第二步:發(fā)送彩信到彩信中心。
構(gòu)建pdu的代碼:
String subject = "測試彩信"; String recipient = "接收彩信的號(hào)碼";//138xxxxxxx final SendReq sendRequest = new SendReq(); final EncodedStringValue[] sub = EncodedStringValue.extract(subject); if (sub != null && sub.length > 0) { sendRequest.setSubject(sub[0]); } final EncodedStringValue[] phoneNumbers = EncodedStringValue.extract(recipient); if (phoneNumbers != null && phoneNumbers.length > 0) { sendRequest.addTo(phoneNumbers[0]); } final PduBody pduBody = new PduBody(); final PduPart part = new PduPart(); part.setName("sample".getBytes()); part.setContentType("image/png".getBytes()); String furl = "file://mnt/sdcard//1.jpg"; final PduPart partPdu = new PduPart(); partPdu.setCharset(CharacterSets.UTF_8);//UTF_16 partPdu.setName(part.getName()); partPdu.setContentType(part.getContentType()); partPdu.setDataUri(Uri.parse(furl)); pduBody.addPart(partPdu); sendRequest.setBody(pduBody); final PduComposer composer = new PduComposer(ctx, sendRequest); final byte[] bytesToSend = composer.make(); Thread t = new Thread(new Runnable() { @Override public void run() { try { HttpConnectInterface.sendMMS(ctx, bytesToSend); // } catch (IOException e) { e.printStackTrace(); } } }); t.start();
發(fā)送pdu到彩信中心的代碼:
public static String mmscUrl = "http://mmsc.monternet.com"; // public static String mmscUrl = "http://www.baidu.com/"; public static String mmsProxy = "10.0.0.172"; public static String mmsProt = "80"; private static String HDR_VALUE_ACCEPT_LANGUAGE = ""; // Definition for necessary HTTP headers. private static final String HDR_KEY_ACCEPT = "Accept"; private static final String HDR_KEY_ACCEPT_LANGUAGE = "Accept-Language"; private static final String HDR_VALUE_ACCEPT = "*/*, application/vnd.wap.mms-message, application/vnd.wap.sic"; public static byte[] sendMMS(Context context, byte[] pdu)throws IOException{ HDR_VALUE_ACCEPT_LANGUAGE = getHttpAcceptLanguage(); if (mmscUrl == null) { throw new IllegalArgumentException("URL must not be null."); } HttpClient client = null; try { // Make sure to use a proxy which supports CONNECT. client = HttpConnector.buileClient(context); HttpPost post = new HttpPost(mmscUrl); //mms PUD START ByteArrayEntity entity = new ByteArrayEntity(pdu); entity.setContentType("application/vnd.wap.mms-message"); post.setEntity(entity); post.addHeader(HDR_KEY_ACCEPT, HDR_VALUE_ACCEPT); post.addHeader(HDR_KEY_ACCEPT_LANGUAGE, HDR_VALUE_ACCEPT_LANGUAGE); //mms PUD END HttpParams params = client.getParams(); HttpProtocolParams.setContentCharset(params, "UTF-8"); HttpResponse response = client.execute(post); LogUtility.showLog(tag, "111"); StatusLine status = response.getStatusLine(); LogUtility.showLog(tag, "status "+status.getStatusCode()); if (status.getStatusCode() != 200) { // HTTP 200 is not success. LogUtility.showLog(tag, "!200"); throw new IOException("HTTP error: " + status.getReasonPhrase()); } HttpEntity resentity = response.getEntity(); byte[] body = null; if (resentity != null) { try { if (resentity.getContentLength() > 0) { body = new byte[(int) resentity.getContentLength()]; DataInputStream dis = new DataInputStream(resentity.getContent()); try { dis.readFully(body); } finally { try { dis.close(); } catch (IOException e) { Log.e(tag, "Error closing input stream: " + e.getMessage()); } } } } finally { if (entity != null) { entity.consumeContent(); } } } LogUtility.showLog(tag, "result:"+new String(body)); return body; } catch (IllegalStateException e) { LogUtility.showLog(tag, "",e); // handleHttpConnectionException(e, mmscUrl); } catch (IllegalArgumentException e) { LogUtility.showLog(tag, "",e); // handleHttpConnectionException(e, mmscUrl); } catch (SocketException e) { LogUtility.showLog(tag, "",e); // handleHttpConnectionException(e, mmscUrl); } catch (Exception e) { LogUtility.showLog(tag, "",e); //handleHttpConnectionException(e, mmscUrl); } finally { if (client != null) { // client.; } } return new byte[0]; }
至此,彩信的發(fā)送算是完成了。
總結(jié):android的彩信相關(guān)操作都是沒有api的,包括彩信的讀取、發(fā)送、存儲(chǔ)。這些過程都是需要手動(dòng)去完成的。想要弄懂這些過程,需要仔細(xì)閱讀android源碼中的mms這個(gè)app。還有就是去研究mmssms.db數(shù)據(jù)庫,因?yàn)椴市诺淖x取和存儲(chǔ)其實(shí)都是對(duì)mmssms.db這個(gè)數(shù)據(jù)庫的操作過程。而且因?yàn)檫@個(gè)是共享的數(shù)據(jù)庫,所以只能用ContentProvider這個(gè)組件去操作db。
總之,想要研究彩信這塊(包括普通短信),你就必須的研究mmssms.db的操作方法,多多了解每個(gè)表對(duì)應(yīng)的哪個(gè)uri,每個(gè)uri能提供什么樣的操作,那些字段代表短信的那些屬性等。
最后推薦個(gè)好用的sqlite查看工具:SQLite Database Browser。
希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。
- 簡單掌握Android開發(fā)中彩信的發(fā)送接收及其附件的處理
- Android實(shí)現(xiàn)帶附件的郵件發(fā)送功能
- Android開發(fā)中怎樣調(diào)用系統(tǒng)Email發(fā)送郵件(多種調(diào)用方式)
- Android中發(fā)送Http請(qǐng)求(包括文件上傳、servlet接收)的實(shí)例代碼
- Android HTTP發(fā)送請(qǐng)求和接收響應(yīng)的實(shí)例代碼
- Android發(fā)送GET與POST請(qǐng)求的DEMO詳解
- Android中Webview打開網(wǎng)頁的同時(shí)發(fā)送HTTP頭信息方法
- android實(shí)現(xiàn)藍(lán)牙文件發(fā)送的實(shí)例代碼,支持多種機(jī)型
- Android發(fā)送短信功能代碼
- Android 后臺(tái)發(fā)送郵件示例 (收集應(yīng)用異常信息+Demo代碼)
- android中可以通過兩種方式調(diào)用接口發(fā)送短信
- Android下通過httpClient發(fā)送GET和POST請(qǐng)求的實(shí)例代碼
- Android實(shí)現(xiàn)將已發(fā)送的短信寫入短信數(shù)據(jù)庫的方法
- Android實(shí)現(xiàn)彩信附件的添加與刪除功能
相關(guān)文章
詳解Flutter中l(wèi)istview的高級(jí)用法
一般我們使用Listview的方式是構(gòu)建要展示的item,然后將這些item傳入ListView的構(gòu)造函數(shù)即可,通常情況下這樣做是夠用了,但是不排除我們會(huì)有一些其他的特殊需求。今天我們會(huì)來講解一下ListView的一些高級(jí)用法,希望對(duì)大家有所幫助2023-01-01Android 在 res/layout 文件夾 下創(chuàng)建一個(gè) 子文件夾實(shí)例
這篇文章主要介紹了Android 在 res/layout 文件夾 下創(chuàng)建一個(gè) 子文件夾實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-03-03Android自定義View實(shí)現(xiàn)豎向滑動(dòng)回彈效果
這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)滑動(dòng)回彈效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04Flutter實(shí)現(xiàn)軟鍵盤與其它區(qū)域絲滑切換效果
這篇文章主要為大家詳細(xì)介紹了如何使用Flutter實(shí)現(xiàn)軟鍵盤與其它區(qū)域絲滑切換效果,文中的示例代碼講解詳細(xì),需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03Android實(shí)現(xiàn)垂直進(jìn)度條VerticalSeekBar
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)垂直進(jìn)度條VerticalSeekBar的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android中RecyclerView布局代替GridView實(shí)現(xiàn)類似支付寶的界面
RecyclerView比GridView來得更加強(qiáng)大,不僅是在分割線的繪制方面,在條目的編輯上也做得同樣出色,下面就來看一下Android中RecyclerView布局代替GridView實(shí)現(xiàn)類似支付寶的界面的實(shí)例2016-06-06Android實(shí)現(xiàn)原生鎖屏頁面音樂控制
這篇文章主要介紹了Android實(shí)現(xiàn)原生鎖屏頁面音樂控制,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12