項目需要后臺闭毕В活,但無論怎么彼勾椋活经伙,只要用戶主動kill掉,app依然是活不了吮成。
發(fā)現(xià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)
第一個沒有過濾條件,鎖屏就停止掃描
第二個可以加過濾條件作瞄,鎖屏不影響掃描?
第三個的掃描結果由PendingIntent發(fā)送茶宵,即使app沒有在運行,系統(tǒng)也可以掃描后喚醒app宗挥,這就是我們要的方法了乌庶。
一、一些概念
PendingIntent是對Intent的封裝契耿,是滿足某些條件或觸發(fā)某些事件后才執(zhí)行指定的行為瞒大,主要用于鬧鐘、通知搪桂、桌面部件透敌。Android的四大組件之間通信用Intent,跨進程通信用PendingIntent踢械。
Android 8.0 引進了Context.startForegroundService()酗电,在系統(tǒng)創(chuàng)建服務后,應用需要在ANR發(fā)生前調用startForeground(int ,android.app.Notification)内列,如果未及時調用該方法撵术,系統(tǒng)將報ANR錯誤 。系統(tǒng)給前臺服務的ANR時間是20秒话瞧。
用startScan藍牙喚醒的原理是:app向系統(tǒng)訂閱了掃描結果(預先加了過濾條件)嫩与,當藍牙連接斷開的時候寝姿,設備就會發(fā)廣播,這時系統(tǒng)就可以掃描到對應的廣播划滋,喚醒對應的service饵筑,這時想做什么操作就根據(jù)你的項目需要了。至于系統(tǒng)會為你掃描多久古毛,這個還沒測試翻翩。
二、權限
<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" /><!-- 后臺運行權限-->
三稻薇、動態(tài)請求定位權限(Android 6以上必須要)
/**
* 請求位置權限
*/
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();
? ? ? ? }
}
}
/**
* 判斷位置權限是否打開
*
* @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;
}
四嫂冻、開啟服務
@Override?
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
? ? setContentView(R.layout.activity_main);
? ? ///請求位置權限
? ? requestLocationService();
? ? ///開啟服務
? ? startForegroundService(new Intent(this, TestService.class));
}
五、開啟藍牙掃描
@RequiresApi(api = Build.VERSION_CODES.O)
private void onOPEN() {
List scanFilterList =new ArrayList<>();
? ? ScanFilter.Builder builder =new ScanFilter.Builder();
? ? builder.setServiceUuid(ParcelUuid.fromString("設備廣播的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);
? ? //啟動藍牙掃描
? ? bluetoothAdapter.getBluetoothLeScanner().startScan(scanFilterList, settings, callbackIntent);
}
(1)setScanMode有四個參數(shù)可以選 :
SCAN_MODE_BALANCED:在平衡電源模式下執(zhí)行藍牙LE掃描塞椎。返回掃描結果的速度能夠很好地權衡掃描頻率和功耗桨仿。
SCAN_MODE_LOW_LATENCY:掃描使用最高占空比。建議只在應用程序在前臺運行時使用此模式案狠。
SCAN_MODE_LOW_POWER:在低功耗模式下執(zhí)行藍牙LE掃描服傍。這是默認的掃描模式,因為它消耗的能量最少骂铁。如果掃描應用程序不在前臺吹零,則強制執(zhí)行此模式。
SCAN_MODE_OPPORTUNISTIC:一種特殊的藍牙LE掃描模式拉庵。使用這種掃描模式的應用程序將被動地偵聽其他掃描結果灿椅,而不啟動BLE掃描本身
(2)settingBuilder.setMatchMode有兩個參數(shù)可以選:
MATCH_MODE_AGGRESSIVE: ?信號弱也會報告?
MATCH_MODE_STICKY: ?信號比較強和掃描到的次數(shù)比較多才會報告
(3)settingBuilder.setCallbackType也有其他參數(shù)可選,但適用的就一個
? (4) ?ScanFilter ?的過濾方法有幾個,如下圖钞支,打勾的是測試了可行的茫蛹,但只有第一個DeviceAddress有唯一性 ?
六、服務里接收掃描結果
@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);
? ? //獲取到的藍牙設備的回調類型? 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);
}