当前定位方式:GPS定位、GSM蜂窝基站定位、AGPS定位是目前最流行的三种定位方式。
在AVD模拟器下面不能使用NETWORK_PROVIDER方式进行定位,因为没有SIM卡的相关信息。
在AVD模拟器下面使用GPS定位时的模拟方法。
1、在eclipse下,window->open perspective->DDMS->Emulator Control->Location Controls->Manual->选中Decimal->输入经纬度(注意要选择发送的设备的进程)下手动设置经纬度,并按send按钮。
(1)首先打开模拟器,然后运行cmd,输入telnet localhost 5554(注:5554是模拟器在本机的端口,有可能不一样哈,具体端口号,模拟器左上方有显示的),这样会出现
Android Console: type 'help' for a list of commands
OK的字样。
如果是使用WIN7的朋友,控制台可能会提示telnet无效什么的,那是因为WIN7下默认是不出现telnet的,需要手动打开。具体为:[1]控制面板-->程序-->打开或关闭Windows功能,然后将Telnet服务器和Telnet客户端勾选上。[2]然后在管理工具-->服务中手动启动Telnet
(2)使用geo命令模拟发送GPS信号:
geo fix 经度 纬度
(3)这时就会发现在模拟器的状态栏上多了一个GPS的标志~再使用Location loc = LocationManager.getLastKnownLocation("gps");就能获取到该坐标的位置了~~
- public class LocationData {
- String lat;
- String lon;
- String address;
- }
- public class LBSTool {
- private Context mContext;
- private LocationManager mLocationManager;
- private LocationData mLocation;
- private LBSThread mLBSThread;
- private MyLocationListner mNetworkListner;
- private MyLocationListner mGPSListener;
- private Looper mLooper;
- public LBSTool(Context context) {
- mContext = context;
- //获取Location manager
- mLocationManager = (LocationManager)mContext.getSystemService(Context.LOCATION_SERVICE);
- }
- ......
- }
- /**
- * 开始定位
- * @param timeout 超时设置
- * @return LocationData位置数据,如果超时则为null
- */
- public LocationData getLocation(long timeout) {
- mLocation = null;
- mLBSThread = new LBSThread();
- mLBSThread.start();//启动LBSThread
- timeout = timeout > 0 ? timeout : 0;
- synchronized (mLBSThread) {
- try {
- Log.i(Thread.currentThread().getName(), "Waiting for LocationThread to complete...");
- mLBSThread.wait(timeout);//主线程进入等待,等待时长timeout ms
- Log.i(Thread.currentThread().getName(), "Completed.Now back to main thread");
- }
- catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
- mLBSThread = null;
- return mLocation;
- }
- private class LBSThread extends Thread {
- @Override
- public void run() {
- setName("location thread");
- Log.i(Thread.currentThread().getName(), "--start--");
- Looper.prepare();//给LBSThread加上Looper
- mLooper = Looper.myLooper();
- registerLocationListener();
- Looper.loop();
- Log.e(Thread.currentThread().getName(), "--end--");
- }
- }
- private void registerLocationListener () {
- Log.i(Thread.currentThread().getName(), "registerLocationListener");
- if (isGPSEnabled()) {
- mGPSListener=new MyLocationListner();
- //五个参数分别为位置服务的提供者,最短通知时间间隔,最小位置变化,listener,listener所在消息队列的looper
- mLocationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 5000, 0, mGPSListener, mLooper);
- }
- if (isNetworkEnabled()) {
- mNetworkListner=new MyLocationListner();
- mLocationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 3000, 0, mNetworkListner, mLooper);
- }
- }
- /**
- * 判断GPS是否开启
- * @return
- */
- public boolean isGPSEnabled() {
- if(mLocationManager.isProviderEnabled(LocationManager.GPS_PROVIDER)) {
- Log.i(Thread.currentThread().getName(), "isGPSEnabled");
- return true;
- }
- else {
- return false;
- }
- }
- /**
- * 判断Network是否开启(包括移动网络和wifi)
- * @return
- */
- public boolean isNetworkEnabled() {
- return (isWIFIEnabled() || isTelephonyEnabled());
- }
- /**
- * 判断移动网络是否开启
- * @return
- */
- public boolean isTelephonyEnabled() {
- boolean enable = false;
- TelephonyManager telephonyManager = (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE);
- if (telephonyManager != null) {
- if (telephonyManager.getNetworkType() != TelephonyManager.NETWORK_TYPE_UNKNOWN) {
- enable = true;
- Log.i(Thread.currentThread().getName(), "isTelephonyEnabled");
- }
- }
- return enable;
- }
- /**
- * 判断wifi是否开启
- */
- public boolean isWIFIEnabled() {
- boolean enable = false;
- WifiManager wifiManager = (WifiManager)mContext.getSystemService(Context.WIFI_SERVICE);
- if(wifiManager.isWifiEnabled()) {
- enable = true;
- Log.i(Thread.currentThread().getName(), "isWIFIEnabled");
- }
- return enable;
- }
- private class MyLocationListner implements LocationListener{
- @Override
- public void onLocationChanged(Location location) {
- // 当LocationManager检测到最小位置变化时,就会回调到这里
- Log.i(Thread.currentThread().getName(), "Got New Location of provider:"+location.getProvider());
- unRegisterLocationListener();//停止LocationManager的工作
- try {
- synchronized (mLBSThread) {
- parseLatLon(location.getLatitude()+"", location.getLongitude()+"");//解析地理位置
- mLooper.quit();//解除LBSThread的Looper,LBSThread结束
- mLBSThread.notify();//通知主线程继续
- }
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- }
- //后3个方法此处不做处理
- @Override
- public void onStatusChanged(String provider, int status, Bundle extras) {}
- @Override
- public void onProviderEnabled(String provider) {}
- @Override
- public void onProviderDisabled(String provider) {}
- };
- /**
- * 使用经纬度从goole服务器获取对应地址
- * @param 经纬度
- */
- private void parseLatLon(String lat, String lon) throws Exception {
- Log.e(Thread.currentThread().getName(), "---parseLatLon---");
- Log.e(Thread.currentThread().getName(), "---"+lat+"---");
- try {
- HttpClient httpClient = new DefaultHttpClient();
- HttpGet get = new HttpGet("http://ditu.google.cn/maps/geo?output=json&q="+lat+","+lon);
- HttpResponse response = httpClient.execute(get);
- String resultString = EntityUtils.toString(response.getEntity());
- JSONObject jsonresult = new JSONObject(resultString);
- if(jsonresult.optJSONArray("Placemark") != null) {
- mLocation = new LocationData();
- mLocation.lat = lat;
- mLocation.lon = lon;
- mLocation.address = jsonresult.optJSONArray("Placemark").optJSONObject(0).optString("address");
- }
- }
- catch (Exception e) {
- e.printStackTrace();
- }
- }
- /**
- * 注销监听器
- */
- private void unRegisterLocationListener () {
- if(mGPSListener!=null){
- mLocationManager.removeUpdates(mGPSListener);
- mGPSListener=null;
- }
- if(mNetworkListner!=null){
- mLocationManager.removeUpdates(mNetworkListner);
- mNetworkListner=null;
- }
- }
- locationBtn.setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- //return mode
- LBSTool lbs = new LBSTool(LBStestActivity.this);
- LocationData location = lbs.getLocation(120000);
- if (location != null) {
- Log.i("---lat---",location.lat);
- Log.i("---lon---",location.lon);
- Log.i("---address---",location.address);
- Toast.makeText(LBStestActivity.this, location.lat + " " + location.lon + " " + location.address, Toast.LENGTH_LONG).show();
- }
- }
- });
- <uses-permission android:name="android.permission.INTERNET" />
- <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
- <uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
- <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
- Criteria criteria = new Criteria();
- criteria.setAccuracy(Criteria.ACCURACY_FINE); // 高精度
- criteria.setAltitudeRequired(false);
- criteria.setBearingRequired(false);
- criteria.setCostAllowed(true);
- criteria.setPowerRequirement(Criteria.POWER_LOW); // 低功耗
- String bestprovider = locationManager.getBestProvider(criteria, true); // 获取GPS信息
- Location location = locationManager.getLastKnownLocation(bestprovider); // 通过GPS获取位置
- Log.e("--bestprovider--", bestprovider);
- Log.e("--bestprovider--", location.getLatitude()+"");
使用百度SDK定位
使用Android自带的LocationManager和Location获取位置的时候,经常会有获取的location为null的情况,并且操作起来也不是很方便,在这个Demo里我使用了百度地图API中的定位SDK,可以一次性获取当前位置经纬度以及详细地址信息,还可以获取周边POI信息,同时可以设定位置通知点,当到达某一位置时,发出通知信息等方式来告知用户。jar包下载以及官方文档请参照:百度定位SDK,前提是需要注册百度开发者账号。
下面来看看定位的基本原理,目前,定位SDK可以通过GPS、基站、Wifi信号进行定位。基本定位流程如下图所示,当应用程序向定位SDK发起定位请求时,定位SDK会根据当前的GPS、基站、Wifi信息生成相对应的定位依据。然后定位SDK会根据定位依据来进行定位。如果需要,定位SDK会向定位服务器发送网络请求。定位服务器会根据请求的定位依据推算出对应的坐标位置,然后根据用户的定制信息,生成定位结果返回给定位SDK。
到官方下载jar文件后添加到工程,工程目录截图如下:
注意要把locSDK_2.4.jar添加到当天工程,右键jar文件-Build path-Add to。。。
上代码
布局文件:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:orientation="vertical" >
- <Button
- android:id="@+id/btn_start"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:layout_marginTop="20dp"
- android:text="Start"/>
- <TextView
- android:id="@+id/tv_loc_info"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:textSize="18sp" />
- </LinearLayout>
配置文件:
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.ericssonlabs"
- android:versionCode="1"
- android:versionName="1.0" >
- <uses-sdk android:minSdkVersion="8" />
- <permission android:name="android.permission.BAIDU_LOCATION_SERVICE" >
- </permission>
- <uses-permission android:name="android.permission.BAIDU_LOCATION_SERVICE" >
- </uses-permission>
- <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" >
- </uses-permission>
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" >
- </uses-permission>
- <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" >
- </uses-permission>
- <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" >
- </uses-permission>
- <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" >
- </uses-permission>
- <uses-permission android:name="android.permission.READ_PHONE_STATE" >
- </uses-permission>
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" >
- </uses-permission>
- <uses-permission android:name="android.permission.INTERNET" />
- <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" >
- </uses-permission>
- <uses-permission android:name="android.permission.READ_LOGS" >
- </uses-permission>
- <application
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name" >
- <activity
- android:name=".LocationDemoActivity"
- android:label="@string/app_name" >
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <service
- android:name="com.baidu.location.f"
- android:enabled="true"
- android:permission="android.permission.BAIDU_LOCATION_SERVICE"
- android:process=":remote" >
- <intent-filter>
- <action android:name="com.baidu.location.service_v2.4" />
- </intent-filter>
- </service>
- </application>
- </manifest>
实现代码:
- public class LocationDemoActivity extends Activity {
- private TextView locationInfoTextView = null;
- private Button startButton = null;
- private LocationClient locationClient = null;
- private static final int UPDATE_TIME = 5000;
- private static int LOCATION_COUTNS = 0;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- locationInfoTextView = (TextView) this.findViewById(R.id.tv_loc_info);
- startButton = (Button) this.findViewById(R.id.btn_start);
- locationClient = new LocationClient(this);
- //设置定位条件
- LocationClientOption option = new LocationClientOption();
- option.setOpenGps(true); //是否打开GPS
- option.setCoorType("bd09ll"); //设置返回值的坐标类型。
- option.setPriority(LocationClientOption.NetWorkFirst); //设置定位优先级
- option.setProdName("LocationDemo"); //设置产品线名称。强烈建议您使用自定义的产品线名称,方便我们以后为您提供更高效准确的定位服务。
- option.setScanSpan(UPDATE_TIME); //设置定时定位的时间间隔。单位毫秒
- locationClient.setLocOption(option);
- //注册位置监听器
- locationClient.registerLocationListener(new BDLocationListener() {
- @Override
- public void onReceiveLocation(BDLocation location) {
- // TODO Auto-generated method stub
- if (location == null) {
- return;
- }
- StringBuffer sb = new StringBuffer(256);
- sb.append("Time : ");
- sb.append(location.getTime());
- sb.append("\nError code : ");
- sb.append(location.getLocType());
- sb.append("\nLatitude : ");
- sb.append(location.getLatitude());
- sb.append("\nLontitude : ");
- sb.append(location.getLongitude());
- sb.append("\nRadius : ");
- sb.append(location.getRadius());
- if (location.getLocType() == BDLocation.TypeGpsLocation){
- sb.append("\nSpeed : ");
- sb.append(location.getSpeed());
- sb.append("\nSatellite : ");
- sb.append(location.getSatelliteNumber());
- } else if (location.getLocType() == BDLocation.TypeNetWorkLocation){
- sb.append("\nAddress : ");
- sb.append(location.getAddrStr());
- }
- LOCATION_COUTNS ++;
- sb.append("\n检查位置更新次数:");
- sb.append(String.valueOf(LOCATION_COUTNS));
- locationInfoTextView.setText(sb.toString());
- }
- @Override
- public void onReceivePoi(BDLocation location) {
- }
- });
- startButton.setOnClickListener(new OnClickListener() {
- @Override
- public void onClick(View v) {
- if (locationClient == null) {
- return;
- }
- if (locationClient.isStarted()) {
- startButton.setText("Start");
- locationClient.stop();
- }else {
- startButton.setText("Stop");
- locationClient.start();
- /*
- *当所设的整数值大于等于1000(ms)时,定位SDK内部使用定时定位模式。
- *调用requestLocation( )后,每隔设定的时间,定位SDK就会进行一次定位。
- *如果定位SDK根据定位依据发现位置没有发生变化,就不会发起网络请求,
- *返回上一次定位的结果;如果发现位置改变,就进行网络请求进行定位,得到新的定位结果。
- *定时定位时,调用一次requestLocation,会定时监听到定位结果。
- */
- locationClient.requestLocation();
- }
- }
- });
- }
- @Override
- protected void onDestroy() {
- super.onDestroy();
- if (locationClient != null && locationClient.isStarted()) {
- locationClient.stop();
- locationClient = null;
- }
- }
- }
来看看最后实现效果,点击Start后进入位置监听状态,根据设置的监听时间间隔进行定位,如果位置有变化则进行位置更新,同时显示了检测位置更新的次数,如果开启了GPS,则获取到卫星后,进行GPS定位:
下载百度的SDK后,引入开发包。
设置manifest的application中服务和KRY信息
<service
android:name="com.baidu.location.f"
android:enabled="true"
android:process=":remote" >
<intent-filter>
<action android:name="com.baidu.location.service_v2.2" >
</action>
</intent-filter>
</service>
<!-- meta-data需要写在application中 -->
<meta-data
android:name="com.baidu.lbsapi.API_KEY"
android:value="*********************" />//申请的KEY值
如果出现Failed to open database '/storage/sdcard/baidu/tempdata/ls.db错误,可能是在模拟器中没有建立SD卡