項目中需要對獲取到的地理位置信息(經緯度)進行反向地址編碼,就是根據經緯度反查出地址來。
首先想到的是android自帶的那個GeoCoder,測試代碼片段如下:
android.location.Geocoder geocoder = new android.location.Geocoder(this, Locale.CHINA);
try {
List<android.location.Address> addressList = geocoder.getFromLocation(location[0], location[1], 3);
for (android.location.Address address : addressList) {
Log.i(TAG, address.toString());
}
} catch (IOException e) {
e.printStackTrace();
}
注意,上面那些個長長的類名,
android.location.Geocoder和<span style="font-family: Arial, Helvetica, sans-serif;">android.location.Address</span>
你正常import之後是不需要這麼寫的,我故意如此是爲了方便和百度地圖SDK做對比,因爲百度地圖裏面也叫GeoCoder啊。。。
在浙江的地圖上隨便找一個點,得到經緯度,然後測試,輸出結果示例:
[addressLines=[0:"中國",1:"浙江省金華市東陽市"],
feature=歌山鎮新浦,admin=浙江省,sub-admin=null,locality=金華市,
thoroughfare=null,postalCode=null,countryCode=CN,countryName=中國,
hasLatitude=true,latitude=29.268876512302988,
hasLongitude=true,longitude=120.35351869072791,phone=null,url=null,extras=null
這個地方看起來可能有點偏。。。一大堆的null。
不管它,然後看看咱們的大百度。
首先呢,需要下載百度地圖SDK,因爲我只需要這個檢索的功能,我就只下載了包含地址檢索的最小庫。解壓之後兩個文件:
很明顯,jar包放入工程的libs目錄下,並且右鍵->add as library,觸發gradle的自動sync files。等待完成後,jar包導入完畢。
那個so文件呢?用過eclipse的朋友都會想當然的在libs目錄下新建一個目錄叫armeabi,然後把so文件放進去即可。
我不會告訴我就是這麼做的,然後,很不幸的,運行時爆了Unsatified 錯誤,做過NDK開發就知道,這個native庫沒加載成功。多半是so文件放的地方不對。
看來eclipse的老經驗在AndroidStudio下完全不好使。
放到哪裏呢?首先想到jni目錄,好吧,在項目(實際上叫module)上右鍵,新建一個jni folder
然後把這個so文件放進去,編譯,嗯,提示我要配置NDK的path,好吧,我承認我都是用Eclipse來測試NDK程序。但是,很明顯,這個提示說明,這個目錄也不對,這裏是放JNI源代碼的地方,這個so是編譯好的庫,正確導入肯定是不需要NDK再次編譯的。
那麼會是哪裏呢?已經有人研究出來了,正解是放到jniLibs目錄下面。
注意路徑:src->main->jniLibs->armeabi->libxxxx.so
接下來呢,就是根據百度的官方文檔,申請密鑰,添加一大堆權限,以及必要的初始化代碼等等,這些不再贅述了。
只單獨講講在使用AS的時候,關於申請密鑰時需要獲取到當前簽名key對應的SHA1指紋,這一點和Eclipse有點不同。
順便問一句:哪位大神知道AS默認使用的debugkey是什麼?在哪裏?我沒找到。。。就新建了一個key。通過菜單
Build-》Generate Signed APK 新建了一個key,比如叫做signkey.jks。
獲取key的SHA1指紋的命令:
cd 到你存放key的目錄下,執行
$ keytool -list -v -keystore signkey.jks
就可以得到SHA1指紋。
那麼怎麼讓AS在debug和release的時候都使用這個新建的key在簽名呢?,看圖
然後進入上圖中 BuildType選項卡中,給debug和release都指定我們在上面新建的配置來簽名。
OK,簽名的問題搞定!這個簽名不搞定,百度可不會讓你使用他的服務哦。
最後,測試百度地圖反向地址編碼的代碼部分:
//利用百度地圖SDK提供的方法進行地理反向編碼查詢
GeoCoder geoCoder = GeoCoder.newInstance();
ReverseGeoCodeOption option = new ReverseGeoCodeOption();
option.location(new LatLng(location[0], location[1]));
geoCoder.setOnGetGeoCodeResultListener(new OnGetGeoCoderResultListener() {
@Override
public void onGetGeoCodeResult(GeoCodeResult geoCodeResult) {
}
@Override
public void onGetReverseGeoCodeResult(ReverseGeoCodeResult reverseGeoCodeResult) {
if (reverseGeoCodeResult == null || reverseGeoCodeResult.error != SearchResult.ERRORNO.NO_ERROR) {
//沒有找到檢索結果
Log.i(TAG, "No result!!");
return;
}
<pre name="code" class="java"><span style="white-space:pre"> </span> Log.i(TAG,<span style="font-family: Arial, Helvetica, sans-serif;">reverseGeoCodeResult.getAddress());</span>
Log.i(TAG,<span style="font-family: Arial, Helvetica, sans-serif;">reverseGeoCodeResult.getBusinessCircle());</span>
List<PoiInfo> poiInfoList = reverseGeoCodeResult.getPoiList(); if (poiInfoList == null) { Log.i(TAG, "poilist null.."); return; } for (PoiInfo info : poiInfoList) { Log.i(TAG, info.toString()); Log.i(TAG, info.address);
Log.i(TAG, info.name);
Log.i(TAG, <span style="font-family: Arial, Helvetica, sans-serif;">info.phoneNum);</span>
} } }); geoCoder.reverseGeoCode(option);
有個重要的初始化代碼,一定要放在最前面執行,在setContentView()之前。
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
<span style="color:#ff0000;">SDKInitializer.initialize(getApplicationContext());</span>
setContentView(R.layout.activity_bmap_address);
最後就是用完了銷燬。
geoCoder.destroy();
測試結果,還是利用剛纔一樣的經緯度,輸出的結果和使用原生方法差不多,就不貼出來了。不過如果有附近商圈和興趣點的需求,那使用百度地圖SDK還是不錯的。
其實用谷歌提供的playservice也是可以的,就是由於你懂的原因,可能不如百度的穩定一些吧。以前也弄過,效果很好的,這裏就不說了。