Android定位功能
不说废话,直接说说实现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定位功能的更多相关文章
- Android定位功能(二)
在前文Android定位功能(一)中,已经大致介绍了一下在Android平台中,和定位功能相关的类,并举例获取了位置信息.但是前文是基于Criteria定制了一个标准,通过getBestProvide ...
- android 百度地图定位功能实现
历经几天时间,终于把定位功能给实现了,可谓是费劲千辛万苦啊,有定位知识还有图层知识,在这里我把代码给大家贴出来,一起分享一下下啦. package com.example.foreveross.off ...
- Android 百度地图开发(二)--- 定位功能之MyLocationOverlay,PopupOverlay的使用
转载请注明出处http://blog.csdn.net/xiaanming/article/details/11380619 这一篇文章主要讲解的是百度地图的定位功能,然后还有MyLocationOv ...
- Android定位&地图&导航——基于百度地图实现的定位功能
一.问题描述 LBS位置服务是android应用中重要的功能,应用越来越广泛,下面我们逐步学习和实现lbs相关的应用如定位.地图.导航等,首先我们看如何基于百度地图实现定位功能 二.配置环境 1.注册 ...
- Flex AIR应用GPS定位功能(Android和IOS)
说明: 使用AIR进行GPS定位功能实现时,会经常判断GPS是否打开.一般的官方或者书上的介绍的方法,测试后,只能对Android系统进行判断,而对ios系统则无法进行判断. 经过研究测试,终于解决实 ...
- [译]:Xamarin.Android平台功能——位置服务
返回索引目录 原文链接:Location Services. 译文链接:Xamarin.Android平台功能--位置服务 本部分介绍位置服务以及与如何使用位置提供商服务 Location Servi ...
- Android定位方式和测试方法
Android常用的三种定位方式有:基于GPS定位.基于基站地位.基于wifi定位. 1.基于GPS定位: GPS定位需要GPS模块(硬件)的支持,没有GPS模块是无法进行GPS定位的. GPS定位最 ...
- ionic 添加地图定位功能
由于项目需求,需要一个定位功能,通过google或百度,搜到一个cordova-plugin-geolocation的插件,在ios上可以用,但是在android就呵呵了,原因就不说了,大家都知道.所 ...
- android 定位的四种方式
[原文] 开发中对于地图及地理位置的定位是我们经常要用地,地图功能的使用使得我们应用功能更加完善,下面总结了一下网络中现有对于介绍android定位的4种方式,希望对大家有帮助: android 定 ...
随机推荐
- hdu 4715 Difference Between Primes 2013年ICPC热身赛A题 素数水题
题意:给出一个偶数(不论正负),求出两个素数a,b,能够满足 a-b=x,素数在1e6以内. 只要用筛选法打出素数表,枚举查询下就行了. 我用set储存素数,然后遍历set里面的元素,查询+x后是否还 ...
- SVM入门(十)将SVM用于多类分类
源地址:http://www.blogjava.net/zhenandaci/archive/2009/03/26/262113.html 从 SVM的那几张图可以看出来,SVM是一种典型的两类分类器 ...
- Python的对象和类型
Python使用对象来存储数据,构造任何类型的值都是一个对象. 任何一个对象都有三个特性:身份,类型和值. 身份是对象的唯一标识,可以通过内建函数id()得到,这个值可以认为是该对象的内存地址. Py ...
- Static关键字的作用及使用
1.使用static声明属性 如果希望一个属性被所有对象共同拥有,可以将其声明为static类型. 声明为static类型的属性或方法,此属性或方法也被称为类方法,可以由类名直接调用. class P ...
- 设置Mysql的连接超时参数
在Mysql的默认设置中,如果一个数据库连接超过8小时没有使用(闲置8小时,即 28800s),mysql server将主动断开这条连接,后续在该连接上进行的查询操作都将失败,将 出现:e ...
- 《转》VC++多线程编程
原地址:http://www.cnblogs.com/wxfasdic/archive/2010/09/23/1833522.html 留个纪念,不错的总结.十个例子清晰列举啦多线程编程的奥妙. V ...
- 《转》Frameset布局
前二天在写一个HTML界面,用到了Frameset,主要学习都是在下面的文章里,内容写得很详细,值得推荐大家看下. 网址:http://captaincook.iteye.com/blog/36563 ...
- 基于SIFT+Kmeans+LDA的图片分类器的实现
原地址:http://www.cnblogs.com/freedomshe/archive/2012/04/24/2468747.html 题记:2012年4月1日回到家,南大计算机研究僧复试以后,等 ...
- Linux Shell脚本编程--curl命令详解
用途说明 curl命令是一个功能强大的网络工具,它能够通过http.ftp等方式下载文件,也能够上传文件.其实curl远不止前面所说的那些功能,大家可以通过man curl阅读手册页获取更多的信息.类 ...
- [Unity3D]Unity3D游戏开发之ACT游戏三连击效果实现综述
各位朋友,大家好,我是秦元培,欢迎大家关注我的博客,我的博客地址是blog.csdn.net/qinyuanpei.在研究了Unity3D Mecanim动画系统的重定向特性后,今天我们继续来探索Me ...