Android O開發(fā)預(yù)覽版已經(jīng)出來了简珠,是時(shí)候開始猜測(cè)一下Android O給我們帶來了什么樣的驚喜。
每個(gè)android 開發(fā)者都應(yīng)該檢查即將到來的操作系統(tǒng)版本的行為變化列表被碗。他已經(jīng)被很好的劃分了其馏,所以你只用看針對(duì)Android O的應(yīng)用成俗的行為的更改象颖,以及影響運(yùn)行Android O的設(shè)備上每個(gè)API級(jí)別的更改。
但是你可以跳過Broadcast 部分,因?yàn)榻酉聛硪f的就是镇饺。
如果你并沒有計(jì)劃將你的app更新到Android O,就可以跳過該文章了送讲,如果你準(zhǔn)備把你的app更新到android上并且在你的AndroidManifest中注冊(cè)了隱式的BroadcastReceiver的話奸笤,就繼續(xù)閱讀吧。
NOTE:有一些不受影響的廣播的列表哼鬓,因此监右,如果你的應(yīng)用程序依賴于ACTION_BOOT_COMPLETED或ACTION_HEADSET_PLUG,還尤其多其他的异希,你的應(yīng)用程序?qū)?huì)繼續(xù)正常工作健盒。
這是關(guān)于節(jié)約電的
最大限度的提高性能和電池壽命一直是Android的一大優(yōu)先事項(xiàng),在Android6.0版本称簿,Doze和App StandBy模式極大的優(yōu)化了以前較差的電池壽命扣癣,這也是現(xiàn)代手機(jī)的真正痛點(diǎn)。
雖然Doze模式限制所有應(yīng)用程序在特殊維護(hù)窗口中執(zhí)行耗電量的后臺(tái)任務(wù)憨降,App Standby模式會(huì)延遲最近最少使用的應(yīng)用程序的網(wǎng)絡(luò)活動(dòng)父虑,隨著Doze推出JobScheduler,Google開始敦促開發(fā)者使用它來開發(fā)所有的后臺(tái)任務(wù)。
節(jié)省電池電量與BroadcastReceivers有什么關(guān)系授药?
使用Android7.0士嚎,在AndroidManifest中注冊(cè)隱式廣播的支持已經(jīng)被刪除了。
當(dāng)app在Android O上運(yùn)行時(shí)悔叽,所有的在manifest中注冊(cè)的隱式廣播將會(huì)停止工作莱衩。
事實(shí)證明,應(yīng)用程序在其清單文件中注冊(cè)了大量的不必要的隱式BroadcastReceivers骄蝇,導(dǎo)致不必要的電池消耗膳殷,例如,CONNECIVITY_ACTION廣播的許多應(yīng)用程序注冊(cè)接收器∽裕“那又怎么樣”册招,你可能會(huì)想,這與電池消耗有什么關(guān)系勒极。
想象一下是掰,你有十幾個(gè)應(yīng)用程序可以監(jiān)聽連接更改事件,當(dāng)wifi掉線辱匿,Android系統(tǒng)發(fā)送CONNECTIVITY_ACTION廣播键痛,并導(dǎo)致所有十幾個(gè)應(yīng)用程序喚醒并對(duì)更改做出反應(yīng)。
由于wifi不再可用匾七,您可以連接到移動(dòng)網(wǎng)絡(luò)絮短,猜猜會(huì)做什么操作,CONNECTIVITY_ACTION廣播再次發(fā)送昨忆,當(dāng)你回家重新連接到您的wifi時(shí)丁频,同樣的順序再次發(fā)生。
由于所有這些應(yīng)用程序都在AndroidManifest中注冊(cè)這些廣播邑贴,所以他們總是被喚醒以接收這些事件席里,即使他們不在前臺(tái),甚至是沒有運(yùn)行拢驾。
我已經(jīng)在我的清單文件中注冊(cè)了一個(gè)BroadcastReceiver奖磁,現(xiàn)在呢?
不要驚慌繁疤,盡管將現(xiàn)有的代碼遷移到箭筒的Android O可能會(huì)令人沮喪咖为,但這并不是一個(gè)重大變化,而是一個(gè)優(yōu)勢(shì)嵌洼。
確實(shí)BroadcastReceiver是隱式還是顯式
根據(jù)文檔案疲,與您的應(yīng)用程序無直接關(guān)系的任何廣播都是隱式廣播,像文檔狀態(tài)一樣麻养,ACTION_PACKAGE_REPLACE是一個(gè)隱式廣播褐啡,因?yàn)樗麜?huì)通知手機(jī)上每個(gè)安裝的包。
同樣鳖昌,與您的應(yīng)用程序直接相關(guān)的任何廣播都是明確的备畦,因此ACTION_MY_PACKAGE_REPLACED是一個(gè)顯式廣播,因?yàn)橹挥心阕约旱膽?yīng)用程序更新時(shí)才通知许昨,你可以監(jiān)聽的大多數(shù)廣播都是隱式的懂盐。
檢查你的應(yīng)用程序是否受影響
Android Developer文檔提供了不受影響并將繼續(xù)正常運(yùn)行的隱式廣播列表,這意味著如果您的應(yīng)用程序僅在AndroidManifest中注冊(cè)這些隱式廣播的監(jiān)聽器糕档,并且只有這些才是安全的莉恼。
最常見的不受影響的是ACTION_BOOT_COMPLETED,ACTION_HEADSET_PLUG,ACTION_CONNECTION_STATE_CHANGED用于藍(lán)牙狀態(tài)更改,還有許多其他更改。
JobScheduler
如果你監(jiān)聽了網(wǎng)絡(luò)連接俐银,接收設(shè)備空閑事件尿背,則可能需要使用JobScheduler來執(zhí)行你的任務(wù)。
Component myService = new ComponentName(this,MyService.class);
JobInfo myInfo = new JobInfo.Builder(myService)
.setRequiresCharging(true)
.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY)
.setPersisted(true)
.build();
JobScheduler jobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
jobScheduler.schedule(myJob);
JobScheduler很好的與Android系統(tǒng)和Doze模式相適應(yīng)捶惜,他允許基于你定義的條件執(zhí)行任務(wù)田藐。
JobScheduler只支持Android棉花糖以及以上的系統(tǒng),如果你的app的minSdkVersion小于21吱七,則有類似API的庫提供向后兼容汽久,正式推薦的是FirebaseJobDispatcher。
Context.registerReceiver()可以一直使用
無論您的廣播是隱式的還是顯示的踊餐,您都可以使用Activity景醇,F(xiàn)ragment泊业,Service甚至自定義視圖中的Context.registerReceiver()方法來注冊(cè)系吭。
如果你的清單文件中有如下的監(jiān)聽器
...
<receiver android:name=".ChargerReceiver">
<intent-filter>
<action android:name="android.intent.action.ACTION_POWER_CONNECTED"/>
</intent-filter>
</receiver>
...
你需要移除上面的代碼,并在fragment镀脂、service苍碟、fragment中監(jiān)聽該邏輯。
public class MainActivity extends AppCompatActivity {
private BroadcastReceiver chargerReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
chargerReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
// TODO: Awesome things
}
};
registerReceiver(
chargerReceiver,
new IntentFilter(Intent.ACTION_POWER_CONNECTED)
);
}
@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(chargerReceiver);
}
}
注冊(cè)BroadcastReceivers與Context.registerReceiver()的區(qū)別在于撮执,最終需要取消注冊(cè)微峰,而你的清單文件中定義的接收者會(huì)喚醒您的應(yīng)用程序。
如果在activity/fragment中完成抒钱,一個(gè)取消注冊(cè)的好地方是onDestroy回調(diào)蜓肆,在服務(wù)中,如果你不需要再收到這些事件谋币,請(qǐng)注銷仗扬。
擺脫Manifest文件中的所有隱式廣播,使用JobScheduler蕾额,F(xiàn)irebaseJobDispatcher或替代方案替代您的任務(wù)早芭,如果不支持你的廣播,那么請(qǐng)?jiān)赼ctivity诅蝶、fragment退个、service中注冊(cè)你的廣播。