Android通過原生APi獲取所在位置的經緯度
在Android開發(fā)當中,經常需要用到定位功能,尤其是依賴于地理位置功能的應用,本文介紹了Android通過原生APi獲取所在位置的經緯度,分享給大家
一、難點介紹
1.難點
我們的應用要新增一個功能,就是在用戶打開附件的人頁面后,將用戶的經緯度通過一個接口返回給服務端,從而讓服務器可以準確定位。
因為只是添加一個小功能所以,引入第三方SDK定位就有些大材小用了,所以就準備借助原生APi:LocationManager來完成。
經過在網絡上一搜,有很多關于通過LocationManager獲取經緯度坐標的,所有的代碼幾乎大同小異,本著負責任的態(tài)度,就看了好多篇,然后總結出了一個比較優(yōu)良的獲取經緯度的工具類。
在驗證的過程中遇到了以下的幾個問題:
①權限問題:Android 6.0之后新增動態(tài)權限,而獲取獲取經緯度坐標的權限如果你的app的(TargetVersion>=23)就需要動態(tài)獲取了(當然也得在清單文件中設置)
②谷歌網絡服務在中國被禁,所以就會導致網絡定位在中國不可用, 只能使用GPS定位,而GPS定位需要用戶打開GPS標志才能獲取到
其中最讓我難以解決的是第二個問題。因為嘗試了很多次,只能通過打開GPS才能獲取到位置。終于在問遍各位大神后,都說了關于是因為谷歌網絡服務在中國被禁。
所幸,我們應用面對的群體是海外用戶,所以不存在谷歌服務被墻的情況。
以上就是我在過程中遇到的問題了,如果解決了你的困惑的話,那就太開心了。
關于Android6.0動態(tài)權限的話,只要寫好對應的回調就可以了,不算很難。
下面貼上我工具類的代碼給大家:
public class LocationUtils {
private volatile static LocationUtils uniqueInstance;
private LocationManager locationManager;
private String locationProvider;
private Location location;
private Context mContext;
private LocationUtils(Context context) {
mContext = context;
getLocation();
}
//采用Double CheckLock(DCL)實現單例
public static LocationUtils getInstance(Context context) {
if (uniqueInstance == null) {
synchronized (LocationUtils.class) {
if (uniqueInstance == null) {
uniqueInstance = new LocationUtils( context );
}
}
}
return uniqueInstance;
}
private void getLocation() {
//1.獲取位置管理器
locationManager = (LocationManager) mContext.getSystemService( Context.LOCATION_SERVICE );
//2.獲取位置提供器,GPS或是NetWork
List<String> providers = locationManager.getProviders( true );
if (providers.contains( LocationManager.NETWORK_PROVIDER )) {
//如果是網絡定位
Log.d( TAG, "如果是網絡定位" );
locationProvider = LocationManager.NETWORK_PROVIDER;
} else if (providers.contains( LocationManager.GPS_PROVIDER )) {
//如果是GPS定位
Log.d( TAG, "如果是GPS定位" );
locationProvider = LocationManager.GPS_PROVIDER;
} else {
Log.d( TAG, "沒有可用的位置提供器" );
return;
}
// 需要檢查權限,否則編譯報錯,想抽取成方法都不行,還是會報錯。只能這樣重復 code 了。
if (Build.VERSION.SDK_INT >= 23 &&
ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED &&
ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {
return;
}
if (ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {
return;
}
//3.獲取上次的位置,一般第一次運行,此值為null
Location location = locationManager.getLastKnownLocation( locationProvider );
if (location != null) {
setLocation( location );
}
// 監(jiān)視地理位置變化,第二個和第三個參數分別為更新的最短時間minTime和最短距離minDistace
locationManager.requestLocationUpdates( locationProvider, 0, 0, locationListener );
}
private void setLocation(Location location) {
this.location = location;
String address = "緯度:" + location.getLatitude() + "經度:" + location.getLongitude();
Log.d( TAG, address );
}
//獲取經緯度
public Location showLocation() {
return location;
}
// 移除定位監(jiān)聽
public void removeLocationUpdatesListener() {
// 需要檢查權限,否則編譯不過
if (Build.VERSION.SDK_INT >= 23 &&
ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_FINE_LOCATION ) != PackageManager.PERMISSION_GRANTED &&
ActivityCompat.checkSelfPermission( mContext, Manifest.permission.ACCESS_COARSE_LOCATION ) != PackageManager.PERMISSION_GRANTED) {
return;
}
if (locationManager != null) {
uniqueInstance = null;
locationManager.removeUpdates( locationListener );
}
}
/**
* LocationListern監(jiān)聽器
* 參數:地理位置提供器、監(jiān)聽位置變化的時間間隔、位置變化的距離間隔、LocationListener監(jiān)聽器
*/
LocationListener locationListener = new LocationListener() {
/**
* 當某個位置提供者的狀態(tài)發(fā)生改變時
*/
@Override
public void onStatusChanged(String provider, int status, Bundle arg2) {
}
/**
* 某個設備打開時
*/
@Override
public void onProviderEnabled(String provider) {
}
/**
* 某個設備關閉時
*/
@Override
public void onProviderDisabled(String provider) {
}
/**
* 手機位置發(fā)生變動
*/
@Override
public void onLocationChanged(Location location) {
location.getAccuracy();//精確度
setLocation( location );
}
};
}
用法:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate( savedInstanceState );
setContentView( R.layout.activity_main );
Button btn = (Button) findViewById( R.id.btn );
final TextView text = (TextView) findViewById( R.id.text );
btn.setOnClickListener( new View.OnClickListener() {
@Override
public void onClick(View v) {
Location location = LocationUtils.getInstance( MainActivity.this ).showLocation();
if (location != null) {
String address = "緯度:" + location.getLatitude() + "經度:" + location.getLongitude();
Log.d( "FLY.LocationUtils", address );
text.setText( address );
}
}
} );
}
@Override
protected void onDestroy() {
super.onDestroy();
LocationUtils.getInstance( this ).removeLocationUpdatesListener();
}
}
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Android編程實現Gallery中每次滑動只顯示一頁的方法
這篇文章主要介紹了Android編程實現Gallery中每次滑動只顯示一頁的方法,涉及Android擴展Gallery控件實現翻頁效果控制的功能,涉及Android事件響應及屬性控制的相關技巧,需要的朋友可以參考下2015-11-11
listView的item中有checkbox,導致setOnItemClick失效的原因及解決辦法
這篇文章主要介紹了listView的item中有checkbox,導致setOnItemClick失效的原因及解決辦法,需要的朋友可以參考下2017-01-01

