項目需要后臺?;睿珶o論怎么保活,只要用戶主動kill掉,app依然是活不了。
發(fā)現(xiàn)了藍(lán)牙喚醒這個方式,用戶主動kill掉也可行。
Android 8.0開始提供了 startscan的方法,
public void startScan(ScanCallback callback)
public void startScan(List<ScanFilter> filters,ScanSettings settings,ScanCallback callback)
public int startScan(List<ScanFilter> filters,ScanSettings settings,PendingIntent callbackIntent)
第一個沒有過濾條件,鎖屏就停止掃描
第二個可以加過濾條件,鎖屏不影響掃描?
第三個的掃描結(jié)果由PendingIntent發(fā)送,即使app沒有在運行,系統(tǒng)也可以掃描后喚醒app,這就是我們要的方法了。
一、一些概念
PendingIntent是對Intent的封裝,是滿足某些條件或觸發(fā)某些事件后才執(zhí)行指定的行為,主要用于鬧鐘、通知、桌面部件。Android的四大組件之間通信用Intent,跨進程通信用PendingIntent。
Android 8.0 引進了Context.startForegroundService(),在系統(tǒng)創(chuàng)建服務(wù)后,應(yīng)用需要在ANR發(fā)生前調(diào)用startForeground(int ,android.app.Notification),如果未及時調(diào)用該方法,系統(tǒng)將報ANR錯誤 。系統(tǒng)給前臺服務(wù)的ANR時間是20秒。
用startScan藍(lán)牙喚醒的原理是:app向系統(tǒng)訂閱了掃描結(jié)果(預(yù)先加了過濾條件),當(dāng)藍(lán)牙連接斷開的時候,設(shè)備就會發(fā)廣播,這時系統(tǒng)就可以掃描到對應(yīng)的廣播,喚醒對應(yīng)的service,這時想做什么操作就根據(jù)你的項目需要了。至于系統(tǒng)會為你掃描多久,這個還沒測試。
二、權(quán)限
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" /><!-- 后臺運行權(quán)限-->
三、動態(tài)請求定位權(quán)限(Android 6以上必須要)
/**
* 請求位置權(quán)限
*/
public void requestLocationService() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (isGpsEnabled()) {
if (this.checkSelfPermission(Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, 99);
? ? ? ? ? ? }
}else {
AlertDialog.Builder builder =new AlertDialog.Builder(this);
? ? ? ? ? ? builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialogInterface, int i) {
// Show location settings when the user acknowledges the alert dialog
? ? ? ? ? ? ? ? ? ? Intent intent =new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
? ? ? ? ? ? ? ? ? ? startActivity(intent);
? ? ? ? ? ? ? ? }
});
? ? ? ? ? ? Dialog alertDialog = builder.create();
? ? ? ? ? ? alertDialog.getWindow().setGravity(Gravity.CENTER);
? ? ? ? ? ? alertDialog.setCanceledOnTouchOutside(false);
? ? ? ? ? ? alertDialog.show();
? ? ? ? }
}
}
/**
* 判斷位置權(quán)限是否打開
*
* @return
*/
public boolean isGpsEnabled() {
LocationManager locationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
? ? if (locationManager ==null) {
return false;
? ? }
boolean gps = locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER);
? ? boolean network = locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER);
? ? return gps || network;
}
四、開啟服務(wù)
@Override?
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
? ? setContentView(R.layout.activity_main);
? ? ///請求位置權(quán)限
? ? requestLocationService();
? ? ///開啟服務(wù)
? ? startForegroundService(new Intent(this, TestService.class));
}
五、開啟藍(lán)牙掃描
@RequiresApi(api = Build.VERSION_CODES.O)
private void onOPEN() {
List scanFilterList =new ArrayList<>();
? ? ScanFilter.Builder builder =new ScanFilter.Builder();
? ? builder.setServiceUuid(ParcelUuid.fromString("設(shè)備廣播的UUID"));
? ? ScanFilter scanFilter = builder.build();
? ? scanFilterList.add(scanFilter);
? ? ScanSettings.Builder settingBuilder =new ScanSettings.Builder();
? ? settingBuilder.setScanMode(ScanSettings.SCAN_MODE_BALANCED);
? ? settingBuilder.setMatchMode(ScanSettings.MATCH_MODE_AGGRESSIVE);
? ? settingBuilder.setCallbackType(ScanSettings.CALLBACK_TYPE_ALL_MATCHES);
? ? settingBuilder.setLegacy(true);
? ? ScanSettings settings = settingBuilder.build();
? ? callbackIntent = PendingIntent.getForegroundService(this, 1, new Intent(action).setPackage(getPackageName()), PendingIntent.FLAG_UPDATE_CURRENT);
? ? //啟動藍(lán)牙掃描
? ? bluetoothAdapter.getBluetoothLeScanner().startScan(scanFilterList, settings, callbackIntent);
}
(1)setScanMode有四個參數(shù)可以選 :
SCAN_MODE_BALANCED:在平衡電源模式下執(zhí)行藍(lán)牙LE掃描。返回掃描結(jié)果的速度能夠很好地權(quán)衡掃描頻率和功耗。
SCAN_MODE_LOW_LATENCY:掃描使用最高占空比。建議只在應(yīng)用程序在前臺運行時使用此模式。
SCAN_MODE_LOW_POWER:在低功耗模式下執(zhí)行藍(lán)牙LE掃描。這是默認(rèn)的掃描模式,因為它消耗的能量最少。如果掃描應(yīng)用程序不在前臺,則強制執(zhí)行此模式。
SCAN_MODE_OPPORTUNISTIC:一種特殊的藍(lán)牙LE掃描模式。使用這種掃描模式的應(yīng)用程序?qū)⒈粍拥貍陕犉渌麙呙杞Y(jié)果,而不啟動BLE掃描本身
(2)settingBuilder.setMatchMode有兩個參數(shù)可以選:
MATCH_MODE_AGGRESSIVE: ?信號弱也會報告?
MATCH_MODE_STICKY: ?信號比較強和掃描到的次數(shù)比較多才會報告
(3)settingBuilder.setCallbackType也有其他參數(shù)可選,但適用的就一個
? (4) ?ScanFilter ?的過濾方法有幾個,如下圖,打勾的是測試了可行的,但只有第一個DeviceAddress有唯一性 ?

六、服務(wù)里接收掃描結(jié)果
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i(TAG, "onStartCommand");
? ? if (intent.getAction() ==null) {
return super.onStartCommand(intent, flags, startId);
? ? }
//獲取返回的錯誤碼-1
? ? int errorCode = intent.getIntExtra(BluetoothLeScanner.EXTRA_ERROR_CODE, -1);//ScanSettings.SCAN_FAILED_*
? ? Log.i(TAG, "errorCode=" + errorCode);
? ? //獲取到的藍(lán)牙設(shè)備的回調(diào)類型? 1=CALLBACK_TYPE_ALL_MATCHES
? ? int callbackType = intent.getIntExtra(BluetoothLeScanner.EXTRA_CALLBACK_TYPE, -1);//ScanSettings.CALLBACK_TYPE_*
? ? Log.i(TAG, "callbackType=" + callbackType);
? ? List scanResults = (List) intent.getSerializableExtra(BluetoothLeScanner.EXTRA_LIST_SCAN_RESULT);
? ? if (scanResults !=null) {
for (ScanResult result : scanResults) {
Log.i(TAG, result.getDevice().getName() +",," + result.getDevice().getAddress() +",," + result.getRssi() +",," + result.getScanRecord());
? ? ? ? }
}
return super.onStartCommand(intent, flags, startId);
}