使用HTTP请求处理飞行模式

时间:2013-09-07 00:42:58

标签: android android-asynctask http-post airplane

我的应用程序有很多时候会发出HTTP帖子请求。我在测试期间发现,如果用户在发出此请求时处于飞行模式,则应用程序崩溃。因此,为了优雅地处理这种情况,我调用了一个函数,用于在进行AsyncTask调用之前测试手机是否处于飞行模式。如果函数返回true,则不进行AsyncTask调用。

测试本身正常工作。用户被通知他们必须关闭飞行模式,然后他们回来再试一次。问题是在飞机模式关闭后,AsyncTask继续进行,我的HTTP帖子会崩溃应用程序。如果飞机模式从未开启过,那么一切都顺利进行。

我无法弄清楚该怎么做。是否有更好的方法来测试飞机模式?或者在阻止AsyncTask恢复系统状态以使http请求成功后,我还需要做些什么。

任何建议表示赞赏。这里有一些代码可以让这个更清楚。如果有其他代码可以提供帮助,请告诉我们。谢谢!

public class LoginActivity extends Activity {
        @Override
        protected void onCreate(Bundle savedInstanceState) {
                super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_login);

            context = this;

            btnLogin.setOnClickListener(new View.OnClickListener() {
                public void onClick(View view) {
                UserFunctions userFunction = new UserFunctions(context);
                if (userFunction.isAirplaneModeOn(context)) {
                    userFunction.warnAboutAirplaneMode(context);
                    return;
                }

                new Login().execute();  

            });
    }

    class Login extends AsyncTask<String, String, String> {

        private JSONParser jsonParser = new JSONParser();
        private JSONObject json;

        @Override
        protected String doInBackground(String... args) {

            String URL = context.getResources().getString(R.string.dbURL) + context.getResources().getString(R.string.attemptLogin_php);

                    //this next line is the one that crashes - iff the user was
                    // previously in airplane mode. If there weren't ever in airplane
                    // mode, this runs just fine
            json = jsonParser.makeHttpRequest(URL, params);

                    ...

    }
}

这是UserFunctions类:

    public class UserFunctions {

            @SuppressLint( "NewApi" )
            @SuppressWarnings("deprecation")
            public boolean isAirplaneModeOn(Context context) {

                if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1){
                    return Settings.Global.getInt(context.getContentResolver(),
                       Settings.Global.AIRPLANE_MODE_ON, 0) != 0;
                } else {
                    return Settings.System.getInt(context.getContentResolver(),
                           Settings.System.AIRPLANE_MODE_ON, 0) != 0;
                }

            }

            public void warnAboutAirplaneMode(Context context) {

                AlertDialog.Builder builder = new AlertDialog.Builder(context)
                    .setTitle("Airplane Mode Is On")
                    .setIcon(R.raw.airplane_mode)
                    .setCancelable(false)
                    .setMessage("Your phone is in airplane mode. Please turn this off then try again.")

                    .setNeutralButton("OK", new DialogInterface.OnClickListener() {
                        public void onClick(DialogInterface dialog, int id) {
                            dialog.cancel();
                        }

                    });

                AlertDialog alert = builder.create();
                alert.show();   
            }

        }

1 个答案:

答案 0 :(得分:1)

单独检查Airplane mode可能不是一个好主意,因为你可以在飞机模式下启用wifi。但是可能有一些可能的答案,但我所做的是注册一个BroadcastReceiver,用以下权限来监听网络状态变化

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

我的接收器在清单

中看起来像这样
<receiver android:name="com.example.NetworkReceiver" >
        <intent-filter>
            <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
        </intent-filter>
    </receiver>

当它被解雇时我检查是否有网络连接并在SharedPreferences中保持app宽布尔值,并且每次在我进行HTTP呼叫之前检查该布尔值。

我找到的情况仍然无法完全解决您的问题,例如SocketTimeOutExceptions用户接收错误且连接超时。