我明白让主UI线程等待是一个非常糟糕的主意,因为它会导致“Application Not Responding”消息。
但这是我的情景。
我正从我的Android应用程序调用我的服务器,我在其中发送用户的坐标,服务器使用该坐标相应地返回结果。 问题是在应用程序的首次运行期间,获取该位置需要很长时间。我正在寻找GPS和NETWORK提供商来检索位置,我在从GPS和NETWORK提供商处获取位置之前运行了20秒计时器。
这是我的班级,可以找到用户的位置。
public class MyLocation {
Timer timer1;
LocationManager lm;
LocationResult locationResult;
boolean gps_enabled=false;
boolean network_enabled=false;
public boolean getLocation(Context context, LocationResult result)
{
//use LocationResult callback class to pass location value from MyLocation to user code.
locationResult=result;
if(lm==null)
lm = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
//exceptions will be thrown if provider is not permitted.
try{gps_enabled=lm.isProviderEnabled(LocationManager.GPS_PROVIDER);}catch(Exception ex){}
try{network_enabled=lm.isProviderEnabled(LocationManager.NETWORK_PROVIDER);}catch(Exception ex){}
//don't start listeners if no provider is enabled
if(!gps_enabled && !network_enabled)
return false;
if(gps_enabled)
lm.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListenerGps);
if(network_enabled)
lm.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListenerNetwork);
timer1=new Timer();
timer1.schedule(new GetLastLocation(), 20000);
return true;
}
LocationListener locationListenerGps = new LocationListener() {
public void onLocationChanged(Location location) {
timer1.cancel();
locationResult.gotLocation(location);
lm.removeUpdates(this);
lm.removeUpdates(locationListenerNetwork);
}
public void onProviderDisabled(String provider) {}
public void onProviderEnabled(String provider) {}
public void onStatusChanged(String provider, int status, Bundle extras) {}
};
LocationListener locationListenerNetwork = new LocationListener() {
public void onLocationChanged(Location location) {
timer1.cancel();
locationResult.gotLocation(location);
lm.removeUpdates(this);
lm.removeUpdates(locationListenerGps);
}
public void onProviderDisabled(String provider) {}
public void onProviderEnabled(String provider) {}
public void onStatusChanged(String provider, int status, Bundle extras) {}
};
class GetLastLocation extends TimerTask {
@Override
public void run() {
lm.removeUpdates(locationListenerGps);
lm.removeUpdates(locationListenerNetwork);
Location net_loc=null, gps_loc=null;
if(gps_enabled)
gps_loc=lm.getLastKnownLocation(LocationManager.GPS_PROVIDER);
if(network_enabled)
net_loc=lm.getLastKnownLocation(LocationManager.NETWORK_PROVIDER);
//if there are both values use the latest one
if(gps_loc!=null && net_loc!=null){
if(gps_loc.getTime()>net_loc.getTime())
locationResult.gotLocation(gps_loc);
else
locationResult.gotLocation(net_loc);
return;
}
if(gps_loc!=null){
locationResult.gotLocation(gps_loc);
return;
}
if(net_loc!=null){
locationResult.gotLocation(net_loc);
return;
}
locationResult.gotLocation(null);
}
}
public void cancelTimer() {
timer1.cancel();
lm.removeUpdates(locationListenerGps);
lm.removeUpdates(locationListenerNetwork);
}
public static abstract class LocationResult{
public abstract void gotLocation(Location location);
}
}
在我的主要活动中,
onCreate(...)
{
.
.
.
.
myLocation.getLocation(context, locationResult); // my call to the MyLocation class to get the user location
sendRequestToServer(); // calls an AsyncTask which uses the coordinates returned by the MyLocation class
}
getLocation(context, locationResult)
将上下文传递给MyLocation类,结果存储在LocationResult callback class
。
所以,在我的主要活动中,我有
LocationResult locationResult = new LocationResult(){
@Override
public void gotLocation(Location location){
if(location != null)
{
mUser.setLastKnownLoc(location);
Log.d("UserLocation", location.getLatitude() + " " + location.getLongitude());
}
}
};
我对服务器的请求使用mUser.getLastKnownLoc()
来获取存储在结果中的位置值。
我的问题:让sendRequestToServer()等到我有位置
我尝试了什么:
myLocation.getLocation(..)
置于AsyncTask的doInBackground()
内并调用sendRequestToServer()
onPostExecute(),但sendRequestToServer仍然不会等待MyLocation类中的计时器完成。mTimer.schedule(RequestToserver,20000)
,以便仅在20秒后调用sendRequestToServer(在MyLocation finsihes中的另一个计时器时)。这似乎有点工作,因为请求仅在20秒后发出,但是由于一些线程处理问题,我的应用程序将崩溃。我试图在UIThread上运行timerTask并将Looper.prepare()连接到它。但是我遇到了一个线程处理错误的恶性循环,可能是因为sendRequestToServer()
调用了AsyncTask,而AsyncTask再次拥有它自己的后台线程。我一直试图将这项工作推迟一段时间。我只需要确保在20秒MyLocation计时器用完后调用AsyncTask。
来自您身边的任何帮助/建议真的很棒!!
答案 0 :(得分:1)
这是愚蠢的解决方案,有时候是work.anway尝试使用它
boolean x = ...... get location();
如果需要,它会在oncreate()中等待结果x.put这个。如果问题存在于帖子后面
答案 1 :(得分:1)
在你的onCreate中只显示阻止用户输入的ProgressDialog
,在后台运行你的getLocation,当它到达某个位置时,只需关闭ProgressDialog
并开始新的AsyncTask
发送数据。
像。
ProgressDialog pd = ...;
pd.show();
MyLocation ml = new MyLocation(new LocationResultsCallback(){
public void onLocationRecieved(Location loc){
//must run on UI thread
pd.dismiss();
new SendResultsTask(loc).execute();
}
});
ml.findLocation(); // start location finding
同样对于你的getLastKnownLocation,最好把它切换成这样的东西。
String bestProvider = locationManager.getBestProvider(new Criteria(), true);
if(bestProvider!=null)
Location local = locationManager.getLastKnownLocation(bestProvider);
如果这是适用的,我不知道你对位置提供商有什么限制。
而不是Timer,使用Handler更合适,但两者都适用于你。