不说废话,直接说说实现android定位有关的API吧。

这些API都在android.location包下,一共有三个接口和八个类。它们配合使用即可实现定位功能。

三个接口

GpsStatus.Listener:这是一个当GPS状态发生改变时,用来接收通知的接口。

GpsStatus.NmeaListener:这是一个用来从GPS里接收Nmea-0183(为海用电子设备制定的标准格式)信息的接口。

LocationListener:位置监听器,用于接收当位置信息发生改变时从LocationManager接收通知的接口。

八个类

Address:描述地址的类,比如:北京天安门

Criteria:用于描述Location Provider标准的类,标准包括位置精度水平,电量消耗水平,是否获取海拔、方位信息,是否允许接收付费服务。

GeoCoder:用于处理地理位置的编码。

GpsSatellite:和GpsStatus联合使用,用于描述当前GPS卫星的状态。

GpsStatus:和GpsStatus.Listener联合使用,用于描述当前GPS卫星的状态。

Location:用于描述位置信息。

LocationManager:通过此类获取和调用系统位置服务

LocationProvider:用于描述Location Provider的抽象超类,一个LocationProvider应该能够周期性的报告当前设备的位置信息。

这里通过一个示例代码来演示一下android定位

首先,在AndroidManifest.xml清单文件里需要加入ACCESS_FINE_LOCATION权限:

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

其次,实现代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
package  com.test;
 
import  java.io.IOException;
import  java.util.List;
 
import  android.app.Activity;
import  android.location.Address;
import  android.location.Criteria;
import  android.location.Geocoder;
import  android.location.Location;
import  android.location.LocationListener;
import  android.location.LocationManager;
import  android.os.Bundle;
import  android.util.Log;
import  android.widget.Toast;
 
public  class  PositionActivity  extends  Activity {
     @Override
     public  void  onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.main);
         // 获取到LocationManager对象
         LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
         // 创建一个Criteria对象
         Criteria criteria =  new  Criteria();
         // 设置粗略精确度
         criteria.setAccuracy(Criteria.ACCURACY_COARSE);
         // 设置是否需要返回海拔信息
         criteria.setAltitudeRequired( false );
         // 设置是否需要返回方位信息
         criteria.setBearingRequired( false );
         // 设置是否允许付费服务
         criteria.setCostAllowed( true );
         // 设置电量消耗等级
         criteria.setPowerRequirement(Criteria.POWER_HIGH);
         // 设置是否需要返回速度信息
         criteria.setSpeedRequired( false );
         // 根据设置的Criteria对象,获取最符合此标准的provider对象 41
         String currentProvider = locationManager
                 .getBestProvider(criteria,  true );
         Log.d( "Location" "currentProvider: "  + currentProvider);
         // 根据当前provider对象获取最后一次位置信息 44
         Location currentLocation = locationManager
                 .getLastKnownLocation(currentProvider);
         // 如果位置信息为null,则请求更新位置信息 46
         if  (currentLocation ==  null ) {
             locationManager.requestLocationUpdates(currentProvider,  0 0 ,
                     locationListener);
         }
         // 直到获得最后一次位置信息为止,如果未获得最后一次位置信息,则显示默认经纬度 50
         // 每隔10秒获取一次位置信息 51
         while  ( true ) {
             currentLocation = locationManager
                     .getLastKnownLocation(currentProvider);
             if  (currentLocation !=  null ) {
                 Log.d( "Location" "Latitude: "  + currentLocation.getLatitude());
                 Log.d( "Location" "location: "  + currentLocation.getLongitude());
                 break ;
             else  {
                 Log.d( "Location" "Latitude: "  0 );
                 Log.d( "Location" "location: "  0 );
             }
             try  {
                 Thread.sleep( 10000 );
             catch  (InterruptedException e) {
                 Log.e( "Location" , e.getMessage());
             }
         }
         // 解析地址并显示 69
         Geocoder geoCoder =  new  Geocoder( this );
         try  {
             int  latitude = ( int ) currentLocation.getLatitude();
             int  longitude = ( int ) currentLocation.getLongitude();
             List<Address> list = geoCoder.getFromLocation(latitude, longitude,
                     2 );
             for  ( int  i =  0 ; i < list.size(); i++) {
                 Address address = list.get(i);
                 Toast.makeText(
                         PositionActivity. this ,
                         address.getCountryName() + address.getAdminArea()
                                 + address.getFeatureName(), Toast.LENGTH_LONG)
                         .show();
             }
         catch  (IOException e) {
             Toast.makeText(PositionActivity. this , e.getMessage(),
                     Toast.LENGTH_LONG).show();
         }
     }
     // 创建位置监听器 85
     private  LocationListener locationListener =  new  LocationListener() {
         // 位置发生改变时调用 87
         @Override
         public  void  onLocationChanged(Location location) {
             Log.d( "Location" "onLocationChanged" );
             Log.d( "Location" ,
                     "onLocationChanged Latitude"  + location.getLatitude());
             Log.d( "Location" ,
                     "onLocationChanged location"  + location.getLongitude());
         }
 
         // provider失效时调用 95
         @Override
         public  void  onProviderDisabled(String provider) {
             Log.d( "Location" "onProviderDisabled" );
         }
 
         // provider启用时调用101
         @Override
         public  void  onProviderEnabled(String provider) {
             Log.d( "Location" "onProviderEnabled" );
         }
 
         // 状态改变时调用107
         @Override
         public  void  onStatusChanged(String provider,  int  status, Bundle extras) {
             Log.d( "Location" "onStatusChanged" );
         }
     };
}

由于代码里的Criteria对象对位置精度要求并不高,所以一般会返回“network”作为provider,而基于network的定位往往会存在一定的位置偏差,这对于需要精确定位的应用程序来说,显然不合要求。这时,需要则需要用到基于GPS的定位方法了。具体请接着看文Android定位功能(二)

Android定位功能(二) (转到第一部分)

在前文Android定位功能(一)中,已经大致介绍了一下在Android平台中,和定位功能相关的类,并举例获取了位置信息。但是前文是基于Criteria定制了一个标准,通过getBestProvider()方法由Android系统自动获取最符合Criteria的LocationProvider,从而实现了定位功能。这样的做法能最大限度的保证定位功能的可实现性,但是却无法保证获取到的位置信息有最大的准确度。因为除了GPS外,其他定位方式都或多或少存在着位置偏移。

在实现GPS定位前,先了解一下GPS的部分特性:

1:GPS定位需要依靠3颗或以上的卫星。

2:GPS定位受环境影响较大,在晴朗的空地上,较容易搜索到卫星,而在室内通常是无法搜索到卫星的。

3:GPS定位需要使用GPS功能模块,而GPS功能模块的耗电量是巨大的。

在Android系统中,实现GPS定位的思路大致是:

1、获取GPS的Location Provider。

2、将此Provider传入到requestLocationUpdates()方法,让Android系统获知搜索位置方式。

3、创建实现了GpsStatus.Listener接口的对象,重写onGpsStatusChanged()方法,向LocationManager添加次监听器,检测卫星状态。(可选步骤)

根据以上思路,仿照Android定位功能(一)中的例子,可以很容易的得到以下实现代码:(此代码的实现前提是GPS功能模块处于打开状态)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
public  class  MainActivity  extends  Activity {
     private  LocationManager locationManager;
     private  GpsStatus gpsstatus;
 
     @Override
     public  void  onCreate(Bundle savedInstanceState) {
         super .onCreate(savedInstanceState);
         setContentView(R.layout.main);
 
         // 获取到LocationManager对象
         locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);
 
         // 根据设置的Criteria对象,获取最符合此标准的provider对象
         String currentProvider = locationManager.getProvider(
                 LocationManager.GPS_PROVIDER).getName();
 
         // 根据当前provider对象获取最后一次位置信息
         Location currentLocation = locationManager
                 .getLastKnownLocation(currentProvider);
         // 如果位置信息为null,则请求更新位置信息
         if  (currentLocation ==  null ) {
             locationManager.requestLocationUpdates(currentProvider,  0 0 ,
                     locationListener);
         }
         // 增加GPS状态监听器
         locationManager.addGpsStatusListener(gpsListener);
 
         // 直到获得最后一次位置信息为止,如果未获得最后一次位置信息,则显示默认经纬度
         // 每隔10秒获取一次位置信息
         while  ( true ) {
             currentLocation = locationManager
                     .getLastKnownLocation(currentProvider);
             if  (currentLocation !=  null ) {
                 Log.d( "Location" "Latitude: "  + currentLocation.getLatitude());
                 Log.d( "Location" "location: "  + currentLocation.getLongitude());
                 break ;
             else  {
                 Log.d( "Location" "Latitude: "  0 );
                 Log.d( "Location" "location: "  0 );
             }
             try  {
                 Thread.sleep( 10000 );
             catch  (InterruptedException e) {
                 Log.e( "Location" , e.getMessage());
             }
         }
     }
 
     private  GpsStatus.Listener gpsListener =  new  GpsStatus.Listener() {
         // GPS状态发生变化时触发
         @Override
         public  void  onGpsStatusChanged( int  event) {
             // 获取当前状态
             gpsstatus = locationManager.getGpsStatus( null );
             switch  (event) {
             // 第一次定位时的事件
             case  GpsStatus.GPS_EVENT_FIRST_FIX:
                 break ;
             // 开始定位的事件
             case  GpsStatus.GPS_EVENT_STARTED:
                 break ;
             // 发送GPS卫星状态事件
             case  GpsStatus.GPS_EVENT_SATELLITE_STATUS:
                 Toast.makeText(MainActivity. this , "GPS_EVENT_SATELLITE_STATUS" ,
                         Toast.LENGTH_SHORT).show();
                 Iterable<GpsSatellite> allSatellites = gpsstatus
                         .getSatellites();
                 Iterator<GpsSatellite> it = allSatellites.iterator();
                 int  count =  0 ;
                 while  (it.hasNext()) {
                     count++;
                 }
                 Toast.makeText(MainActivity. this "Satellite Count:"  + count,
                         Toast.LENGTH_SHORT).show();
                 break ;
             // 停止定位事件
             case  GpsStatus.GPS_EVENT_STOPPED:
                 Log.d( "Location" "GPS_EVENT_STOPPED" );
                 break ;
             }
         }
     };
 
     // 创建位置监听器
     private  LocationListener locationListener =  new  LocationListener() {
         // 位置发生改变时调用
         @Override
         public  void  onLocationChanged(Location location) {
             Log.d( "Location" "onLocationChanged" );
         }
 
         // provider失效时调用
         @Override
         public  void  onProviderDisabled(String provider) {
             Log.d( "Location" "onProviderDisabled" );
         }
 
         // provider启用时调用
         @Override
         public  void  onProviderEnabled(String provider) {
             Log.d( "Location" "onProviderEnabled" );
         }
 
         // 状态改变时调用
         @Override
         public  void  onStatusChanged(String provider,  int  status, Bundle extras) {
             Log.d( "Location" "onStatusChanged" );
         }
     };
}

通过以上代码中的注释部分,可以清晰的知道Android定位功能里相关方法的具体含义。希望对大家有用。

另外,因为GPS的自身特性,此代码在室内几乎无法定位,所以建议再真正的实际项目里,至少使用network和GPS两种不同的Location Provider实现定位功能。

本人暂时未找到同时关闭网络和GPS功能实现定位的方法,本人也未找到通过代码在没有ROOT的前提下直接代开网络和GPS功能的代码。如果大家在这两方面有自己的体会,请不吝赐教,留言评论或给出参考地址都可。大家一同探讨,一同进步。

Android定位功能的更多相关文章

  1. Android定位功能(二)

    在前文Android定位功能(一)中,已经大致介绍了一下在Android平台中,和定位功能相关的类,并举例获取了位置信息.但是前文是基于Criteria定制了一个标准,通过getBestProvide ...

  2. android 百度地图定位功能实现

    历经几天时间,终于把定位功能给实现了,可谓是费劲千辛万苦啊,有定位知识还有图层知识,在这里我把代码给大家贴出来,一起分享一下下啦. package com.example.foreveross.off ...

  3. Android 百度地图开发(二)--- 定位功能之MyLocationOverlay,PopupOverlay的使用

    转载请注明出处http://blog.csdn.net/xiaanming/article/details/11380619 这一篇文章主要讲解的是百度地图的定位功能,然后还有MyLocationOv ...

  4. Android定位&地图&导航——基于百度地图实现的定位功能

    一.问题描述 LBS位置服务是android应用中重要的功能,应用越来越广泛,下面我们逐步学习和实现lbs相关的应用如定位.地图.导航等,首先我们看如何基于百度地图实现定位功能 二.配置环境 1.注册 ...

  5. Flex AIR应用GPS定位功能(Android和IOS)

    说明: 使用AIR进行GPS定位功能实现时,会经常判断GPS是否打开.一般的官方或者书上的介绍的方法,测试后,只能对Android系统进行判断,而对ios系统则无法进行判断. 经过研究测试,终于解决实 ...

  6. [译]:Xamarin.Android平台功能——位置服务

    返回索引目录 原文链接:Location Services. 译文链接:Xamarin.Android平台功能--位置服务 本部分介绍位置服务以及与如何使用位置提供商服务 Location Servi ...

  7. Android定位方式和测试方法

    Android常用的三种定位方式有:基于GPS定位.基于基站地位.基于wifi定位. 1.基于GPS定位: GPS定位需要GPS模块(硬件)的支持,没有GPS模块是无法进行GPS定位的. GPS定位最 ...

  8. ionic 添加地图定位功能

    由于项目需求,需要一个定位功能,通过google或百度,搜到一个cordova-plugin-geolocation的插件,在ios上可以用,但是在android就呵呵了,原因就不说了,大家都知道.所 ...

  9. android 定位的四种方式

    [原文]  开发中对于地图及地理位置的定位是我们经常要用地,地图功能的使用使得我们应用功能更加完善,下面总结了一下网络中现有对于介绍android定位的4种方式,希望对大家有帮助: android 定 ...

随机推荐

  1. android studio下的NDK开发详解(一)

    源地址:http://www.voidcn.com/blog/chengkaizone/article/p-5761016.html 好记性不如烂笔头,开始坚持写博客,学一点记一点,只为了生活更好. ...

  2. RT3070 USB WIFI 在连接socket编程过程中问题总结

    最近耗时多天,成功的将RT3070驱动.并解决了socket的网络编程,成功的在BA9G10上面实现了USB wif.连上家里的无线路由器,通过ubuntu下面建立的服务端程序,将BA9G10中的数据 ...

  3. Android 异步链式调用设计

    本文讨论一下异步链式调用的设计与实现. 考虑如下情况: 情况1: 访问网络(或其他耗时的事情).通常的做法是: 1.显示一个ProgressDialog对话框,提示用户. 2.启动工作线程来执行耗时操 ...

  4. 黑马程序员:Java基础总结----静态代理模式&动态代理

    黑马程序员:Java基础总结 静态代理模式&动态代理   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 静态代理模式 public  class  Ts {   ...

  5. python之enumerate()函数的探究

    原地址:http://www.newsmth.net/nForum/#!article/Python/95860 最近用到enumerate()函数,于是查了相关的资料.           其中的一 ...

  6. c#中的jQuery——HtmlAgilityPack

    原文:c#中的jQuery--HtmlAgilityPack c#中是否有javascript中的jQuery类库? jQuery在访问和操作HTML 的DOM的便捷是前端开发工程师的一种福音,在c# ...

  7. leetcode回文子串拆分-最小拆分次数

    转载请注明来自souldak,微博:@evagle 上一篇是要输出所有的可能拆分,这回是要输出拆分次数最少的切割次数. 如果直接按照上一篇那么做的话,就会超时,因为我们在判断s[i][j]是否是回文的 ...

  8. linux signal之初学篇

    前言 本博文只总结signal的应用,对signal的kernel实现暂不讨论. 1. linux signal是什么? signal是linux提供的用于进程间通信的一种IPC机制. 2. 如何发送 ...

  9. cocos2dx游戏开发学习笔记3-lua面向对象分析

    在lua中,能够通过元表来实现类.对象.继承等.与元表相关的方法有setmetatable().__index.getmetatable().__newindex. 详细什么是元表在这里就不细说了,网 ...

  10. HDU ACM 1054 Strategic Game 二分图最小顶点覆盖?树形DP

    分析:这里使用树形DP做. 1.最小顶点覆盖做法:最小顶点覆盖 == 最大匹配(双向图)/2. 2.树形DP: dp[i][0]表示i为根节点,而且该节点不放,所需的最少的点数. dp[i][1]表示 ...