Android studio 百度地图开发(3)地图导航

email:chentravelling@163.com

开发环境:win7 64位,Android Studio,请注意是Android Studio,使用的导航SDK版本:3.1.0

地图显示、工程配置请参考:Android studio 百度地图开发(1)配置工程、显示地图

百度地图定位请参考:Android studio 百度地图开发(2)地图定位

一.我为百度做点事

因为在写定位功能时自己想从头到尾地写,但最后完全是参考了百度官网上的Demo才弄出来,后来发现用Android Studio做导航的资料太少了,于是这次写导航功能的时候就直接参考了百度Demo。但是问题依旧,开始使用了百度的RoutePlanDemo.java这个Demo,发现有几个包找不到,捣腾了很久没有成功,最后甚至准备用Myclipse试试,结果配置环境的时候还把MyEclipse玩坏了,重装!实在怀疑自己解决问题的能力,百般无奈之下,还是回到了Android Studio,静下心来分析问题出在了哪里:是不是本身SDK3.1.0里就没有这几个包了呢?果然,我打开jar包一看,没有。欣喜,换了导航SDK3.1.0中的Demo+SDK3.1.0,于是,导航就真正开始了。下面的代码都来自百度Demo,我只是自己理解了一番,后面将结合第二部分的地图定位和这部分的导航功能,自己再写一个可以实现手动选择起点和终点的Demo出来,喜欢吐槽的朋友请键盘留情,然后悄悄飘过吧。

一直以来都是喜欢百度的,尽管百度的的确确不能和Google比。但是,在中国的科技氛围里,我觉得百度已经够好了,所以也贴一点百度地图导航SDK的功能介绍。

最后,我想问问那些嘴里骂着百度心里又离不开百度的朋友:你为中国的科技、IT、社会做了什么?

二.导航SDK的应用

请注意:代码来自百度官方Demo。

导航SDK版本:BaiDuNaviSDK_3.1.0.jar和httpmime-4.1.2.jar

IDE:Android Studio

第一步:工程配置

这部分看似简单,没有关于Android Studio版的官方教程,真的很麻烦。

(1)将BaiDuNaviSDK_3.1.0.jar和httpmime-4.1.2.jar复制到/libs文件夹下

(2)方法一:右键这两个jar包:add as library->确定;

方法二:打开build.gradle,在dependencies中增加

compile files('libs/httpmime-4.1.2.jar')compile files('libs/BaiduNaviSDK_3.1.0.jar')

(3)在/src/main文件夹下新建assets,添加BaiduNaviSDK_Resource_X_X.png, BaiduNaviSDK _X_X.png和chanel文件【这三个文件可以在Demo中找到,直接复制过来就ok】

(4)在src/main/JNIlibs/armeabi-v7a中不是armeabi】都添加:libapp_BaiduNaviApplib.so、libapp_BaiduVIlib.so、libcurl.so、libgnustl_shared.so、 libbds.so、 libbd_etts.so 、libBDSpeechDecoder_V1.so

第二步:AndroidManifest.xml:增加权限、注册activity、API_KEY和定位服务

<?xml version="1.0" encoding="utf-8"?><manifest xmlns:android="http://schemas.android.com/apk/res/android"    package="intvehapp.intvehapp" >    <!-- 百度地图API所需权限 -->    <uses-permission android:name="android.permission.GET_ACCOUNTS" />    <uses-permission android:name="android.permission.USE_CREDENTIALS" />    <uses-permission android:name="android.permission.MANAGE_ACCOUNTS" />    <uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" />    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />    <uses-permission android:name="android.permission.INTERNET" />    <uses-permission android:name="com.android.launcher.permission.READ_SETTINGS" />    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />    <uses-permission android:name="android.permission.READ_PHONE_STATE" />    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />    <uses-permission android:name="android.permission.BROADCAST_STICKY" />    <uses-permission android:name="android.permission.WRITE_SETTINGS" />    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />    <!-- 访问精确位置的权限 -->    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />    <!-- SDK1.5需要android.permission.GET_TASKS权限判断本程序是否为当前运行的应用? -->    <uses-permission android:name="android.permission.GET_TASKS" />    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />    <uses-permission android:name="android.permission.BROADCAST_STICKY" />    <uses-permission android:name="android.permission.WRITE_SETTINGS" />    <!-- 来电消音 -->    <uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS" />    <uses-permission android:name="android.permission.READ_PHONE_STATE" />    <uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />    <uses-permission android:name="android.permission.RECORD_AUDIO" />    <!-- 摄影机 -->    <uses-permission android:name="android.permission.CAMERA" />    <application        android:allowBackup="true"        android:icon="@mipmap/ic_launcher"        android:label="@string/app_name"        android:supportsRtl="true"        android:theme="@style/AppTheme" >        <activity android:name=".BNDemoMainActivity" >            <intent-filter>                <action android:name="android.intent.action.MAIN" />                <category android:name="android.intent.category.LAUNCHER" />            </intent-filter>        </activity>        <!--百度API_KEY-->        <meta-data            android:name="com.baidu.lbsapi.API_KEY"            android:value="iXDGdZtFiPifnNm6dUEFwDRXYQVeZ37V" />        <!--百度定位服务-->        <service            android:name="com.baidu.location.f"            android:enabled="true"            android:process=":remote" >        </service>        <!--注册activity-->        <activity android:name=".BNDemoGuideActivity" />    </application></manifest>

第三步:BNDemoMainActivity.java

package intvehapp.intvehapp;import java.io.File;import java.util.ArrayList;import java.util.LinkedList;import java.util.List;import com.baidu.navisdk.adapter.BNOuterLogUtil;import com.baidu.navisdk.adapter.BNOuterTTSPlayerCallback;import com.baidu.navisdk.adapter.BNRoutePlanNode;import com.baidu.navisdk.adapter.BNRoutePlanNode.CoordinateType;import com.baidu.navisdk.adapter.BNaviSettingManager;import com.baidu.navisdk.adapter.BaiduNaviManager;import com.baidu.navisdk.adapter.BaiduNaviManager.NaviInitListener;import com.baidu.navisdk.adapter.BaiduNaviManager.RoutePlanListener;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.os.Environment;import android.os.Handler;import android.os.Message;import android.util.Log;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.Toast;public class BNDemoMainActivity extends Activity {public static List<Activity> activityList = new LinkedList<Activity>();/** * App在SD卡中的目录名 */private static final String APP_FOLDER_NAME = "intvehapp";/** * 4个按钮控件 */private Button mWgsNaviBtn = null;private Button mGcjNaviBtn = null;private Button mBdmcNaviBtn = null;private Button mDb06ll = null;/** * SD卡的路径 */private String mSDCardPath = null;public static final String ROUTE_PLAN_NODE = "routePlanNode";public static final String SHOW_CUSTOM_ITEM = "showCustomItem";public static final String RESET_END_NODE = "resetEndNode";public static final String VOID_MODE = "voidMode";@Overrideprotected void onCreate(Bundle savedInstanceState) {// TODO Auto-generated method stubsuper.onCreate(savedInstanceState);activityList.add(this);setContentView(R.layout.activity_main_guide);/** * 获取按钮控件 */mWgsNaviBtn = (Button) findViewById(R.id.wgsNaviBtn);mGcjNaviBtn = (Button) findViewById(R.id.gcjNaviBtn);mBdmcNaviBtn = (Button) findViewById(R.id.bdmcNaviBtn);mDb06ll = (Button) findViewById(R.id.mDb06llNaviBtn);BNOuterLogUtil.setLogSwitcher(true);/** * 初始化按钮监听函数  */initListener();if (initDirs()) {/** * 使用SDK前,先进行百度服务授权和引擎初始化。 */initNavi();}// BNOuterLogUtil.setLogSwitcher(true);}@Overrideprotected void onResume() {super.onResume();}/** * 为每一个按钮控件增加点击监听事件,增加算路节点 */private void initListener() {if (mWgsNaviBtn != null) {mWgsNaviBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View arg0) {/** * 判断百度导航是否初始化 */if (BaiduNaviManager.isNaviInited()) {/** * 添加起点、终点 */routeplanToNavi(CoordinateType.WGS84);}}});}if (mGcjNaviBtn != null) {mGcjNaviBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View arg0) {if (BaiduNaviManager.isNaviInited()) {routeplanToNavi(CoordinateType.GCJ02);}}});}if (mBdmcNaviBtn != null) {mBdmcNaviBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View arg0) {if (BaiduNaviManager.isNaviInited()) {routeplanToNavi(CoordinateType.BD09_MC);}}});}if (mDb06ll != null) {mDb06ll.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View arg0) {if (BaiduNaviManager.isNaviInited()) {routeplanToNavi(CoordinateType.BD09LL);}}});}}/** * 初始化SD卡,在SD卡路径下新建文件夹:App目录名,文件中包含了很多东西,比如log、cache等等 * @return */private boolean initDirs() {mSDCardPath = getSdcardDir();if (mSDCardPath == null) {return false;}File f = new File(mSDCardPath, APP_FOLDER_NAME);if (!f.exists()) {try {f.mkdir();} catch (Exception e) {e.printStackTrace();return false;}}return true;}String authinfo = null;/** * 内部TTS播报状态回传handler */private Handler ttsHandler = new Handler() {    public void handleMessage(Message msg) {        int type = msg.what;        switch (type) {            case BaiduNaviManager.TTSPlayMsgType.PLAY_START_MSG: {                 showToastMsg("Handler : TTS play start");                break;            }            case BaiduNaviManager.TTSPlayMsgType.PLAY_END_MSG: {                 showToastMsg("Handler : TTS play end");                break;            }            default :                break;        }    }};/** * 内部TTS播报状态回调接口 */private BaiduNaviManager.TTSPlayStateListener ttsPlayStateListener = new BaiduNaviManager.TTSPlayStateListener() {                @Override        public void playEnd() {//            showToastMsg("TTSPlayStateListener : TTS play end");        }                @Override        public void playStart() {//            showToastMsg("TTSPlayStateListener : TTS play start");        }    };public void showToastMsg(final String msg) {    BNDemoMainActivity.this.runOnUiThread(new Runnable() {            @Override            public void run() {                Toast.makeText(BNDemoMainActivity.this, msg, Toast.LENGTH_SHORT).show();            }        });}/** * 使用SDK前,先进行百度服务授权和引擎初始化 */private void initNavi() {BNOuterTTSPlayerCallback ttsCallback = null;BaiduNaviManager.getInstance().init(this, mSDCardPath, APP_FOLDER_NAME, new NaviInitListener() {@Overridepublic void onAuthResult(int status, String msg) {if (0 == status) {authinfo = "key校验成功!";} else {authinfo = "key校验失败, " + msg;}BNDemoMainActivity.this.runOnUiThread(new Runnable() {@Overridepublic void run() {Toast.makeText(BNDemoMainActivity.this, authinfo, Toast.LENGTH_LONG).show();}});}public void initSuccess() {Toast.makeText(BNDemoMainActivity.this, "百度导航引擎初始化成功", Toast.LENGTH_SHORT).show();initSetting();}public void initStart() {Toast.makeText(BNDemoMainActivity.this, "百度导航引擎初始化开始", Toast.LENGTH_SHORT).show();}public void initFailed() {Toast.makeText(BNDemoMainActivity.this, "百度导航引擎初始化失败", Toast.LENGTH_SHORT).show();}},  null, ttsHandler, null);}private String getSdcardDir() {if (Environment.getExternalStorageState().equalsIgnoreCase(Environment.MEDIA_MOUNTED)) {return Environment.getExternalStorageDirectory().toString();}return null;}/** * 算路设置起、终点,算路偏好,是否模拟导航等参数,然后在回调函数中设置跳转至诱导。 * @param coType */private void routeplanToNavi(CoordinateType coType) {BNRoutePlanNode sNode = null;BNRoutePlanNode eNode = null;switch (coType) {case GCJ02: {sNode = new BNRoutePlanNode(116.30142, 40.05087, "百度大厦", null, coType);eNode = new BNRoutePlanNode(116.39750, 39.90882, "北京天安门", null, coType);break;}case WGS84: {sNode = new BNRoutePlanNode(116.300821, 40.050969, "百度大厦", null, coType);eNode = new BNRoutePlanNode(116.397491, 39.908749, "北京天安门", null, coType);break;}case BD09_MC: {sNode = new BNRoutePlanNode(12947471, 4846474, "百度大厦", null, coType);eNode = new BNRoutePlanNode(12958160, 4825947, "北京天安门", null, coType);break;}case BD09LL: {sNode = new BNRoutePlanNode(116.30784537597782, 40.057009624099436, "百度大厦", null, coType);eNode = new BNRoutePlanNode(116.40386525193937, 39.915160800132085, "北京天安门", null, coType);break;}default:;}if (sNode != null && eNode != null) {List<BNRoutePlanNode> list = new ArrayList<BNRoutePlanNode>();list.add(sNode);list.add(eNode);/** * 发起算路操作并在算路成功后通过回调监听器进入导航过程,返回是否执行成功 */BaiduNaviManager.getInstance().launchNavigator(this,//建议是应用的主Activitylist,//传入的算路节点,顺序是起点、途经点、终点,其中途经点最多三个1,//算路偏好 1:推荐 8:少收费 2:高速优先 4:少走高速 16:躲避拥堵true,//true表示真实GPS导航,false表示模拟导航new DemoRoutePlanListener(sNode)//开始导航回调监听器,在该监听器里一般是进入导航过程页面);}}/** * 导航回调监听器 */public class DemoRoutePlanListener implements RoutePlanListener {private BNRoutePlanNode mBNRoutePlanNode = null;public DemoRoutePlanListener(BNRoutePlanNode node) {mBNRoutePlanNode = node;}@Overridepublic void onJumpToNavigator() {/* * 设置途径点以及resetEndNode会回调该接口 */ for (Activity ac : activityList) {   if (ac.getClass().getName().endsWith("BNDemoGuideActivity")) { return;}}/** * 导航activity */Intent intent = new Intent(BNDemoMainActivity.this, BNDemoGuideActivity.class);Bundle bundle = new Bundle();bundle.putSerializable(ROUTE_PLAN_NODE, (BNRoutePlanNode) mBNRoutePlanNode);intent.putExtras(bundle);startActivity(intent);}@Overridepublic void onRoutePlanFailed() {// TODO Auto-generated method stubToast.makeText(BNDemoMainActivity.this, "算路失败", Toast.LENGTH_SHORT).show();}}/** * 导航设置管理器 */private void initSetting(){/** * 日夜模式 1:自动模式 2:白天模式 3:夜间模式 */    BNaviSettingManager.setDayNightMode(BNaviSettingManager.DayNightMode.DAY_NIGHT_MODE_DAY);/** * 设置全程路况显示 */    BNaviSettingManager.setShowTotalRoadConditionBar(BNaviSettingManager.PreViewRoadCondition.ROAD_CONDITION_BAR_SHOW_ON);/** * 设置语音播报模式 */    BNaviSettingManager.setVoiceMode(BNaviSettingManager.VoiceMode.Veteran);/** * 设置省电模式 */        BNaviSettingManager.setPowerSaveMode(BNaviSettingManager.PowerSaveMode.DISABLE_MODE);/** * 设置实时路况条 */        BNaviSettingManager.setRealRoadCondition(BNaviSettingManager.RealRoadCondition.NAVI_ITS_ON);}private BNOuterTTSPlayerCallback mTTSCallback = new BNOuterTTSPlayerCallback() {@Overridepublic void stopTTS() {// TODO Auto-generated method stubLog.e("test_TTS", "stopTTS");}@Overridepublic void resumeTTS() {// TODO Auto-generated method stubLog.e("test_TTS", "resumeTTS");}@Overridepublic void releaseTTSPlayer() {// TODO Auto-generated method stubLog.e("test_TTS", "releaseTTSPlayer");}@Overridepublic int playTTSText(String speech, int bPreempt) {// TODO Auto-generated method stubLog.e("test_TTS", "playTTSText" + "_" + speech + "_" + bPreempt);return 1;}@Overridepublic void phoneHangUp() {// TODO Auto-generated method stubLog.e("test_TTS", "phoneHangUp");}@Overridepublic void phoneCalling() {// TODO Auto-generated method stubLog.e("test_TTS", "phoneCalling");}@Overridepublic void pauseTTS() {// TODO Auto-generated method stubLog.e("test_TTS", "pauseTTS");}@Overridepublic void initTTSPlayer() {// TODO Auto-generated method stubLog.e("test_TTS", "initTTSPlayer");}@Overridepublic int getTTSState() {// TODO Auto-generated method stubLog.e("test_TTS", "getTTSState");return 1;}};}

第四步:BNDemoGuideActivity.java

package intvehapp.intvehapp;import java.util.ArrayList;import java.util.List;import com.baidu.navisdk.adapter.BNRouteGuideManager;import com.baidu.navisdk.adapter.BNRouteGuideManager.CustomizedLayerItem;import com.baidu.navisdk.adapter.BNRouteGuideManager.OnNavigationListener;import com.baidu.navisdk.adapter.BNRoutePlanNode;import com.baidu.navisdk.adapter.BNaviBaseCallbackModel;import com.baidu.navisdk.adapter.BaiduNaviCommonModule;import com.baidu.navisdk.adapter.NaviModuleFactory;import com.baidu.navisdk.adapter.NaviModuleImpl;import com.baidu.navisdk.adapter.BNRoutePlanNode.CoordinateType;import android.app.Activity;import android.content.Intent;import android.os.Build;import android.os.Bundle;import android.os.Handler;import android.util.Log;import android.view.View;/** * 诱导界面 *  * @author sunhao04 * */public class BNDemoGuideActivity extends Activity {    private final String TAG = BNDemoGuideActivity.class.getName();    private BNRoutePlanNode mBNRoutePlanNode = null;    private BaiduNaviCommonModule mBaiduNaviCommonModule = null;    /*     * 对于导航模块有两种方式来实现发起导航。 1:使用通用接口来实现 2:使用传统接口来实现     */    // 是否使用通用接口    private boolean useCommonInterface = true;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        BNDemoMainActivity.activityList.add(this);        createHandler();        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {        }        View view = null;        if (useCommonInterface) {            //使用通用接口            mBaiduNaviCommonModule = NaviModuleFactory.getNaviModuleManager().getNaviCommonModule(                    NaviModuleImpl.BNaviCommonModuleConstants.ROUTE_GUIDE_MODULE, this,                    BNaviBaseCallbackModel.BNaviBaseCallbackConstants.CALLBACK_ROUTEGUIDE_TYPE, mOnNavigationListener);            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onCreate();                view = mBaiduNaviCommonModule.getView();            }                    } else {            //使用传统接口            view = BNRouteGuideManager.getInstance().onCreate(this,mOnNavigationListener);        }                if (view != null) {            setContentView(view);        }        Intent intent = getIntent();        if (intent != null) {            Bundle bundle = intent.getExtras();            if (bundle != null) {                mBNRoutePlanNode = (BNRoutePlanNode) bundle.getSerializable(BNDemoMainActivity.ROUTE_PLAN_NODE);            }        }    }    @Override    protected void onResume() {        super.onResume();        if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onResume();            }        } else {            BNRouteGuideManager.getInstance().onResume();        }                      if (hd != null) {            hd.sendEmptyMessageAtTime(MSG_SHOW, 2000);        }    }    protected void onPause() {        super.onPause();                if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onPause();            }        } else {            BNRouteGuideManager.getInstance().onPause();        }          };    @Override    protected void onDestroy() {        super.onDestroy();        if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onDestroy();            }        } else {            BNRouteGuideManager.getInstance().onDestroy();        }        BNDemoMainActivity.activityList.remove(this);          }    @Override    protected void onStop() {        super.onStop();        if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onStop();            }        } else {            BNRouteGuideManager.getInstance().onStop();        }           }    @Override    public void onBackPressed() {        if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onBackPressed(false);            }        } else {            BNRouteGuideManager.getInstance().onBackPressed(false);        }    }    public void onConfigurationChanged(android.content.res.Configuration newConfig) {        super.onConfigurationChanged(newConfig);        if(useCommonInterface) {            if(mBaiduNaviCommonModule != null) {                mBaiduNaviCommonModule.onConfigurationChanged(newConfig);            }        } else {            BNRouteGuideManager.getInstance().onConfigurationChanged(newConfig);        }    };    private void addCustomizedLayerItems() {        List<CustomizedLayerItem> items = new ArrayList<CustomizedLayerItem>();        CustomizedLayerItem item1 = null;        if (mBNRoutePlanNode != null) {            item1 = new CustomizedLayerItem(mBNRoutePlanNode.getLongitude(), mBNRoutePlanNode.getLatitude(),                    mBNRoutePlanNode.getCoordinateType(), getResources().getDrawable(R.drawable.ic_launcher),                    CustomizedLayerItem.ALIGN_CENTER);            items.add(item1);            BNRouteGuideManager.getInstance().setCustomizedLayerItems(items);        }        BNRouteGuideManager.getInstance().showCustomizedLayer(true);    }    private static final int MSG_SHOW = 1;    private static final int MSG_HIDE = 2;    private static final int MSG_RESET_NODE = 3;    private Handler hd = null;    private void createHandler() {        if (hd == null) {            hd = new Handler(getMainLooper()) {                public void handleMessage(android.os.Message msg) {                    if (msg.what == MSG_SHOW) {                        addCustomizedLayerItems();                    } else if (msg.what == MSG_HIDE) {                        BNRouteGuideManager.getInstance().showCustomizedLayer(false);                    } else if (msg.what == MSG_RESET_NODE) {                        BNRouteGuideManager.getInstance().resetEndNodeInNavi(                                new BNRoutePlanNode(116.21142, 40.85087, "百度大厦11", null, CoordinateType.GCJ02));                    }                };            };        }    }    private OnNavigationListener mOnNavigationListener = new OnNavigationListener() {        @Override        public void onNaviGuideEnd() {            finish();        }        @Override        public void notifyOtherAction(int actionType, int arg1, int arg2, Object obj) {            if (actionType == 0) {                Log.i(TAG, "notifyOtherAction actionType = " + actionType + ",导航到达目的地!");            }            Log.i(TAG, "actionType:" + actionType + "arg1:" + arg1 + "arg2:" + arg2 + "obj:" + obj.toString());        }    };}

第五步:Run看效果



三.个人理解

目前Run起来后还有点问题,比如还没发现语音播报这个可不可以。

单从这个Demo来看,导航功能实现起来并不会复杂,百度服务授权和引擎初始化成功后,主要是传入起点、终点、途经点等需要计算的节点,然后在回调函数中进入导航回调监听器,然后进入诱导界面。后面还有很多地方可以个性化修改或者按照自己的需要进行功能完善,这便是后面我需要做的工作了。

可能第一次用Android Studio做百度地图真的会遇到不少问题,比如SDK版本和.so文件(需要添加到armeabi-v7a文件夹中而不是armeabi中)总之,不管多难,不要轻易放弃,问题始终是可以解决的。


更多相关文章

  1. AndroidStudio中使用SVG
  2. android 网络定位失败 location为null的真正原因
  3. android quicksearchbox修改默认搜索引擎为百度
  4. LeakCanary2.3 核心原理浅析
  5. 【ArcGIS for Android】经纬度坐标、地图投影坐标、屏幕坐标互相
  6. 深入学习百度地图Android(安卓)SDK v4.0.0【第五关】瓦片图层
  7. 高德地图-地图中心固定Marker,Marker跳跃、掉落、生长动画
  8. 【Android(安卓)百度地图实战】2.几种地图图层的显示
  9. 百度地图系列01——最简单的调用百度地图

随机推荐

  1. android TextView 容纳不下内容,让字向左
  2. Android属性系统
  3. Android横竖屏切换总结
  4. Android(安卓)TV webview禁止自动获取焦
  5. android studio打开react-native的androi
  6. 从Android界面开发谈起
  7. android控件属性
  8. 使用 Android(安卓)Studio 跑新浪微博SDK
  9. Android学习札记15:对Android中View绘制流
  10. Android(安卓)提供的一系列辅助系统开发