彻底解决Android GPS没法定位这一顽固问题

判断GPS模块是否正常

private void openGPSSettings() {
    LocationManager alm = (LocationManager) this
            .getSystemService(Context.LOCATION_SERVICE);
    if (alm.isProviderEnabled(android.location.LocationManager.GPS_PROVIDER)) {
        Toast.makeText(this, "GPS模块正常", Toast.LENGTH_SHORT).show();
        baiDuMapManager.startLocation();
        return;
    } else {
        Toast.makeText(this, "请开启GPS!", Toast.LENGTH_SHORT).show();
        Intent intent = new Intent(Settings.ACTION_SECURITY_SETTINGS);
        startActivityForResult(intent, 0); // 此为设置完成后返回到获取界面
    }

}

加下百度地图下service,解决小米手机定位不到问题

<service
    android:name="com.baidu.location.f"
    android:enabled="true"
    android:process=":remote" >
</service>

大家去网上搜索Android定位location为null没法定位问题,估计有一大堆文章介绍如何来解决,但是最后大家发现基本没用。本文将从Android定位实现原理来深入分析没法定位原因并提出真正的解决方案。在分析之前,我们肯定得先看看android官方提供的定位SDK。

默认Android GPS定位实例

        获取LocationManager:
mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE);  
         选择Location Provider:

Android系统存在多种provider,分别是

GPS_PROVIDER:
这个就是手机里有GPS芯片,然后利用该芯片就能利用卫星获得自己的位置信息。但是在室内,GPS定位基本没用,很难定位的到。

NETWORK_PROVIDER:
这个就是利用网络定位,通常是利用手机基站和WIFI节点的地址来大致定位位置,这种定位方式取决于服务器,即取决于将基站或WIF节点信息翻译成位置信息的服务器的能力。由于目前大部分Android手机没有安装google官方的location manager库,大陆网络也不允许,即没有服务器来做这个事情,自然该方法基本上没法实现定位。

PASSIVE_PROVIDER:
被动定位方式,这个意思也比较明显,就是用现成的,当其他应用使用定位更新了定位信息,系统会保存下来,该应用接收到消息后直接读取就可以了。比如如果系统中已经安装了百度地图,高德地图(室内可以实现精确定位),你只要使用它们定位过后,再使用这种方法在你的程序肯定是可以拿到比较精确的定位信息。用户可以直接指定某一个provider

String provider = mLocationManager.getProvider(LocationManager.GPS_PROVIDER);  

也可以提供配置,由系统根据用户的配置为用户选择一个最接近用户需求的provider

Criteria crite = new Criteria();    

crite.setAccuracy(Crite.ACCURACY_FINE); //精度  

crite.setPowerRequirement(Crite.POWER_LOW); //功耗类型选择  

String provider = mLocationManager.getBestProvider(crite, true);   
        获取Location
Location location = mLocationManager.getLocation(provider);    

然后你会发现,这个返回的location永远为null,你自然没法定位。然后网上到处是咨询为啥获得的location为null,同样网络到处是解决这个问题的所谓解决方案。

所谓解决方案

网上有人说,一开始location是很有可能是null的,这是因为程序还从来没有请求 过,只需重新请求更新location,并注册监听器以接收更新后的location信息。

LocationListener locationListener = new LocationListener() {  
        @Override  
        public void onStatusChanged(String provider, int status, Bundle extras) {  
        }  
        @Override  
        public void onProviderEnabled(String provider) {  
        }  
  
        @Override  
        public void onProviderDisabled(String provider) {  
        }  
  
        @Override  
        public void onLocationChanged(Location location) {  
            longitude = location.getLongitude();  
            latitude  = location.getLatitude();  
            Log.d(TAG,"Location longitude:"+ longitude +" latitude: "+ latitude );  
        }  
        
};  
mLocationManager.requestLocationUpdates(serviceProvider, 10000, 1, this);  

然后你发现onLocationChanged永远不会被调用,你仍然没法获取定位信息。

为什么就没法获取到location呢?

其实在上面我已经提到了,所有上面的解决的方案都没有解决根本问题,那就是当你在室内开发时,你的手机根本就没法获取位置信息,你叫系统如何将位置信息通知给你的程序。所以要从根本上解决这个问题,就要解决位置信息获取问题。刚刚也提到了,只有NETWORK_PROVIDER这种模式才是室内定位可靠的方式,只不过由于大陆的怪怪网络,且大部分厂商也不会用google的服务,这种定位方式默认是没法用的。那怎么办?好办,找个替代的服务商就可以了,百度的位置信息sdk就可以解决这个问题。它的基本原理在上面已经提到过了,就是搜集你的wifi节点信息和你的手机基站信息来定位。

真正的解决方案,使用百度位置定位SDK

        SDK下载:

        http://pan.baidu.com/s/1i3xGMih

        当然大家可以在官网下载,这样可以下载到最新的sd

        http://lbsyun.baidu.com/sdk/download

        SDK使用:
1.  申请百度的服务密钥,具体操作步骤见官网:
     http://api.map.baidu.com/lbsapi/cloud/geosdk.htm
     
2.将上面下载的sdk文件locSDK_4.1.jar拷贝到你项目的libs下

3.  修改AndroidManifest文件,在该文件里添加如下配置
<service  
           android:name="com.baidu.location.f"  
           android:enabled="true"  
           android:process=":remote" >  
       </service>  
       <meta-data  
           android:name="com.baidu.lbsapi.API_KEY"  
           android:value="xxxxx " />  
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />  
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />  
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />  

上面meta-data中value的值改为你自己的密钥

         代码里调用sdk:
public class LocationUtil {  
    private final static boolean DEBUG = true;  
    private final static String TAG = "LocationUtil";  
    private static LocationUtil mInstance;  
    private BDLocation mLocation = null;  
    private MLocation  mBaseLocation = new MLocation();  
  
    public static LocationUtil getInstance(Context context) {  
        if (mInstance == null) {  
            mInstance = new LocationUtil(context);  
        }  
        return mInstance;  
    }  
  
    Context mContext;  
    String mProvider;  
    public BDLocationListener myListener = new MyLocationListener();  
    private LocationClient mLocationClient;  
      
    public LocationUtil(Context context) {  
        mLocationClient = new LocationClient(context.getApplicationContext());  
        initParams();  
        mLocationClient.registerLocationListener(myListener);  
    }  
  
    public void startMonitor() {  
        if (DEBUG) Log.d(TAG, "start monitor location");  
        if (!mLocationClient.isStarted()) {  
            mLocationClient.start();  
        }  
        if (mLocationClient != null && mLocationClient.isStarted()) {  
            mLocationClient.requestLocation();  
        } else {  
             Log.d("LocSDK3", "locClient is null or not started");  
        }  
    }  
  
    public void stopMonitor() {  
        if (DEBUG) Log.d(TAG, "stop monitor location");  
        if (mLocationClient != null && mLocationClient.isStarted()) {  
            mLocationClient.stop();  
        }  
    }  
      
    public BDLocation getLocation() {  
        if (DEBUG) Log.d(TAG, "get location");  
        return mLocation;  
    }  
      
    public MLocation getBaseLocation() {  
        if (DEBUG) Log.d(TAG, "get location");  
        return mBaseLocation;  
    }  
      
    private void initParams() {  
        LocationClientOption option = new LocationClientOption();  
        option.setOpenGps(true);  
        //option.setPriority(LocationClientOption.NetWorkFirst);  
        option.setAddrType("all");//返回的定位结果包含地址信息  
        option.setCoorType("bd09ll");//返回的定位结果是百度经纬度,默认值gcj02  
        option.setScanSpan(5000);//设置发起定位请求的间隔时间为5000ms  
        option.disableCache(true);//禁止启用缓存定位  
        option.setPoiNumber(5);    //最多返回POI个数     
        option.setPoiDistance(1000); //poi查询距离          
        option.setPoiExtraInfo(true); //是否需要POI的电话和地址等详细信息          
        mLocationClient.setLocOption(option);  
    }  
public class MyLocationListener implements BDLocationListener {  
        @Override  
        public void onReceiveLocation(BDLocation location) {  
            if (location == null) {  
                return ;  
            }  
            mLocation = location;  
            mBaseLocation.latitude = mLocation.getLatitude();  
            mBaseLocation.longitude = mLocation.getLongitude();  
            StringBuffer sb = new StringBuffer(256);  
            sb.append("time : ");  
            sb.append(location.getTime());  
            sb.append("\nerror code : ");  
            sb.append(location.getLocType());  
            sb.append("\nlatitude : ");  
            sb.append(location.getLatitude());  
            sb.append("\nlontitude : ");  
            sb.append(location.getLongitude());  
            sb.append("\nradius : ");  
            sb.append(location.getRadius());  
            sb.append("\ncity : ");  
            sb.append(location.getCity());  
            if (location.getLocType() == BDLocation.TypeGpsLocation){  
                sb.append("\nspeed : ");  
                sb.append(location.getSpeed());  
                sb.append("\nsatellite : ");  
                sb.append(location.getSatelliteNumber());  
            } else if (location.getLocType() == BDLocation.TypeNetWorkLocation){  
                sb.append("\naddr : ");  
                sb.append(location.getAddrStr());  
            }  
            if (DEBUG) Log.d(TAG, "" + sb);  
        }  
          public void onReceivePoi(BDLocation poiLocation) {  
        }  
    }        

    public class MLocation {      
        public double latitude;                  
        public double longitude;  
    } 
}  

当然别忘了在setting里将gps定位打开

  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Unlocker是一款实用的软件,可以帮助用户删除顽固文件。所谓顽固文件,指的是那些由于某些原因无法被正常删除的文件。比如,该文件正在被其他程序占用,或者用户没有删除该文件的权限等等。 使用Unlocker删除顽固文件非常简单。用户只需双击打开Unlocker程序,在程序界面中选择要删除的文件或文件夹,然后点击“删除”按钮即可。Unlocker会自动检测文件是否被占用,并提供解除占用的选项。用户可以选择立即解锁文件,或者在下次启动系统时解锁。完成这些步骤后,Unlocker会重新尝试删除文件,通常能够成功删除问题文件。 除了删除文件外,Unlocker还提供了其他有用的功能。比如,用户可以查看文件被哪个进程占用,甚至可以终止该进程以彻底释放文件。此外,Unlocker还支持批量删除文件,用户只需选中要删除的文件批量处理即可。 需要注意的是,由于Unlocker是一款免费的软件,用户在下载和使用时要小心安全。建议从官方信誉良好的网站进行下载,并定期更新软件版本,以确保获得最新的功能和安全性。此外,如果用户对删除的文件不确定,建议先备份文件,以避免数据丢失。 总之,Unlocker是一款方便实用的软件,可以帮助用户轻松删除顽固文件。它的简单操作和多种功能使其成为解决文件无法删除问题的好帮手。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值