我是Android Development的初学者。我想创建一个Android应用程序,它将GPS和网络位置(lat& long)发送到我的服务器(静态IP)。 我找到了以下代码,它成功运行,数据接收在已经在服务器端制作的应用程序(现在它与客户端应用程序无关)
点击我的Android应用上的“发送位置”按钮,在我的服务器上收到数据。
GET /31.4244456/74.7349772 HTTP / 1.1
主持人:180.178.169.77:6081
连接:keep-Alive
User-Agent:Apache-HttpClient / Unavailable(java 1.4)
180.178.169.77是我的服务器IP,6081是端口
此时一切正常,我几乎实现了我的想法,但我面临的真正问题是我的Android应用程序在将此数据发送到服务器后挂起并崩溃。如果有人有想法,请解决我的问题。
这是我的AddressActivity.java
package com.gideon.address;
import java.io.IOException;
import java.util.List;
import java.util.Locale;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import android.app.Activity;
import android.content.Context;
import android.location.Address;
import android.location.Geocoder;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
public class AddressActivity extends Activity {
/** Called when the activity is first created. */
String lat="", lon="";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button btnLocation = (Button)findViewById(R.id.btnLocation);
btnLocation.setOnClickListener(new OnClickListener() {
public void onClick(View arg0) {
// Acquire a reference to the system Location Manager
LocationManager locationManager = (LocationManager) AddressActivity.this.getSystemService(Context.LOCATION_SERVICE);
// Define a listener that responds to location updates
LocationListener locationListener = new LocationListener() {
public void onLocationChanged(Location location) {
// Called when a new location is found by the network location provider.
lat = Double.toString(location.getLatitude());
lon = Double.toString(location.getLongitude());
TextView tv = (TextView) findViewById(R.id.txtLoc);
tv.setText("Your Location is:" + lat + "--" + lon);
}
public void onStatusChanged(String provider, int status, Bundle extras) {}
public void onProviderEnabled(String provider) {}
public void onProviderDisabled(String provider) {}
};
// Register the listener with the Location Manager to receive location updates
locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListener);
}
});
Button btnSend = (Button)findViewById(R.id.btnSend);
btnSend.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
postData(lat, lon);
}
});
Button btnAdd = (Button)findViewById(R.id.btnAddress);
btnAdd.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
TextView tv = (TextView)findViewById(R.id.txtAddress);
tv.setText(GetAddress(lat, lon));
}
});
}
public void postData(String la, String lo) {
// Create a new HttpClient and Post Header
HttpClient httpclient = new DefaultHttpClient();
//HttpGet htget = new HttpGet("http://myappurl.com/Home/Book/"+la+"/"+lo);
HttpGet htget = new HttpGet("http://180.178.169.77:6081/"+la+"/"+lo);
try {
// Execute HTTP Post Request
HttpResponse response = httpclient.execute(htget);
String resp = response.getStatusLine().toString();
Toast.makeText(this, resp, 5000).show();
} catch (ClientProtocolException e) {
Toast.makeText(this, "Error", 5000).show();
} catch (IOException e) {
Toast.makeText(this, "Error", 5000).show();
}
}
public String GetAddress(String lat, String lon)
{
Geocoder geocoder = new Geocoder(this, Locale.ENGLISH);
String ret = "";
try {
List<Address> addresses = geocoder.getFromLocation(Double.parseDouble(lat), Double.parseDouble(lon), 1);
if(addresses != null) {
Address returnedAddress = addresses.get(0);
StringBuilder strReturnedAddress = new StringBuilder("Address:\n");
for(int i=0; i<returnedAddress.getMaxAddressLineIndex(); i++) {
strReturnedAddress.append(returnedAddress.getAddressLine(i)).append("\n");
}
ret = strReturnedAddress.toString();
}
else{
ret = "No Address returned!";
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
ret = "Can't get Address!";
}
return ret;
}
}
这是我的AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.gideon.address"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="8" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.INTERNET"/>
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:name=".AddressActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
答案 0 :(得分:2)
将postData
从主线程移到辅助线程。请在Async
任务中查看示例:http://developer.android.com/reference/android/os/AsyncTask.html
答案 1 :(得分:0)
你的问题是那个
public void onClick(View v) {
postData(lat, lon);
}
在UI线程上运行。这意味着当您执行“发布”操作时,UI不会刷新且无法访问 - 您的应用会在帖子结束之前陷入困境。
您需要执行以下操作:
public void onClick(View v) {
PostAsyncTask postAsyncTask = new PostAsyncTask();
postAsyncTask.execute();
}
private class PostAsyncTask extends AsyncTask<URL, Integer, Long> {
protected Long doInBackground(URL... urls) {
postData(lat, lon);
return 0;
}
protected void onPreExecute() {
// show progress dialog
}
protected void onPostExecute(Long result) {
// hide progress dialog
}
}
答案 2 :(得分:0)
问题的解决方案是:
添加了新帖子
new Thread(new Runnable() {
public void run() {
// your code goes here
postData(lat, lon);
}
}).start();
}
,代码如下:
Button btnSend = (Button)findViewById(R.id.btnSend);
btnSend.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
new Thread(new Runnable() {
public void run() {
// your code goes here
postData(lat, lon);
}
}).start();
}
});
我对AddressActivity.java的完整代码如下:
package com.gideon.address;
import java.io.IOException;
import java.util.List;
import java.util.Locale;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.DefaultHttpClient;
import android.app.Activity;
import android.content.Context;
//import android.location.Address;
//import android.location.Geocoder;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
public class AddressActivity extends Activity {
/** Called when the activity is first created. */
String lat="", lon="";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button btnLocation = (Button)findViewById(R.id.btnLocation);
btnLocation.setOnClickListener(new OnClickListener() {
public void onClick(View arg0) {
// Acquire a reference to the system Location Manager
LocationManager locationManager = (LocationManager) AddressActivity.this.getSystemService(Context.LOCATION_SERVICE);
// Define a listener that responds to location updates
LocationListener locationListener = new LocationListener() {
public void onLocationChanged(Location location) {
// Called when a new location is found by the network location provider.
lat = Double.toString(location.getLatitude());
lon = Double.toString(location.getLongitude());
TextView tv = (TextView) findViewById(R.id.txtLoc);
tv.setText("Your Location is:" + lat + "--" + lon);
}
public void onStatusChanged(String provider, int status, Bundle extras) {}
public void onProviderEnabled(String provider) {}
public void onProviderDisabled(String provider) {}
};
// Register the listener with the Location Manager to receive location updates
locationManager.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListener);
}
});
Button btnSend = (Button)findViewById(R.id.btnSend);
btnSend.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
new Thread(new Runnable() {
public void run() {
// your code goes here
postData(lat, lon);
}
}).start();
}
});
// Button btnAdd = (Button)findViewById(R.id.btnAddress);
//btnAdd.setOnClickListener(new OnClickListener() {
//public void onClick(View v) {
//TextView tv = (TextView)findViewById(R.id.txtAddress);
//tv.setText(GetAddress(lat, lon));
//}
//});
}
public void postData(String la, String lo) {
// Create a new HttpClient and Post Header
HttpClient httpclient = new DefaultHttpClient();
HttpPost httppost = new HttpPost("http://180.178.169.77:6081/"+la+"/"+lo);
try {
// Execute HTTP Post Request
//HttpResponse response = httpclient.execute(htget);
HttpResponse response = httpclient.execute(httppost);
//String resp = response.getStatusLine().toStrinkatyg(); try this now
//Toast.makeText(this, resp, 5000).show();
} catch (ClientProtocolException e) {
//Toast.makeText(this, "Error", 5000).show();
}
catch (IOException e) {
//Toast.makeText(this, "Error", 5000).show();
}
}
//public String GetAddress(String lat, String lon)
//{
//Geocoder geocoder = new Geocoder(this, Locale.ENGLISH);
// String ret = "";
//try {
//List<Address> addresses = geocoder.getFromLocation(Double.parseDouble(lat), Double.parseDouble(lon), 1);
//if(addresses != null) {
//Address returnedAddress = addresses.get(0);
//StringBuilder strReturnedAddress = new StringBuilder("Address:\n");
//for(int i=0; i<returnedAddress.getMaxAddressLineIndex(); i++) {
//strReturnedAddress.append(returnedAddress.getAddressLine(i)).append("\n");
//}
//ret = strReturnedAddress.toString();
//}
//else{
//ret = "No Address returned!";
//}
//} catch (IOException e) {
// TODO Auto-generated catch block
//e.printStackTrace();
//ret = "Can't get Address!";
//}
//return ret;
//}
}
我不需要地址,地理位置功能,因此我评论了与之相关的所有内容。 这是我的第一个Android应用程序,所以如果我错了,请指导我,告诉我如何改进它。
如果出现以下情况,请帮助我:
1)我想删除发送位置按钮,以便我的位置在定义的时间间隔(例如10分钟)内定期自动进入我的服务器。
第二件事:
2)我希望我的应用程序选择最佳提供商,GPS或GSM网络,如果没有,请使用另一个发送位置。