編輯:關於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權限:
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刮獎控件
直接上效果圖 功能特色: 1、可以設置刮開後顯示文字或圖片 2、可以統計已刮開區域所占百分比 Demo下載地址:RubberDem
Android基礎篇之Android快速入門--你必須要知道的基礎
Android快速入門 1. 搭建開發環境>解壓壓縮文件,得到:①Android SDK (類似於JDK)② Eclipse ③ADT>配置兩個pat
紅米3X怎麼預約購買 紅米手機3X預約地址介紹
小米聯合中國聯通推出了新機——紅米3X,具備4100mAh超大電池,售價899元。現在,這款手機已經在中國聯通官網開啟預約,只需輸入
Android 圖片陰影處理分析!
最近在做一個Android位圖陰影處理的效果,今天把思路總結一下:分析如下:Android sdk提供了BlurMaskFilter工具類可以給圖片添加陰影效果,代碼如下