开箱即用的Google与百度定位坐标系转换实例

目录
  • 集成谷歌定位与百度定位并在地图上显示
    • 项目背景
  • 一. 集成谷歌定位
    • 定位Api使用流程:
  • 二. 集成百度定位
    • Application中初始化SDK
  • 三. 坐标系转换与地图上的显示

集成谷歌定位与百度定位并在地图上显示

项目背景

东南亚某小国的App,需要用到定位与地图显示,我们都知道海外人士都是喜欢谷歌地图与谷歌定位的,那么必然是要优先使用谷歌定位的,但是中国手机卖的很好,部分中国手机的海外版是没有谷歌服务的,那么我使用百度地图进行降级处理。

两者使用的定位坐标系不同,如果需要在谷歌地图上展示百度定位,需要如何转换呢?或者说谷歌的定位如何在百度地图上展示呢?

一. 集成谷歌定位

跟项目下的build.gradle

别的先不说,先把谷歌服务插件给引入

    dependencies {
        classpath "com.android.tools.build:gradle:7.0.3"
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
        classpath "com.google.gms:google-services:4.3.8"
    }

app模块下的build.gradle最底部 应用谷歌服务插件

apply plugin: "com.google.gms.google-services"

然后去谷歌后台申请App-Key相关的项目,需要填写包名和一些签名文件的SHA1值,注意这里relese包的SHA1值,和谷歌市场上架的SHA1值是不同的,上架之后需要去谷歌市场把谷歌市场上的SHA1值设置进去,因为谷歌市场上架之后会把你的签名文件包装一次,完全不同的签名文件了。

google-services.json
location
   //定位功能
    api "com.google.android.gms:play-services-location:16.0.0"

定位Api使用流程:

权限定义

    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

注意使用之前要动态申请权限,这里不做演示

     //初始化谷歌地图API
     if (googleApiClient == null) {
        googleApiClient = new GoogleApiClient.Builder(mActivity)
            .addConnectionCallbacks(this)
            .addOnConnectionFailedListener(this)
            .addApi(LocationServices.API)
            .build();
    }
   //Google-Location-Api 服务连接成功回调
    @Override
    public void onConnected(@Nullable Bundle bundle) {
        Log.e("MapLastLocationUtils", "Google-Api服务连接成功了");
        getAPILastLocation();
    }
    //Google-Location-Api 服务异常连接暂停
    @Override
    public void onConnectionSuspended(int i) {
        Log.e("MapLastLocationUtils", "Google-Api服务被暂停了");
        if (googleApiClient != null && !googleApiClient.isConnected())
            googleApiClient.connect();
    }
    //Google-Location-Api 连接异常
    @Override
    public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
        Log.e("MapLastLocationUtils", "Google-Api服务连接错误");
        //如果连接错误直接用百度定位
    }
    public void stopLocation() {
        if (googleApiClient != null && googleApiClient.isConnected()){
            googleApiClient.disconnect();
        }
    }
    private void getAPILastLocation() {
    FusedLocationProviderClient fusedLocationProviderClient = LocationServices.getFusedLocationProviderClient(mActivity);
    fusedLocationProviderClient.getLastLocation().addOnSuccessListener(mActivity, location -> {
        if (location != null && location.getLatitude() > 0 && location.getLongitude() > 0) {
            double latitude = location.getLatitude();
            double longitude = location.getLongitude();
            Log.e("MapLastLocationUtils", "Google-API-获取到的最后的地址为:" + "Lat:" + latitude + " Lon:" + longitude);
        } else {
                //如果没有值直接用百度定位
        }
    });
  }

二. 集成百度定位

我们也是需要在百度开发者平台上申请应用的App,不过这一块就比较简单,只需要验证包名就行了,然后会给你一个AppId。

我们下载定位的jar包与so动态库,导入到项目,然后再清单文件配置

              <meta-data
           android:name="com.baidu.lbsapi.API_KEY"
            android:value="123456789">
        </meta-data>
        <service
            android:name="com.baidu.location.f"
            android:enabled="true"
            android:process=":remote">
            <intent-filter>
                <action android:name="com.baidu.location.service_v2.2"></action>
            </intent-filter>
        </service>

Application中初始化SDK

    SDKInitializer.initialize(context);

判断开启谷歌定位还是百度定位的逻辑封装

public class MapLastLocationUtils implements GoogleApiClient.ConnectionCallbacks, GoogleApiClient.OnConnectionFailedListener,
        LifecycleObserver {
    private static MapLastLocationUtils ourInstance;
    private Activity mActivity;
    private GoogleApiClient googleApiClient;
    private LocationClient mBDLocationClient;
    private LocationRequest mLocationRequest;
    private boolean isNeedBackNotifyLocation = false;
    public static MapLastLocationUtils getInstance(Activity context) {
        if (ourInstance == null) {
            ourInstance = new MapLastLocationUtils(context);
        }
        return ourInstance;
    }
    private MapLastLocationUtils(Activity context) {
        mActivity = context;
        //初始化谷歌地图API
        if (googleApiClient == null) {
            googleApiClient = new GoogleApiClient.Builder(mActivity)
                    .addConnectionCallbacks(this)
                    .addOnConnectionFailedListener(this)
                    .addApi(LocationServices.API)
                    .build();
        }
    }
    /**
     * 只是获取当前的位置一次
     */
    public void getLastLocation() {
        int code = GoogleApiAvailability.getInstance().isGooglePlayServicesAvailable(CommUtils.getContext());
        if (code == ConnectionResult.SUCCESS) {
            // 支持Google服务
            getAPILastLocation();
        } else {
            //开启百度定位
            getBDLastLocation();
        }
    }
    //API连接成功尝试获取最后的位置
    @SuppressLint("MissingPermission")
    private void getAPILastLocation() {
        FusedLocationProviderClient fusedLocationProviderClient = LocationServices.getFusedLocationProviderClient(mActivity);
        fusedLocationProviderClient.getLastLocation().addOnSuccessListener(mActivity, location -> {
            if (location != null && location.getLatitude() > 0 && location.getLongitude() > 0) {
                double latitude = location.getLatitude();
                double longitude = location.getLongitude();
                Log.e("MapLastLocationUtils", "Google-API-获取到的最后的地址为:" + "Lat:" + latitude + " Lon:" + longitude);
                if (mListener != null) {
                    //这里转换一下,如果国内转换为火星坐标,国外直接用
                    Gps gps = GPS84ToGCJ02(longitude, latitude);
                    mListener.onLastLocation(gps.getLatitude(), gps.getLongitude());
                }
            } else {
                getBDLastLocation();
            }
        });
    }
    //百度定位尝试获取最后的位置
    private void getBDLastLocation() {
        mBDLocationClient = new LocationClient(mActivity);
        //声明LocationClient类
        mBDLocationClient.registerLocationListener(mBDLastLocationListener);
        //配置百度定位的选项
        LocationClientOption option = new LocationClientOption();
        option.setLocationMode(LocationClientOption.LocationMode.Hight_Accuracy); // 可选,默认高精度,设置定位模式,高精度,低功耗,仅设备
        option.setCoorType("gcj02"); // 可选,默认gcj02,设置返回的定位结果坐标系,如果配合百度地图使用,建议设置为bd09ll; 国际WGS84
        option.setScanSpan(0); // 可选,默认0,即仅定位一次,设置发起连续定位请求的间隔需要大于等于1000ms才是有效的
        option.setIsNeedAddress(false); // 可选,设置是否需要地址信息,默认不需要
        option.setIsNeedLocationDescribe(false); // 可选,设置是否需要地址描述
        option.setNeedDeviceDirect(false); // 可选,设置是否需要设备方向结果
        option.setLocationNotify(false); // 可选,默认false,设置是否当gps有效时按照1S1次频率输出GPS结果
        option.setIgnoreKillProcess(true); // 可选,默认true,定位SDK内部是一个SERVICE,并放到了独立进程,设置是否在stop
        option.setIsNeedLocationDescribe(false); // 可选,默认false,设置是否需要位置语义化结果,可以在BDLocation
        option.setIsNeedLocationPoiList(false); // 可选,默认false,设置是否需要POI结果,可以在BDLocation
        option.SetIgnoreCacheException(false); // 可选,默认false,设置是否收集CRASH信息,默认收集
        option.setOpenGps(false); // 可选,默认false,设置是否开启Gps定位
        option.setIsNeedAltitude(false); // 可选,默认false,设置定位时是否需要海拔信息,默认不需要,除基础定位版本都可用
        mBDLocationClient.setLocOption(option);
        //开启定位
        if (mBDLocationClient != null && !mBDLocationClient.isStarted()) {
            mBDLocationClient.start();
        }
    }
    //Google-Location-Api 服务连接成功回调
    @Override
    public void onConnected(@Nullable Bundle bundle) {
        Log.e("MapLastLocationUtils", "Google-Api服务连接成功了");
        getAPILastLocation();
    }
    //Google-Location-Api 服务异常连接暂停
    @Override
    public void onConnectionSuspended(int i) {
        Log.e("MapLastLocationUtils", "Google-Api服务被暂停了");
        if (googleApiClient != null && !googleApiClient.isConnected())
            googleApiClient.connect();
    }
    //Google-Location-Api 连接异常
    @Override
    public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
        Log.e("MapLastLocationUtils", "Google-Api服务连接错误");
        //如果连接错误直接用百度定位吧
        getBDLastLocation();
    }
    /**
     * 停止全部的定位
     */
    public void stopLocation() {
        if (googleApiClient != null && googleApiClient.isConnected()) {
            googleApiClient.disconnect();
        }
        if (mBDLocationClient != null) {
            if (mBDLastLocationListener != null) {
                mBDLocationClient.unRegisterLocationListener(mBDLastLocationListener);
            }
            YYLogUtils.w("定位stop了");
            mBDLocationClient.stop();
        }
    }
    private OnLocationCallbackListener mListener;
    public void setOnLocationCallbackListener(OnLocationCallbackListener listener) {
        mListener = listener;
    }
    public interface OnLocationCallbackListener {
        void onLastLocation(double lat, double lon);
    }
    /**
     * 百度的监听回调(最后的地址)
     */
    BDAbstractLocationListener mBDLastLocationListener = new BDAbstractLocationListener() {
        //百度定位成功的展示
        @Override
        public void onReceiveLocation(BDLocation bdLocation) {
            if (bdLocation != null && bdLocation.getLocType() != BDLocation.TypeServerError &&
                    !(bdLocation.getLatitude() + "").equals("4.9E-324") &&
                    !(bdLocation.getLongitude() + "").equals("4.9E-324")) {
                double latitude = bdLocation.getLatitude();    //获取纬度信息
                double longitude = bdLocation.getLongitude();    //获取经度信息
                Log.w("MapLastLocationUtils", "百度定位获取到的最后的地址为:" + "Lat:" + latitude + " Lon:" + longitude);
                //获取到了最后的位置-直接回调
                if (mListener != null) {
                    mListener.onLastLocation(latitude, longitude);
                }
            } else {
                Log.e("MapLastLocationUtils", "百度定位-获取位置失败");
                if (googleApiClient != null && !googleApiClient.isConnected()) {
                    googleApiClient.connect();
                } else {
                    getAPILastLocation();
                }
            }
        }
        @Override
        public void onConnectHotSpotMessage(String s, int i) {
            super.onConnectHotSpotMessage(s, i);
        }
        //百度定位的错误信息展示
        @Override
        public void onLocDiagnosticMessage(int locType, int diagnosticType, String diagnosticMessage) {
            super.onLocDiagnosticMessage(locType, diagnosticType, diagnosticMessage);
            int tag = 2;
            StringBuffer sb = new StringBuffer(256);
            sb.append("诊断结果: ");
            if (locType == BDLocation.TypeNetWorkLocation) {
                if (diagnosticType == 1) {
                    sb.append("网络定位成功,没有开启GPS,建议打开GPS会更好");
                    sb.append("
" + diagnosticMessage);
                } else if (diagnosticType == 2) {
                    sb.append("网络定位成功,没有开启Wi-Fi,建议打开Wi-Fi会更好");
                    sb.append("
" + diagnosticMessage);
                }
            } else if (locType == BDLocation.TypeOffLineLocationFail) {
                if (diagnosticType == 3) {
                    sb.append("定位失败,请您检查您的网络状态");
                    sb.append("
" + diagnosticMessage);
                }
            } else if (locType == BDLocation.TypeCriteriaException) {
                if (diagnosticType == 4) {
                    sb.append("定位失败,无法获取任何有效定位依据");
                    sb.append("
" + diagnosticMessage);
                } else if (diagnosticType == 5) {
                    sb.append("定位失败,无法获取有效定位依据,请检查运营商网络或者Wi-Fi网络是否正常开启,尝试重新请求定位");
                    sb.append(diagnosticMessage);
                } else if (diagnosticType == 6) {
                    sb.append("定位失败,无法获取有效定位依据,请尝试插入一张sim卡或打开Wi-Fi重试");
                    sb.append("
" + diagnosticMessage);
                } else if (diagnosticType == 7) {
                    sb.append("定位失败,飞行模式下无法获取有效定位依据,请关闭飞行模式重试");
                    sb.append("
" + diagnosticMessage);
                } else if (diagnosticType == 9) {
                    sb.append("定位失败,无法获取任何有效定位依据");
                    sb.append("
" + diagnosticMessage);
                }
            } else if (locType == BDLocation.TypeServerError) {
                if (diagnosticType == 8) {
                    sb.append("定位失败,请确认您定位的开关打开状态,是否赋予APP定位权限");
                    sb.append("
" + diagnosticMessage);
                }
            }
            Log.e("MapLastLocationUtils", sb.toString());
        }
    };
    // =======================  update begin ↓ =========================
    /**
     * 开启后台的通知栏定位
     */
    public void startBackNotifyLocation() {
        isNeedBackNotifyLocation = true;
        if (mBDLocationClient == null) {
            mBDLocationClient = new LocationClient(mActivity);
        }
        LocationClientOption mOption = new LocationClientOption();
        mOption.setScanSpan(15000);
        mOption.setIsNeedAddress(false);
        mOption.setOpenGps(true);
        mBDLocationClient.setLocOption(mOption);
//                mClient.registerLocationListener(mBDUpdateLocationListener);
        Notification notification;
        if (Build.VERSION.SDK_INT >= 26) {
            NotificationUtils mNotificationUtils = new NotificationUtils(mActivity);
            Notification.Builder builder2 = mNotificationUtils.getAndroidChannelNotification("YY Circle", "Locating in the background");
            builder2.setSmallIcon(R.drawable.ic_launcher_map);
            notification = builder2.build();
        } else {
            //获取一个Notification构造器
            Notification.Builder builder = new Notification.Builder(mActivity);
            builder.setContentTitle("YY Circle") // 设置下拉列表里的标题
                    .setSmallIcon(R.drawable.ic_launcher_map) // 设置状态栏内的小图标
                    .setContentText("Locating in the background") // 设置上下文内容
                    .setWhen(System.currentTimeMillis()); // 设置该通知发生的时间
            notification = builder.build(); // 获取构建好的Notification
        }
        notification.defaults = Notification.DEFAULT_SOUND; //设置为默认的声音
        //开启前台通知的后台定位
        mBDLocationClient.enableLocInForeground(1, notification);
        mBDLocationClient.start();
    }
    // =======================  转换 begin ↓ =========================
    private double pi = 3.1415926535897932384626;
    private double a = 6378245.0;
    private double ee = 0.00669342162296594323;
    /**
     * 国际 GPS84 坐标系
     * 转换成
     * [国测局坐标系] 火星坐标系 (GCJ-02)
     * <p>
     * World Geodetic System ==> Mars Geodetic System
     *
     * @param lon 经度
     * @param lat 纬度
     * @return GPS实体类
     */
    public Gps GPS84ToGCJ02(double lon, double lat) {
        if (outOfChina(lat, lon)) {
            YYLogUtils.e("GPS84ToGCJ02:国外的坐标不做处理");
        return new Gps(lon, lat);
        } else {
            YYLogUtils.e("GPS84ToGCJ02:处理国内的坐标");
            double dLat = transformLat(lon - 105.0, lat - 35.0);
            double dLon = transformLon(lon - 105.0, lat - 35.0);
            double radLat = lat / 180.0 * pi;
            double magic = Math.sin(radLat);
            magic = 1 - ee * magic * magic;
            double sqrtMagic = Math.sqrt(magic);
            dLat = (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * pi);
            dLon = (dLon * 180.0) / (a / sqrtMagic * Math.cos(radLat) * pi);
            double mgLat = lat + dLat;
            double mgLon = lon + dLon;
            return new Gps(mgLon, mgLat);
        }
    }
    /**
     * [国测局坐标系] 火星坐标系 (GCJ-02)
     * 转换成
     * 国际 GPS84 坐标系
     *
     * @param lon 火星经度
     * @param lat 火星纬度
     */
    public Gps GCJ02ToGPS84(double lon, double lat) {
        Gps gps = transform(lon, lat);
        double lontitude = lon * 2 - gps.getLongitude();
        double latitude = lat * 2 - gps.getLatitude();
        return new Gps(lontitude, latitude);
    }
    private Gps transform(double lon, double lat) {
        if (outOfChina(lon, lat)) {
            return new Gps(lon, lat);
        }
        double dLat = transformLat(lon - 105.0, lat - 35.0);
        double dLon = transformLon(lon - 105.0, lat - 35.0);
        double radLat = lat / 180.0 * pi;
        double magic = Math.sin(radLat);
        magic = 1 - ee * magic * magic;
        double sqrtMagic = Math.sqrt(magic);
        dLat = (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * pi);
        dLon = (dLon * 180.0) / (a / sqrtMagic * Math.cos(radLat) * pi);
        double mgLat = lat + dLat;
        double mgLon = lon + dLon;
        return new Gps(mgLon, mgLat);
    }
    /**
     * 不在中国范围内
     *
     * @param lon 经度
     * @param lat 纬度
     * @return boolean值
     */
    public static boolean outOfChina(double lat, double lon) {
        if (lon < 72.004 || lon > 137.8347)
            return true;
        if (lat < 0.8293 || lat > 55.8271)
            return true;
        return false;
    }
    /**
     * 纬度转化算法
     *
     * @param x
     * @param y
     * @return
     */
    private double transformLat(double x, double y) {
        double ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y
                + 0.2 * Math.sqrt(Math.abs(x));
        ret += (20.0 * Math.sin(6.0 * x * pi) + 20.0 * Math.sin(2.0 * x * pi)) * 2.0 / 3.0;
        ret += (20.0 * Math.sin(y * pi) + 40.0 * Math.sin(y / 3.0 * pi)) * 2.0 / 3.0;
        ret += (160.0 * Math.sin(y / 12.0 * pi) + 320 * Math.sin(y * pi / 30.0)) * 2.0 / 3.0;
        return ret;
    }
    /**
     * 经度转化算法
     *
     * @param x
     * @param y
     * @return
     */
    private double transformLon(double x, double y) {
        double ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1
                * Math.sqrt(Math.abs(x));
        ret += (20.0 * Math.sin(6.0 * x * pi) + 20.0 * Math.sin(2.0 * x * pi)) * 2.0 / 3.0;
        ret += (20.0 * Math.sin(x * pi) + 40.0 * Math.sin(x / 3.0 * pi)) * 2.0 / 3.0;
        ret += (150.0 * Math.sin(x / 12.0 * pi) + 300.0 * Math.sin(x / 30.0
                * pi)) * 2.0 / 3.0;
        return ret;
    }
}

三. 坐标系转换与地图上的显示

网上一搜坐标系那可太多了,我们不整那么复杂,我们Android就知道2个 gcj02(中国国测局) 和 wgs84(通过坐标系) 。由于我们不是国内的应用,并且集成了非单一的定位软件,所以我们就不知道私有的坐标系如bd09之类的。

可以看到我们上面的工具类都是使用的gcj02坐标系定位的,在中国就需要gcj02定位,这样比较准确,如果要在对应的地图上展示,则需要对应的坐标系经纬度,在对应的坐标系地图上展示。

如我们的定位是gcj02的坐标系,在谷歌地图上展示则需要转换为wgs之后才对,不然会有大概800米的误差。百度地图则可以设置坐标系,设置为gcj02之后就可以在百度地图上展示。

坐标系的转换工具类在上面就有了。