`
kobe学java
  • 浏览: 257847 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

Android获取所在地城市名

 
阅读更多

 

【Android开发笔记】4.简单基站定位程序

 

前言

经过前面几节的学习,我们已经对Android程序的开发流程有了个大体的了解,为了提高我们的学习兴趣,在这一节我们将编写一个简单的基站定位程序。现在LBS(Location Based Service,基于位置的服务)移动应用相当流行(如:微信,切客,嘀咕,街旁等),基站定位是这类程序用到的关键性技术之一,我们来揭开它的神秘面纱吧。

在这一节里,我们会接触到事件、TelephonyManager、HTTP通信、JSON的使用等知识点。

声明

本系列文章不是教程,仅为笔记,如有不当之处请指正。

目录

正文

在Android操作系统下,基站定位其实很简单,先说一下实现流程:

调用SDK中的API(TelephonyManager)获得MCC、MNC、LAC、CID等信息,然后通过google的API获得所在位置的经纬度,最后再通过google map的API获得实际的地理位置。(google真牛!)

有同学会问:MNC、MCC、LAC、CID都是些什么东西?google又怎么通过这些东西就获得经纬度了呢?

我们一起来学习一下:

了解了这几个名词的意思,相信有些朋友已经知道后面的事了:google存储了这些信息,直接查询就能得到经纬度了。(至于google怎么得到移动、联通的基站信息,这就不得而知了,反正google免费提供接口,直接调用就是)

下面开始动手。

一、设置界面

我们在上一节的程序的基础上进行开发,在DemoActivity的界面上实现这个功能。(没有代码的同学可点击这里下载,感谢yuejianjun同学的建议,以后我会在每一节的最后把例子打包提供下载)

首先我们将DemoActivity使用的布局修改一下:

第1行为TextView,显示提示文字;第2行为一个Button,触发事件;第3行、第4行分别显示基站信息和地理位置(现在为空,看不到)。

layout/main.xml文件内容如下:

 



——————————————————分割线————————————————————

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
<?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" >
 
    <TextView
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="Please click the button below to get your location" />
 
    <Button
        android:id="@+id/button1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Click Me" />
 
    <TextView
        android:id="@+id/cellText"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="" />
 
    <TextView
        android:id="@+id/lacationText"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="" />
 
</LinearLayout>

 



 

接下来我们打开DemoActivity.java编写代码。

二、为按钮绑定事件

我们在Activity创建时绑定事件,将以下代码添加到setContentView(R.layout.main);后:

 



 

1
2
3
4
5
6
7
8
9
/** 为按钮绑定事件 */
Button btnGetLocation = (Button)findViewById(R.id.button1);
btnGetLocation.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View arg0) {
        // TODO Auto-generated method stub
        onBtnClick();
    }
});

 



 

同时还需要在头部import相关组件:

 



 

1
2
3
import android.view.View;
import android.widget.Button;
import android.view.View.OnClickListener;

 



 

我们来分析一下这段代码:

首先我们通过findViewById(R.id.button1)找到按钮这个对象,前面加(Button)表示显示的转换为Button对象;

然后设置按钮点击事件的监听器,参数为OnClickListener对象,再重载这个类的onClick方法,调用onBtnClick方法(这个方法得由我们自己去写,他在点击按钮时被调用)。

好了,调用方法写好了,我们来写实现(调用后需要做什么事)。动手编码之前先在脑中整理好思路,养成好习惯。

我们需要在DemoActivty类中添加如下私有方法:

 

我们需要刚刚提到的onBtnClick回调方法,被调用时实现取得基站信息、获取经纬度、获取地理位置、显示的功能。但是很显然,全部揉到一个方法里面并不是个好主意,我们将它分割为几个方法;

添加获取基站信息的方法getCellInfo,返回基站信息;

添加获取经纬度的方法getItude,传入基站信息,返回经纬度;

添加获取地理位置的方法getLocation,传入经纬度,返回地理位置;

添加显示结果的方法showResult,传入得到的信息在界面上显示出来。

 

好了,先将方法添上,完整代码如下:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
package com.android.demo;
 
import android.R.bool;
import android.R.integer;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.view.View.OnClickListener;
 
public class DemoActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
         
        /** 为按钮绑定事件 */
        Button btnGetLocation = (Button)findViewById(R.id.button1);
        btnGetLocation.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                onBtnClick();
            }
        });
    }
     
    /** 基站信息结构体 */
    public class SCell{
        public int MCC;
        public int MNC;
        public int LAC;
        public int CID;
    }
     
    /** 经纬度信息结构体 */
    public class SItude{
        public String latitude;
        public String longitude;
    }
     
    /** 按钮点击回调函数 */
    private void onBtnClick(){
         
    }
     
    /** 获取基站信息 */
    private SCell getCellInfo(){
 
    }
     
    /** 获取经纬度 */
    private SItude getItude(SCell cell){
         
    }
     
    /** 获取地理位置 */
    private String getLocation(SItude itude){
         
    }
     
    /** 显示结果 */
    private void showResult(SCell cell, String location){
         
    }
}

 



 

现在在onBtnClick方法中编码,依次调用后面几个方法,代码如下:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/** 按钮点击回调函数 */
private void onBtnClick(){
    /** 弹出一个等待状态的框 */
    ProgressDialog mProgressDialog = new ProgressDialog(this);
    mProgressDialog.setMessage("正在获取中...");
    mProgressDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);
    mProgressDialog.show();
     
    try {
        /** 获取基站数据 */
        SCell cell = getCellInfo();
         
        /** 根据基站数据获取经纬度 */
        SItude itude = getItude(cell);
         
        /** 获取地理位置 */
        String location = getLocation(itude);
         
        /** 显示结果 */
        showResult(cell, location);
         
        /** 关闭对话框 */
        mProgressDialog.dismiss();
    }catch (Exception e) {
        /** 关闭对话框 */
        mProgressDialog.dismiss();
        /** 显示错误 */
        TextView cellText = (TextView)findViewById(R.id.cellText);
        cellText.setText(e.getMessage());
    }
}

 



 

按钮相关的工作就完成了,接下来编写获取基站信息的方法。

三、获取基站信息

获取基站信息我们需要调用SDK提供的API中的TelephonyManager,需要在文件头部引入:

 



 

1
2
import android.telephony.TelephonyManager;
import android.telephony.gsm.GsmCellLocation;

 



 

完整代码为:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/**
 * 获取基站信息
 *
 * @throws Exception
 */
private SCell getCellInfo() throws Exception {
    SCell cell = new SCell();
 
    /** 调用API获取基站信息 */
    TelephonyManager mTelNet = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
    GsmCellLocation location = (GsmCellLocation) mTelNet.getCellLocation();
    if (location == null)
        throw new Exception("获取基站信息失败");
 
    String operator = mTelNet.getNetworkOperator();
    int mcc = Integer.parseInt(operator.substring(03));
    int mnc = Integer.parseInt(operator.substring(3));
    int cid = location.getCid();
    int lac = location.getLac();
 
    /** 将获得的数据放到结构体中 */
    cell.MCC = mcc;
    cell.MNC = mnc;
    cell.LAC = lac;
    cell.CID = cid;
 
    return cell;
}

 



 

如果获得的位置信息为null将抛出错误,不再继续执行。最后将获取的基站信息封装为结构体返回。

四、获取经纬度

在这一步,我们需要采用HTTP调用google的API以获取基站所在的经纬度。

Android作为一款互联网手机,联网的功能必不可少。Android提供了多个接口供我们使用,这里我们使用DefaultHttpClient。

完整的方法代码如下:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
/**
 * 获取经纬度
 *
 * @throws Exception
 */
private SItude getItude(SCell cell) throws Exception {
    SItude itude = new SItude();
 
    /** 采用Android默认的HttpClient */
    HttpClient client = new DefaultHttpClient();
    /** 采用POST方法 */
    HttpPost post = new HttpPost("http://www.google.com/loc/json");
    try {
        /** 构造POST的JSON数据 */
        JSONObject holder = new JSONObject();
        holder.put("version""1.1.0");
        holder.put("host""maps.google.com");
        holder.put("address_language""zh_CN");
        holder.put("request_address"true);
        holder.put("radio_type""gsm");
        holder.put("carrier""HTC");
 
        JSONObject tower = new JSONObject();
        tower.put("mobile_country_code", cell.MCC);
        tower.put("mobile_network_code", cell.MNC);
        tower.put("cell_id", cell.CID);
        tower.put("location_area_code", cell.LAC);
 
        JSONArray towerarray = new JSONArray();
        towerarray.put(tower);
        holder.put("cell_towers", towerarray);
 
        StringEntity query = new StringEntity(holder.toString());
        post.setEntity(query);
 
        /** 发出POST数据并获取返回数据 */
        HttpResponse response = client.execute(post);
        HttpEntity entity = response.getEntity();
        BufferedReader buffReader = new BufferedReader(newInputStreamReader(entity.getContent()));
        StringBuffer strBuff = new StringBuffer();
        String result = null;
        while ((result = buffReader.readLine()) != null) {
            strBuff.append(result);
        }
 
        /** 解析返回的JSON数据获得经纬度 */
        JSONObject json = new JSONObject(strBuff.toString());
        JSONObject subjosn = new JSONObject(json.getString("location"));
 
        itude.latitude = subjosn.getString("latitude");
        itude.longitude = subjosn.getString("longitude");
         
        Log.i("Itude", itude.latitude + itude.longitude);
         
    catch (Exception e) {
        Log.e(e.getMessage(), e.toString());
        throw new Exception("获取经纬度出现错误:"+e.getMessage());
    finally{
        post.abort();
        client = null;
    }
     
    return itude;
}

 



 

代笔中关键的地方都作了注释,同学们还有不理解的举手哈。

在这里采用POST方法将JSON数据发送到googleAPI,google返回JSON数据,我们得到数据后解析,得到经纬度信息。

关于google 基站信息API的官方说明>>请到这里查看

五、获取物理位置

得到经纬度后,我们将之转换为物理地址。

我们仍然使用DefaultHttpClient来调用google地图的API,获得物理信息,不过在这里我们使用GET方法。

完整的方法代码如下:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
/**
 * 获取地理位置
 *
 * @throws Exception
 */
private String getLocation(SItude itude) throws Exception {
    String resultString = "";
 
    /** 这里采用get方法,直接将参数加到URL上 */
    String urlString = String.format("http://maps.google.cn/maps/geo?key=abcdefg&q=%s,%s", itude.latitude, itude.longitude);
    Log.i("URL", urlString);
 
    /** 新建HttpClient */
    HttpClient client = new DefaultHttpClient();
    /** 采用GET方法 */
    HttpGet get = new HttpGet(urlString);
    try {
        /** 发起GET请求并获得返回数据 */
        HttpResponse response = client.execute(get);
        HttpEntity entity = response.getEntity();
        BufferedReader buffReader = new BufferedReader(newInputStreamReader(entity.getContent()));
        StringBuffer strBuff = new StringBuffer();
        String result = null;
        while ((result = buffReader.readLine()) != null) {
            strBuff.append(result);
        }
        resultString = strBuff.toString();
 
        /** 解析JSON数据,获得物理地址 */
        if (resultString != null && resultString.length() > 0) {
            JSONObject jsonobject = new JSONObject(resultString);
            JSONArray jsonArray = newJSONArray(jsonobject.get("Placemark").toString());
            resultString = "";
            for (int i = 0; i < jsonArray.length(); i++) {
                resultString = jsonArray.getJSONObject(i).getString("address");
            }
        }
    catch (Exception e) {
        throw new Exception("获取物理位置出现错误:" + e.getMessage());
    finally {
        get.abort();
        client = null;
    }
 
    return resultString;
}

 



 

GET方法就比POST方法简单多了,得到的数据同样为JSON格式,解析一下得到物理地址。

六、显示结果

好了,我们已经得到我们想要的信息了,我们把它显示出来,方法代码如下:

 



 

1
2
3
4
5
6
7
8
9
/** 显示结果 */
private void showResult(SCell cell, String location) {
    TextView cellText = (TextView) findViewById(R.id.cellText);
    cellText.setText(String.format("基站信息:mcc:%d, mnc:%d, lac:%d, cid:%d",
            cell.MCC, cell.MNC, cell.LAC, cell.CID));
 
    TextView locationText = (TextView) findViewById(R.id.lacationText);
    locationText.setText("物理位置:" + location);
}

 



 

七、运行程序

我们的编码工作已经完成了。在上面的代码中有些地方需要的引入代码没有提到,下面把完整的代码贴出来:

 



 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
package com.android.demo;
 
import java.io.BufferedReader;
import java.io.InputStreamReader;
 
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.DefaultHttpClient;
 
import org.json.JSONArray;
import org.json.JSONObject;
 
import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Context;
import android.os.Bundle;
import android.telephony.TelephonyManager;
import android.telephony.gsm.GsmCellLocation;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import android.view.View.OnClickListener;
 
public class DemoActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
 
        /** 为按钮绑定事件 */
        Button btnGetLocation = (Button) findViewById(R.id.button1);
        btnGetLocation.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                onBtnClick();
            }
        });
    }
     
    /** 基站信息结构体 */
    public class SCell{
        public int MCC;
        public int MNC;
        public int LAC;
        public int CID;
    }
     
    /** 经纬度信息结构体 */
    public class SItude{
        public String latitude;
        public String longitude;
    }
     
    /** 按钮点击回调函数 */
    private void onBtnClick() {
        /** 弹出一个等待状态的框 */
        ProgressDialog mProgressDialog = new ProgressDialog(this);
        mProgressDialog.setMessage("正在获取中...");
        mProgressDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);
        mProgressDialog.show();
 
        try {
            /** 获取基站数据 */
            SCell cell = getCellInfo();
 
            /** 根据基站数据获取经纬度 */
            SItude itude = getItude(cell);
 
            /** 获取地理位置 */
            String location = getLocation(itude);
 
            /** 显示结果 */
            showResult(cell, location);
 
            /** 关闭对话框 */
            mProgressDialog.dismiss();
        catch (Exception e) {
            /** 关闭对话框 */
            mProgressDialog.dismiss();
            /** 显示错误 */
            TextView cellText = (TextView) findViewById(R.id.cellText);
            cellText.setText(e.getMessage());
            Log.e("Error", e.getMessage());
        }
    }
     
    /**
     * 获取基站信息
     *
     * @throws Exception
     */
    private SCell getCellInfo() throws Exception {
        SCell cell = new SCell();
 
        /** 调用API获取基站信息 */
        TelephonyManager mTelNet = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
        GsmCellLocation location = (GsmCellLocation) mTelNet.getCellLocation();
        if (location == null)
            throw new Exception("获取基站信息失败");
 
        String operator = mTelNet.getNetworkOperator();
        int mcc = Integer.parseInt(operator.substring(03));
        int mnc = Integer.parseInt(operator.substring(3));
        int cid = location.getCid();
        int lac = location.getLac();
 
        /** 将获得的数据放到结构体中 */
        cell.MCC = mcc;
        cell.MNC = mnc;
        cell.LAC = lac;
        cell.CID = cid;
 
        return cell;
    }
     
    /**
     * 获取经纬度
     *
     * @throws Exception
     */
    private SItude getItude(SCell cell) throws Exception {
        SItude itude = new SItude();
 
        /** 采用Android默认的HttpClient */
        HttpClient client = new DefaultHttpClient();
        /** 采用POST方法 */
        HttpPost post = new HttpPost("http://www.google.com/loc/json");
        try {
            /** 构造POST的JSON数据 */
            JSONObject holder = new JSONObject();
            holder.put("version""1.1.0");
            holder.put("host""maps.google.com");
            holder.put("address_language""zh_CN");
            holder.put("request_address"true);
            holder.put("radio_type""gsm");
            holder.put("carrier""HTC");
 
            JSONObject tower = new JSONObject();
            tower.put("mobile_country_code", cell.MCC);
            tower.put("mobile_network_code", cell.MNC);
            tower.put("cell_id", cell.CID);
            tower.put("location_area_code", cell.LAC);
 
            JSONArray towerarray = new JSONArray();
            towerarray.put(tower);
            holder.put("cell_towers", towerarray);
 
            StringEntity query = new StringEntity(holder.toString());
            post.setEntity(query);
 
            /** 发出POST数据并获取返回数据 */
            HttpResponse response = client.execute(post);
            HttpEntity entity = response.getEntity();
            BufferedReader buffReader = new BufferedReader(newInputStreamReader(entity.getContent()));
            StringBuffer strBuff = new StringBuffer();
            String result = null;
            while ((result = buffReader.readLine()) != null) {
                strBuff.append(result);
            }
 
            /** 解析返回的JSON数据获得经纬度 */
            JSONObject json = new JSONObject(strBuff.toString());
            JSONObject subjosn = new JSONObject(json.getString("location"));
 
            itude.latitude = subjosn.getString("latitude");
            itude.longitude = subjosn.getString("longitude");
             
            Log.i("Itude", itude.latitude + itude.longitude);
             
        catch (Exception e) {
            Log.e(e.getMessage(), e.toString());
            throw new Exception("获取经纬度出现错误:"+e.getMessage());
        finally{
            post.abort();
            client = null;
        }
         
        return itude;
    }
     
    /**
     * 获取地理位置
     *
     * @throws Exception
     */
    private String getLocation(SItude itude) throws Exception {
        String resultString = "";
 
        /** 这里采用get方法,直接将参数加到URL上 */
        String urlString = String.format("http://maps.google.cn/maps/geo?key=abcdefg&q=%s,%s", itude.latitude, itude.longitude);
        Log.i("URL", urlString);
 
        /** 新建HttpClient */
        HttpClient client = new DefaultHttpClient();
        /** 采用GET方法 */
        HttpGet get = new HttpGet(urlString);
        try {
            /** 发起GET请求并获得返回数据 */
            HttpResponse response = client.execute(get);
            HttpEntity entity = response.getEntity();
            BufferedReader buffReader = new BufferedReader(newInputStreamReader(entity.getContent()));
            StringBuffer strBuff = new StringBuffer();
            String result = null;
            while ((result = buffReader.readLine()) != null) {
                strBuff.append(result);
            }
            resultString = strBuff.toString();
 
            /** 解析JSON数据,获得物理地址 */
            if (resultString != null && resultString.length() > 0) {
                JSONObject jsonobject = new JSONObject(resultString);
                JSONArray jsonArray = newJSONArray(jsonobject.get("Placemark").toString());
                resultString = "";
                for (int i = 0; i < jsonArray.length(); i++) {
                    resultString = jsonArray.getJSONObject(i).getString("address");
                }
            }
        catch (Exception e) {
            throw new Exception("获取物理位置出现错误:" + e.getMessage());
        finally {
            get.abort();
            client = null;
        }
 
        return resultString;
    }
     
    /** 显示结果 */
    private void showResult(SCell cell, String location) {
        TextView cellText = (TextView) findViewById(R.id.cellText);
        cellText.setText(String.format("基站信息:mcc:%d, mnc:%d, lac:%d, cid:%d",
                cell.MCC, cell.MNC, cell.LAC, cell.CID));
 
        TextView locationText = (TextView) findViewById(R.id.lacationText);
        locationText.setText("物理位置:" + location);
    }
}

 



 

我们连上手机在手机上运行程序看看。

不出意外的话程序运行起来了,自动跳转到了主界面。点击“Click Me”,出错了!

原来是没有权限,经过前面的学习,我们知道Android在应用的安全上下了一番功夫,要用一些特殊功能必须先报告,安装应用的时候列给用户看,必须要得到用户的允许。这里我们用了获取基站信息的功能,涉及到用户的隐私了,所以我们必须申明一下。

打开AndroidManifest.xml配置文件,在里面添加相应的配置信息:

 



 

1
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>

 



 

我们继续把网络连接的权限申明也加上:

 



 

1
<uses-permission android:name="android.permission.INTERNET"></uses-permission>

 



 

再编译运行看看(点击“Click Me”后程序会卡住,等待一段时间才有反应,取决于网络情况):

成功啦!

可能有的同学还是出现错误,没有成功:

█ 提示“www.google.com…”什么的错误

请确认你的手机能访问互联网,调用google的API是必须联网的。

█ 提示获取不到基站信息

你确定你是在手机上测试的吗?模拟器可不行哦。或者你的手机使用的CMDA网络?这个例子只支持GSM网络…

█ 获取不到经纬度

很有可能你中奖了,你所在的基站还没纳入google的数据库…(话说我之前也遇到过,怎么查就是查不出经纬度来,返回数据为空)

█ 获取到的地理地址不正确

这个可能程序出错了,可能google出错了?

其实google map API返回的数据中还包含了很多其他信息,我们可以用来开发一些更有趣的功能,如制作我们专属的地图软件、足迹记录软件等,充分发挥你的创造力:)

八、总结

这个程序基本实现了基站定位功能,但还有很多问题,如:点击了按钮后界面会卡住(访问网络时阻塞了进程)、未对异常进一步处理、不兼容CMDA网络等。

另外这个程序的精度也不够,获得的位置实际上是基站的物理位置,与人所在的位置还有一定差距。在城市里面,一般采用密集型的小功率基站,精度一般在几百米范围内,而在郊区常为大功率基站,密度很小,精度一般在几千米以上。

想要取得更高的精度需要通过一些其他的算法来实现,如果大家有兴趣的话我们可以一起来研究一下,再专门写篇笔记。

可见写一段程序和做一个实际的产品是有很大差别的。

九、程序代码

结尾

这一节基本实现了最简单的基站定位,只是作为学习的例子,远远达不到产品的要求,请大家见谅。

我们进一步熟悉了JAVA编码,之前没怎么接触JAVA看起来有点吃力的同学建议找点JAVA基础的书来看看。

话说我这段代码也是写得乱七八糟,JAVA没系统学习过,不知道JAVA编程的习惯,命名规则等,大家见笑了。

相对前面几节来说这一节没那么详细了,我也很困惑:详细点呢进度很慢,进度提上去了又怕不够详细,看看大家的意见,请留言哈。

下一节我们回归理论知识,一起深入学习Android的Activity组件。

如果你喜欢看到这些文字,请点击右下角的“推荐”支持我一下,谢谢!

分享到:
评论

相关推荐

    Android 获取当前位置,地区名称完整实

    在Android开发中,获取设备的当前位置以及地区名称是...以上就是Android获取当前位置及地区名称的完整实现过程中的关键知识点。在实际开发中,要根据应用的特性和用户需求,合理选择定位策略,同时关注性能和用户体验。

    android城市列表选择、定位

    在获取到位置信息后,通常会显示在地图上,并且可以将该位置与城市列表进行匹配,以确定并显示用户当前所在城市。 **最近访问和热门城市**功能则增加了用户体验的便捷性。最近访问记录保存用户最近选择的城市,可以...

    Android城市切换选择列表

    在Android开发中,城市切换选择列表是一个常见的功能,特别是在应用涉及到地理位置信息或者提供服务时,用户需要方便地选择他们的所在城市。本项目“Android城市切换选择列表”旨在提供一个高效、用户友好的解决方案...

    Android使用百度API定位当前城市得到城市名城

    在Android应用开发中,获取用户当前所在城市的地理位置信息是一项常见的需求。百度地图API提供了一种高效、便捷的方式来实现这一功能。本教程将详细介绍如何利用百度地图API来定位并获取当前城市的名字。 首先,...

    仿微信城市选择(android)

    省份通常会显示省份名称,而城市则会显示城市名。 - 添加一个头部布局,用于显示当前选中的省份和城市。 2. **数据结构**: - 数据通常以二维数组或链表的形式存储,第一层为省份,第二层为对应省份的城市。 - ...

    android 仿美团城市定位选择

    通常,这会包含一个搜索框供用户输入城市名,以及一个下拉列表或滚动视图显示所有可选城市。可以使用`RecyclerView`来展示城市列表,配合`Adapter`进行数据绑定。同时,利用`AutoCompleteTextView`作为搜索框,它...

    根据IP取城市名称

    2. 城市定位:通过IP地址获取城市名称是一种常见的地理定位方式。这通常涉及到DNS解析、IP库查询等技术。当用户访问网站或使用应用时,服务器可以通过其IP地址获取到大致的地理位置信息,从而实现个性化推荐、广告...

    Android城市选择控件

    "当前城市"通常指的是用户的设备所在地,这可以通过GPS定位或网络IP来获取。"热门城市"是指用户经常搜索或访问的一线城市,例如北京、上海、广州等,这些数据可以从服务器获取或者根据用户行为分析得出。"历史城市...

    安卓android城市切换的demo

    - 数据结构设计:每个城市条目可能包含城市名、省份、经纬度等信息,便于搜索和排序。 - 列表展示:使用`RecyclerView`控件展示城市列表,配合`Adapter`进行数据绑定,并实现滚动、点击等交互效果。 - 搜索功能:...

    Android通过原生方式获取经纬度与城市信息的方法

    在项目中需要获取用户所在位置的经纬度和城市上送给风控系统。一般来说,定位有两种方式: 用第三方SDK定位,如百度地图、高德地图、谷歌地图; 用Android原生SDK中的api定位; 本文讲述定位的第二种方式–用...

    android 城市选择三级联动

    在Android开发中,"城市选择三级联动"是一个常见的功能需求,尤其在应用涉及到地理位置信息时。这个功能通常涉及省、市、区三个级别的选择,用户可以通过级联的方式快速找到并选择他们所在的地区。下面我们将详细...

    Android三级城市选择demo

    这个功能在地图应用、电商应用、生活服务类应用等中非常常见,用于帮助用户精确地定位到他们所在的地理位置或者指定的送达地址。下面我们将详细探讨如何在Android中实现这样一个功能。 首先,我们需要一个数据结构...

    android 城市列表选择器 可以直接下载使用

    在Android开发中,有时我们需要实现一个功能,让用户能够选择他们所在的省份、城市和区县,这就是所谓的省市区三级联动选择器。"android 城市列表选择器 可以直接下载使用"是一个针对这种需求的解决方案,它提供了一...

    android demo,不适应GPS,通过网络定位到所在地的地址

    Android提供了`Location`服务,允许应用获取设备的位置信息。此Demo主要依赖于网络定位,而不是GPS。网络定位通常通过蜂窝网络信号塔或Wi-Fi接入点的数据来估算位置,相比GPS定位,它的优势在于不需要直接的卫星...

    android 城市选择器

    在本例中,`WheelView`被用于展示中国的省份和城市,让用户能够方便地滚动并选择他们所在的区域。 1. **WheelView的基本使用**:`WheelView`通常需要自定义数据源,如省份和城市列表。开发者需要创建一个包含所有...

    基于Android平台的城市天气预报源码 [修改]

    城市天气预报源码是针对Android平台开发的一款应用,旨在为用户提供实时、准确的气象信息,让用户可以方便地了解所在城市或任何指定城市的天气情况。这个源码经过了修改,可能包含了一些优化或新功能,使其更适合...

    Android 城市天气预报源码.rar

    Android系统提供了LocationManager和Geocoder类来获取用户的位置信息,源码可能利用这些API来定位用户所在城市并自动加载对应天气信息。 在数据存储方面,应用可能使用SharedPreferences进行轻量级的数据持久化,...

    Android 地区选择器,简洁集成,节省资源

    在Android应用开发中,地区选择器是一个常见的组件,用于让用户选择他们的城市或所在区域。"Android 地区选择器,简洁集成,节省资源"的标题表明这是一个专为开发者设计的解决方案,旨在简化集成过程,同时优化性能...

    CELLID_Android

    二、Android系统中的CELLID获取 在Android设备上,开发者可以通过以下API来获取当前连接的CELLID信息: 1. TelephonyManager类:提供对手机通话状态和电话功能的访问,包括getCellLocation()方法,它可以返回一个...

    城市选择器

    它通常包含国家、省份(或州)、城市三级或四级的选择,用户可以通过这个组件方便地筛选并选择他们所在的地区。在Android应用中,城市选择器可以设计为下拉菜单、弹出对话框或者滑动选择器等形式,以适应不同场景和...

Global site tag (gtag.js) - Google Analytics