【从 0 开始开发一款直播 APP】15 Android 定位详解之 LocationManager & Geocoder 实现直播定位

本文为菜鸟窝作者蒋志碧的连载。“从 0 开始开发一款直播 APP ”系列来聊聊时下最火的直播 APP,如何完整的实现一个类"腾讯直播"的商业化项目


现在Android 开发获取用户地理位置已经愈发简单,各种地图 SDK 都提供精确定位方法。不过如果需求中只是需要模糊定位到用户的城市,系统 API 完全可以满足需求,这时如果再集成一个地图 SDK 显得很臃肿。系统 API 进行定位都很普遍了,只是在 Android 6.0 版本加入了危险权限的动态验证,只需要在原来基础上判断权限即可。

Android 系统提供了地理位置服务相关的 API 方便开发者去获取当前的地理位置,在 android.loaction 包下主要有以下两个类。

1、LocationManager:用于获取地理位置的经纬度信息

2、Geocoder:根据经纬度获取详细信息/根据相信地址获取经纬度信息

接下来我们将实现直播中的定位功能。

代码中涉及到 MVP实现定位,Android 6.0 运行时权限验证,带动画的自定义 switch 控件,请戳链接


【从 0 开始开发一款直播 APP】5.1 MVP 完全解析 -- 实现直播登录
【从 0 开始开发一款直播 APP】13 Android 6.0 运行时权限
【从 0 开始开发一款直播 APP】14 animation-list 逐帧动画自定义 Switch 控件


定位主要有以下权限:

//网络权限

//模糊定位权限:一般用于网络定位

//精确定位权限:一般用于gps定位

一、LocationManager

LocationManager 提供系统定位服务,这些服务允许应用程序定期更新设备的地理位置,或者当设备进入给定地理位置附近时,触发应用指定意图。

LocationManager 不能直接实例化,需要使用如下方法

Context.getSystemService(Context.LOCATION_SERVICE)

在 LocationManager 中必须了解两个重要知识点:

1、provider

2、LocationListener

1.1、provider

位置信息提供者。android系统一般提供三种方式获取地理位置信息。

1、GPS_PROVIDER:通过 GPS 来获取地理位置的经纬度信息。优点(获取地理位置信息精确度高),缺点(只能在户外使用,获取经纬度信息耗时,耗电)。

2、NETWORK_PROVIDER:通过移动网络的基站或者 Wi-Fi 来获取地理位置。优点(只要有网络,就可以快速定位,室内室外都可),缺点(精确度不高)。

3、PASSIVE_PROVIDER:被动接收更新地理位置信息,而不用自己请求地理位置信息。该 PASSIVE_PROVIDER 返回的位置是通过其他 providers 产生的,你可以查询 getProvider() 方法决定位置更新的由来,需要 ACCESS_FINE_LOCATION 权限,但是如果未启用 GPS,则此 provider 可能只返回粗略位置匹配。

LocationManager 提供了以下几种方法来获得地理位置提供者。

//返回当前设备的所有 provider
public List getAllProviders();
//当 enabledOnly 为 true 时,返回当前设备可使用的 provider,enabledOnly 为发false,返回所有 provider
public List getProviders(boolean enabledOnly);
//返回当前设备最符合条件的 provider。Criteria:指定条件,enabledOnly:返回当前设备可用的 provider
public String getBestProvider(Criteria criteria, boolean enabledOnly);
//返回符合条件的 provider,Criteria:指定条件,enabledOnly:返回当前设备可用的 provider
public List getProviders(Criteria criteria, boolean enabledOnly);

Criteria

指示选择 provider 的应用程序标准的类。providers 可以根据准确性,功率使用情况,报告高度,速度和方位的能力以及货币成本进行选择。该类用于指定率先选择最符合条件的 provider,根据 Cirteria 指定的条件,设备会自动选择哪种 provider。

//指示纬度和经度所需的精度。参数:Criteria.ACCURACY_FINE:表示高精确度。Criteria.ACCURACY_COARSE:表示模糊精确度。
public void setAccuracy(int accuracy);
//是否需要海拔信息
public void setAltitudeRequired(boolean altitudeRequired);
//设置方位精度。参数:Criteria.NO_REQUIREMENT 无, Criteria.ACCURACY_LOW 低, Criteria.ACCURACY_HIGH 高。
public void setBearingAccuracy(int accuracy);
//指示是否要求方位信息
public void setBearingRequired(boolean bearingRequired);
//是否允许收费
public void setCostAllowed(boolean costAllowed);
//设置水平方向精准度
public void setHorizontalAccuracy(int accuracy);
//设置垂直方向精准度
public void setVerticalAccuracy(int accuracy);
//设置电池消耗要求 参数:Criteria.NO_REQUIREMENT 无, Criteria.POWER_LOW 低, Criteria.POWER_MEDIUM 中, Criteria.POWER_HIGH 高。
public void setPowerRequirement(int level);
//设置速度精度
public void setSpeedAccuracy(int accuracy);
//是否要求速度信息
public void setSpeedRequired(boolean speedRequired);
//-----------------------------split line-------------------------------------
                             Criteria 的使用
//-----------------------------split line-------------------------------------
Criteria criteria = new Criteria();
criteria.setAccuracy(Criteria.ACCURACY_FINE);//设置定位精准度
criteria.setAltitudeRequired(false);//是否要求海拔
criteria.setBearingRequired(true);//是否要求方向
criteria.setCostAllowed(true);//是否要求收费
criteria.setSpeedRequired(true);//是否要求速度
criteria.setPowerRequirement(Criteria.NO_REQUIREMENT);//设置电池耗电要求
criteria.setBearingAccuracy(Criteria.ACCURACY_HIGH);//设置方向精确度
criteria.setSpeedAccuracy(Criteria.ACCURACY_HIGH);//设置速度精确度
criteria.setHorizontalAccuracy(Criteria.ACCURACY_HIGH);//设置水平方向精确度
criteria.setVerticalAccuracy(Criteria.ACCURACY_HIGH);//设置垂直方向精确度
//返回满足条件的,当前设备可用的location provider,当第二个参数为false时,返回当前设备所有provider中最符合条件的那个provider(但是不一定可用)。
String mProvider  = mLocationManager.getBestProvider(criteria,true);

1.2、LocationListener 位置监听器接口

public interface LocationListener {
    //当坐标改变时触发此函数
    void onLocationChanged(Location location);
  
    //当provider的状态改变时,该方法被调用。状态有三种:
    //LocationProvider#OUT_OF_SERVICE:无服务
    //LocationProvider#TEMPORARILY_UNAVAILABLE:provider不可用
    //LocationProvider#AVAILABLE:provider可用
    void onStatusChanged(String provider, int status, Bundle extras);
  
    //当provider可用时调用,比如 GPS 可用时就会调用该方法。
    void onProviderEnabled(String provider);
  
    //当provider不可用时调用该方法。比如 GPS 未打开,GPS 不可用就会调用该方法。
    void onProviderDisabled(String provider);
}

1.3、获得 LocationManager 实例

protected LocationManager  locationManager;
locationManager=(LocationManager) getSystemService(Context.LOCATION_SERVICE);

1.4、绑定监听和解绑监听

locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListener);

locationManager.removeUpdates(locationListener);

二、Geocoder

Geocoder 用于处理正向编码和反向编码。地理编码是将街道地址或其他地理位置变换为(纬度,经度)坐标的过程。逆向地理编码是将(纬度,经度)坐标转换为(部分)地址的过程。反向地理编码位置描述中的细节数量可能会有所不同,例如可能包含最近建筑物的完整街道地址,而另一个可能只包含城市名称和邮政编码。Geocoder 类需要一个未包含在核心 android 框架中的后端服务。如果平台中没有后端服务,Geocoder 查询方法将返回一个空列表。使用 isPresent() 方法来确定Geocoder 实现是否存在。由于国内使用不了Google Services 服务,因此一般的手机厂商都会在自己的手机内内置百度地图服务,或者高德地图服务来替代 Google Services 服务。

主要有以下方法

isPresent():判断当前设备是否内置了地理位置服务。返回 true 表示 Geocoder 地理编码可以使用,否则不可使用。

getFromLocationName():返回描述地理位置信息的集合。locationName:地址,maxResults:返回地址数目(1-5)

getFromLocation():根据经纬度返回对应的地理位置信息。latitude:纬度,longitude:经度,maxResults:返回地址的数目(1-5)。

【从 0 开始开发一款直播 APP】15 Android 定位详解之 LocationManager & Geocoder 实现直播定位_第1张图片

Geocoder 代码示例

private static String getAddressFromLocation(final Activity activity, Location location) {
    //Geocoder初始化
    Geocoder geocoder = new Geocoder(activity);
    //判断Geocoder地理编码是否可用
    boolean falg = geocoder.isPresent();
    try {
        //获取纬度和经度
        double latitude = location.getLatitude();
        double longitude = location.getLongitude();
        //根据经纬度获取地理信息
        List
addresses = geocoder.getFromLocation(latitude, longitude, 1); if (addresses.size() > 0) { //返回当前位置,精度可调 Address address = addresses.get(0); String sAddress; if (!TextUtils.isEmpty(address.getLocality())) { if (!TextUtils.isEmpty(address.getFeatureName())) { //存储 市 + 周边地址 sAddress = address.getLocality() + " " + address.getFeatureName(); //address.getCountryName() 国家 //address.getPostalCode() 邮编 //address.getCountryCode() 国家编码 //address.getAdminArea() 省份 //address.getSubAdminArea() 二级省份 //address.getThoroughfare() 道路 //address.getSubLocality() 二级城市 } else { sAddress = address.getLocality(); } } else { sAddress = "定位失败"; } return sAddress; } } catch (IOException e) { e.printStackTrace(); } return ""; }

三、LocationManager 使用初探

1、获取 LocationManager

2、判断定位服务(GPS,WIFI,基站)是否可用

3、设置定位监听,获取经纬度

public void getMyLocation(Context context){
  // 获取 LocationManager 实例
  mLocationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
  // 判断网络定位是否可用,可替换成 GPS 定位。
  if (mLocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) {
            mLocationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER,
                    0, 0, new LocationListener() {
                @Override
                public void onLocationChanged(Location location) {
                    //位置发生改变时回调该函数
                    location.getLatitude();//纬度
                    location.getLongitude();//经度
                }

                @Override
                public void onStatusChanged(String provider, int status, Bundle extras){
                      //状态改变回调
                      //provider:定位器名称(NetWork,Gps等)
                      //status: 3种状态,超出服务范围,临时不可用,正常可用
                      //extras: 包含定位器一些细节信息
                }

                @Override
                public void onProviderEnabled(String provider) {
                    //定位开启回调
                }

                @Override
                public void onProviderDisabled(String provider) {
                    //定位关闭回调
                }
            });
        } 
}

四、定位工具类封装:LocationMgr

主要功能:监测定位权限,通过网络获取位置信息,通过经纬度解码地理位置

public class LocationMgr {
    private static String TAG = LocationMgr.class.getSimpleName();
    private static LocationListener mLocationListener;

    //监测定位权限
    public static boolean checkLocationPermission(final Activity activity) {
        if (Build.VERSION.SDK_INT >= 23) {
            if (PackageManager.PERMISSION_GRANTED != ActivityCompat.checkSelfPermission(activity, Manifest.permission.ACCESS_FINE_LOCATION)) {
                ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, Constants.LOCATION_PERMISSION_REQ_CODE);
                return false;
            }
        }
        return true;
    }

    /**
     * 根据经纬度解码地理位置
     * @param activity
     * @param location
     * @return
     */
    private static String getAddressFromLocation(final Activity activity, Location location) {
        Geocoder geocoder = new Geocoder(activity);
        try {
            double latitude = location.getLatitude();
            double longitude = location.getLongitude();
            Log.d(TAG, "getAddressFromLocation->lat:" + latitude + ", long:" + longitude);
            List
addresses = geocoder.getFromLocation(latitude, longitude, 1); if (addresses.size() > 0) { //返回当前位置,精度可调 Address address = addresses.get(0); String sAddress; if (!TextUtils.isEmpty(address.getLocality())) { if (!TextUtils.isEmpty(address.getFeatureName())) { sAddress = address.getLocality() + " " + address.getFeatureName(); } else { sAddress = address.getLocality(); } } else { sAddress = "定位失败"; } return sAddress; } } catch (IOException e) { e.printStackTrace(); } return ""; } /** * 获取位置 * @param activity * @param locationListener * @return */ public static boolean getMyLocation(final Activity activity, final onLocationListener locationListener) { final LocationManager locationManager = (LocationManager) activity.getSystemService(Context.LOCATION_SERVICE); //判断网络定位是否可用 if (!locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)) { //调用定位提示对话框,打开定位功能 AlertDialog.Builder builder = new AlertDialog.Builder(activity); builder.setMessage("尚未开启位置定位服务"); builder.setPositiveButton("开启", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { //启动定位Activity Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS); activity.startActivity(intent); } }); builder.setNegativeButton("取消", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { } }); builder.show(); return false; } //权限检查 if (!checkLocationPermission(activity)) { return true; } // 通过网络获取位置 Location curLoc = locationManager.getLastKnownLocation(LocationManager.NETWORK_PROVIDER); if (null == curLoc) { mLocationListener = new LocationListener() { @Override public void onLocationChanged(Location location) { //获取解码的地理位置 String strAddress = getAddressFromLocation(activity, location); if (TextUtils.isEmpty(strAddress)) { //定位失败回调 locationListener.onLocationChanged(-1, 0, 0, strAddress); } else { //定位成功回调 locationListener.onLocationChanged(0, location.getLatitude(), location.getLongitude(), strAddress); } //关闭 GPS 定位功能 locationManager.removeUpdates(this); } @Override public void onStatusChanged(String provider, int status, Bundle extras) { //关闭 GPS 定位功能 locationManager.removeUpdates(this); } @Override public void onProviderEnabled(String provider) { //关闭 GPS 定位功能 locationManager.removeUpdates(this); } @Override public void onProviderDisabled(String provider) { //关闭 GPS 定位功能 locationManager.removeUpdates(this); } }; //设置监听器,自动更新的最小时间为间隔N秒(1秒为1*1000,这样写主要为了方便)或最小位移变化超过N米 locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER,8000,0,mLocationListener); }else { //获取解码的地理位置 String strAddress = getAddressFromLocation(activity,curLoc); if (TextUtils.isEmpty(strAddress)) { //定位失败回调 locationListener.onLocationChanged(-1, 0, 0, strAddress); } else { //定位成功回调 locationListener.onLocationChanged(0, curLoc.getLatitude(), curLoc.getLongitude(), strAddress); } } return true; } //自定义定位监听回调接口 public interface onLocationListener { void onLocationChanged(int code, double lat1, double long1, String location); } }

五、LocationMgr 的使用

5.1、PublishPresenter :定位逻辑实现类

初始化 onLocationListener 定位监听接口,如果定位成功,通过调用 mIPublishView.doLocationSuccess(location) 方法,提示定位成功,mIPublishView.doLocationFailed() 提示定位失败。定位按钮点击调用 doLocation() 方法最终完成位置信息展示。

public class PublishPresenter extends IPublishPresenter {
    private IPublishView mIPublishView;
    private String TAG = PublishPresenter.class.getSimpleName();

    public PublishPresenter(IPublishView iPublishView) {
        super(iPublishView);
        this.mIPublishView = iPublishView;
    }

  public LocationMgr.onLocationListener getOnLocationListener() {
      return mOnLocationListener;
  }
    //初始化定位监听回调接口
  private LocationMgr.onLocationListener mOnLocationListener = new LocationMgr.onLocationListener() {
      @Override
      public void onLocationChanged(int code, double lat1, double long1, String location) {
          //0 表示成功
          if (0 == code) {
              mIPublishView.doLocationSuccess(location);
          } else {
              mIPublishView.doLocationFailed();
          }
      }
  };
  
  //定位
  @Override
  public void doLocation() {
      if (LocationMgr.checkLocationPermission(mIPublishView.getActivity())) {
          //成功返回地理位置信息结果
          boolean success = LocationMgr.getMyLocation(mIPublishView.getActivity(), mOnLocationListener);
          if (!success) {
              mIPublishView.doLocationFailed();
          }
      }
  }
}

5.2、PublishActivity

实例化定位按钮,调用定位功能 mPublishPresenter.doLocation(),实现定位成功和失败方法并作处理。

public class PublishActivity extends BaseActivity implements View.OnClickListener, IPublishView{
private TextView mTvLBS;
private PublishPresenter mPublishPresenter;
@Override
protected void setListener() {
    mBtnLBS.setOnClickListener(this);
}
@Override
public void onClick(View v) {
    switch (v.getId()) {
        //定位
        case R.id.btn_lbs:
            if (mBtnLBS.isChecked()) {
                mBtnLBS.setChecked(false, true);
                mTvLBS.setText("不显示地理位置");
            } else {
                mBtnLBS.setChecked(true, true);
                mTvLBS.setText("正在定位中");
                //调用定位功能
                mPublishPresenter.doLocation();
            }
            break;
    }
}
//定位成功
@Override
public void doLocationSuccess(String location) {
    mTvLBS.setText(location);
}
//定位失败
@Override
public void doLocationFailed() {
    mTvLBS.setText("定位失败");
    mBtnLBS.setChecked(false, false);
}

    /**
     * 权限验证回调
     * 

* 1、权限通过 ActivityCompat 类的 checkSelfPermission() 方法判断是否有所需权限。PublishPresenter.java # checkPublishPermission() * 2、权限请求是通过 ActivityCompat 类中的 requestPermissions() 方法, * 在 OnRequestPermissionsResultCallback # onRequestPermissionsResult() 方法中回调。PublishActivity.java # onRequestPermissionsResult() * 3、应用程序可以提供一个额外的合理的使用权限调用 Activitycompat # shouldShowRequestPermissionRationale() 方法。 * Android 原生系统中,如果第二次弹出权限申请的对话框,会出现「以后不再弹出」的提示框,如果用户勾选了,你再申请权限, * 则 shouldShowRequestPermissionRationale() 返回 true,意思是说要给用户一个解释,告诉用户为什么要这个权限。 * * @param requestCode 请求码 * @param permissions 权限数组 * @param grantResults 授予结果数组 */ @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode) { //定位权限 case Constants.LOCATION_PERMISSION_REQ_CODE: if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { //获取地理位置失败处理 if (!LocationMgr.getMyLocation(this, mPublishPresenter.getOnLocationListener())) { mTvLBS.setText("定位失败"); mBtnLBS.setChecked(false, false); } } break; } } }

运行结果


【从 0 开始开发一款直播 APP】15 Android 定位详解之 LocationManager & Geocoder 实现直播定位_第2张图片

详情转至 GitHub
参考:

https://developer.android.com/reference/android/location/LocationManager.html#GPS_PROVIDER

https://developer.android.com/reference/android/location/Geocoder.html

http://blog.csdn.net/feiduclear_up/article/details/50704127

https://my.oschina.net/JumpLong/blog/89266

撸这个项目的一半,你就是大神 , 戳http://mp.weixin.qq.com/s/ZagocTlDfxZpC2IjUSFhHg

你可能感兴趣的:(【从 0 开始开发一款直播 APP】15 Android 定位详解之 LocationManager & Geocoder 实现直播定位)