定时器用完后调用AsyncTask

时间:2013-03-31 11:32:09

标签: android multithreading android-asynctask

我明白让主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类中的计时器完成。
  • 在onCreate中创建另一个计时器并设置mTimer.schedule(RequestToserver,20000),以便仅在20秒后调用sendRequestToServer(在MyLocation finsihes中的另一个计时器时)。这似乎有点工作,因为请求仅在20秒后发出,但是由于一些线程处理问题,我的应用程序将崩溃。我试图在UIThread上运行timerTask并将Looper.prepare()连接到它。但是我遇到了一个线程处理错误的恶性循环,可能是因为sendRequestToServer()调用了AsyncTask,而AsyncTask再次拥有它自己的后台线程。

我一直试图将这项工作推迟一段时间。我只需要确保在20秒MyLocation计时器用完后调用AsyncTask。

来自您身边的任何帮助/建议真的很棒!!

2 个答案:

答案 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更合适,但两者都适用于你。