android定位功能
来源:互联网 发布:java多线程解压zip文件 编辑:程序博客网 时间:2024/06/05 09:27
不说废话,直接说说实现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权限:
<
uses-permission
android:name
=
"android.permission.ACCESS_FINE_LOCATION"
></
uses-permission
>
其次,实现代码如下:
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功能模块处于打开状态)
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定位功能
- Android定位功能
- Android定位功能学习
- Android定位功能
- Android定位功能总结
- android定位功能1
- Android定位功能(二)
- Android定位功能(一)
- Android定位功能(二)
- Android定位功能(一)
- Android定位功能(二)
- Android定位功能(一)
- Android定位功能(二)
- Android 百度 Map----定位功能
- Android定位功能(一)
- 数据库中删除语句Drop、Delete、Truncate的相同点和不同点的比较(举例说明)
- u-boot命令行接口
- oracle变量绑定
- 12 线性区的底层处理
- Oracle Data Types - NVARCHAR2 (zz)
- android定位功能
- HDU 1166 树状数组 基础题
- 13 分配线性地址区间
- 浅谈HTTP中Get与Post的区别
- Qt下 QString转char*
- 使用TextView/EditText应该注意的地方
- 14 释放线性地址区间
- DirectShow中网络播放器的源Source Filter
- 15 缺页异常处理程序