我的应用程序在很多情况下都发出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();
            }

        }

最佳答案

单独检查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中保留一个应用程序布尔值,并在每次进行HTTP调用之前每次都检查该布尔值。

我发现仍然无法完全解决您的问题的实例,例如SocketTimeOutExceptions,其中用户的接收不良并且连接超时。

10-08 18:14