7

我正在使用 AltBeacon 库开发蓝牙应用程序。似乎BeaconManager每个应用程序只允许在实例上。我面临的问题是:我想要一个持续运行的后台服务,不断进行蓝牙测距并发送通知。如果我打开我的应用程序(将其带到前台),我就是暂停测距的服务。然后,前台活动将进行测距并在屏幕上显示内容。

问题是BeaconManager beaconManager = BeaconManager.getInstanceForApplication(this);活动和服务中的信标管理器(来自)是同一个实例。因此,当活动关闭时,beaconManager.unbind(this);被调用并且服务中的范围通知器不再触发。

是否可以获得信标管理器的两个单独实例?如果没有,我该如何在持续运行的服务和活动中进行测距?

测距活动

@Override
protected void onCreate(Bundle savedInstanceState) {
...
regionEstimote = new Region("estimote", null, null, null);
beaconManager.getBeaconParsers().add(new BeaconParser().setBeaconLayout("m:2-3=0215,i:4-19,i:20-21,i:22-23,p:24-24"));
beaconManager.bind(this);
}
@Override
protected void onDestroy() {
    super.onDestroy();
    notificationManager.cancel(NOTIFICATION_ID);
    //beaconManager.unbind(this);
}
@Override
public void onBeaconServiceConnect() {
    beaconManager.setRangeNotifier(new RangeNotifier() {
        ....
    });
    try {
        beaconManager.startRangingBeaconsInRegion(regionEstimote);
    } catch (RemoteException e) {
        Log.e(TAG, "RangingActivity", e);
    }
}

信标服务.java

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    if(beaconHistory == null)
        beaconHistory = new HashMap<Integer, Date>();

    mBeaconManager = BeaconManager.getInstanceForApplication(this);
    mBeaconManager.getBeaconParsers().add(new BeaconParser().
            setBeaconLayout("m:2-3=0215,i:4-19,i:20-21,i:22-23,p:24-24"));

    return START_STICKY;
}

@Override
public void onDestroy() {
    super.onDestroy();
    beaconHistory = null;
    mBeaconManager.unbind(this);
}
@Override
public void onBeaconServiceConnect() {
    mBeaconManager.setRangeNotifier(new RangeNotifier() {
        @Override
        public void didRangeBeaconsInRegion(Collection<Beacon> beacons, Region region) {
            if(ActivityBase.isActivityVisible()) {   //don't do ranging logic if any activity from my app is in the foreground
                return;
            }
            ...
        }
    });

    try {
        mBeaconManager.startRangingBeaconsInRegion(regionMint);
    } catch (RemoteException e) {
        Log.e(TAG, "BeaconService", e);
    }
}
4

2 回答 2

6

这是自定义android.app.Application类非常有用的情况。这BeaconManager是一个单例,因此只允许同时存在一个。同样,Application该类对于每个活动的 Android 应用程序都有一个实例。如果您想同时在 anActivity和 a中进行信标检测Service,请使用集中Application类进行绑定BeaconManager,然后将回调转发到 yourActivity和 your Service.

您可以在此处查看绑定BeaconManager到类中的示例,Application然后将回调传递给Activity参考应用程序中的 主/java/org/altbeacon/beaconreference/BeaconReferenceApplication.java

于 2014-10-28T14:14:38.993 回答
1

您应该扩展 Applicaton 类并像这里解释的那样启动区域后台监控(请参阅“在后台启动应用程序”)。

为了进行测距,在同一个类中实现 RangeNotifier:

public class AndroidApp extends Application implements BootstrapNotifier, RangeNotifier {...

在 didEnterRegion 中开始测距:

@Override
public void didEnterRegion(Region region) {
    try {
        mBeaconManager.startRangingBeaconsInRegion(region);
    }
    catch (RemoteException e) {
        if (BuildConfig.DEBUG) Log.d(Const.TAG, "Can't start ranging");
    }
}
  1. 实现 didRangeBeaconsInRegion 方法:

    @Override
    public void didRangeBeaconsInRegion(Collection<Beacon> beacons, Region region) {
        if (beacons.size() > 0) {
            Beacon firstBeacon = beacons.iterator().next();
            if (BuildConfig.DEBUG) Log.d(Const.TAG, "Beacon ranged: UUID: "
                    + firstBeacon.getId1().toString() + " Major: "
                    + firstBeacon.getId2().toString() + " Minor: "
                    + firstBeacon.getId3().toString());
    
            // Do something with the result
    
            // Stop ranging
            try {
                mBeaconManager.stopRangingBeaconsInRegion(region);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    }
    
于 2015-12-17T06:03:08.550 回答