Обработка изменений ориентации с помощью динамических счетчиков

Я искал в Интернете и пытался в течение 5 часов, но я не мог найти способ решить проблему. Я хочу сохранить состояние своих спиннеров после изменения ориентации.

У меня есть 2 счетчика, которые создаются динамически. Они получают свои элементы по http-запросу, а первый счетчик изменяет элементы во втором с помощью метода setOnItemSelectedListener(). Я читаю строки в глобальные списки.

public class Global {
    public static String userName;
    public static String userType;
    public static String serverIp;
    public static int spinnerLeaguePos=0;
    public static int spinnerMatchPos=0;
    public static List<String> leagues;
    public static List<String> matches;
}

Моей первой проблемой было предотвращение перезапуска активности.

android:configChanges="orientation|keyboardHidden"

Я внес это изменение в свой файл манифеста, чтобы самостоятельно обрабатывать изменения ориентации и избегать перезапуска.

Я нашел способ обработки изменений ориентации в Интернете, но у меня это не сработало. Я получаю исключение нулевого указателя, когда меняю ориентацию.

07-05 12:41:08.119: E/AndroidRuntime(26388): FATAL EXCEPTION: main
07-05 12:41:08.119: E/AndroidRuntime(26388): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example/com.example.MatchSelectionActivity}: java.lang.NullPointerException
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1956)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1981)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3351)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.access$700(ActivityThread.java:123)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1151)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.os.Handler.dispatchMessage(Handler.java:99)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.os.Looper.loop(Looper.java:137)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.main(ActivityThread.java:4424)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at java.lang.reflect.Method.invokeNative(Native Method)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at java.lang.reflect.Method.invoke(Method.java:511)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at dalvik.system.NativeStart.main(Native Method)
07-05 12:41:08.119: E/AndroidRuntime(26388): Caused by: java.lang.NullPointerException
07-05 12:41:08.119: E/AndroidRuntime(26388):    at com.example.MatchSelectionActivity.setLeagues(MatchSelectionActivity.java:90)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at com.example.MatchSelectionActivity.onCreate(MatchSelectionActivity.java:47)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.Activity.performCreate(Activity.java:4465)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1049)
07-05 12:41:08.119: E/AndroidRuntime(26388):    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1920)
07-05 12:41:08.119: E/AndroidRuntime(26388):    ... 12 more

Это то, что я сделал до сих пор. Я даже не уверен в том, как я пытаюсь это сделать. Поэтому любой более подходящий способ сделать это будет оценен.

@Override
protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);
    spinnerLeague = (Spinner) findViewById(R.id.spinner_league);
    spinnerMatch = (Spinner) findViewById(R.id.spinner_match);

    if (savedInstanceState != null) {
        setLeagues();
        setMatches();
        spinnerLeague.setSelection(Global.spinnerLeaguePos);
        spinnerMatch.setSelection(Global.spinnerMatchPos);
    }
    else{
        Global.leagues = new ArrayList<String>();
        Global.matches = new ArrayList<String>();

        Global.leagues.add(getString(R.string.league_select));
        Global.matches.add(getString(R.string.match_select));

        GetLeagues task = new GetLeagues();
        String requestString = "http://" + Global.serverIp + ":8080/server/GetCurrentLeagues";
        task.execute(new String[] { requestString });
    }

    setContentView(R.layout.match_selection_layout);

}

public class GetLeagues extends AsyncTask<String, Void, String[]> {
    @Override
    protected String[] doInBackground(String... urls) {
        //this part works properly, and reads the leagues into Global.leagues List
    }

    @Override
    protected void onPostExecute(String[] result) {
        setLeagues();
    }
}
public class GetMatches extends AsyncTask<String, Void, String[]> {
    @Override
    protected String[] doInBackground(String... urls) {
        //this part works properly, and reads the matches into Global.matches List
    }

    @Override
    protected void onPostExecute(String[] result) {
        setMatches();
    }
}

 public void setLeagues() {

    spinnerLeague = (Spinner) findViewById(R.id.spinner_league);
    ArrayAdapter<String> dataAdapter = new ArrayAdapter<String>(this,android.R.layout.simple_spinner_item, Global.leagues);
    dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
    spinnerLeague.setAdapter(dataAdapter);

    spinnerLeague.setOnItemSelectedListener(new OnItemSelectedListener() {

        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {

            spinnerLeague = (Spinner) findViewById(R.id.spinner_league);

            if(position!=0){
                spinnerMatch.setEnabled(true);
                Global.spinnerLeaguePos=position;

                GetMatches task = new GetMatches();
                String requestString = "http://" + Global.serverIp +":8080/server/GetCurrentMatches/"+Global.spinnerLeaguePos;
                task.execute(new String[] { requestString });
            }
            else{
                setMatches();
                spinnerMatch.setEnabled(false);
                Global.spinnerLeaguePos=position;
            }

        }

        public void onNothingSelected(AdapterView<?> parent) {
            // TODO Auto-generated method stub

        }
    });
  }

    public void setMatches() {

    spinnerMatch = (Spinner) findViewById(R.id.spinner_match);
    ArrayAdapter<String> dataAdapter = new ArrayAdapter<String>(this,android.R.layout.simple_spinner_item, Global.matches);
    dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
    spinnerMatch.setAdapter(dataAdapter);


    spinnerMatch.setOnItemSelectedListener(new OnItemSelectedListener() {

        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
            Global.spinnerMatchPos=position;
        }

        public void onNothingSelected(AdapterView<?> parent) {
            // TODO Auto-generated method stub

        }
    });
  }

person Srht    schedule 05.07.2012    source источник
comment
было бы полезно, если бы вы напечатали всю трассировку стека исключения nullpointer.   -  person RaB    schedule 05.07.2012
comment
возможно, вам стоит взглянуть на метод onConfigChanged(). Это позволяет вам очень легко справляться с изменениями конфигурации.   -  person pogo2065    schedule 05.07.2012


Ответы (1)


Вам нужно позвонить setContentView(R.layout.match_selection_layout), прежде чем звонить findViewById. В противном случае findViewById вернет null (поскольку родитель View еще не был надут/прикреплен к экрану).

Также вам не следует не использовать configChanges в качестве пластыря, чтобы зафиксировать свою активность. Есть много способов перезапустить Activity; например, если пользователь изменил языковые настройки устройства с английского на испанский (это приведет к уничтожению и созданию всего Activity). Всегда следите за тем, чтобы изменения ориентации работали правильно... не отказывайтесь от них и просто переопределяйте onConfigurationChanged.

person Alex Lockwood    schedule 05.07.2012
comment
настройка просмотра содержимого, похоже, работает. Но я думаю, что мне следует использовать метод public void onConfigurationChanged (Configuration newConfig). Есть ли какой-нибудь полезный учебник по его использованию? - person Srht; 05.07.2012
comment
по какой причине вы используете configChanges? Я постараюсь найти его, но у вас должна быть веская причина для его использования, прежде чем вы это сделаете... - person Alex Lockwood; 05.07.2012
comment
Я хочу обрабатывать только изменения ориентации. Мне нужно изменить отображение экрана в зависимости от ориентации. Если вы спросите, почему я не запрещаю менять ориентацию, я думаю, что пользователям будет лучше использовать приложение так, как они хотят. - person Srht; 05.07.2012
comment
@Srht ты никогда, никогда, никогда не расскажешь кому-нибудь, никогда, сколько бы я ни жил, ( всегда, всегда, всегда), чтобы запретить изменение ориентации экрана. Я ненавижу это больше всего на свете... это просто лень со стороны разработчиков, лол - person Alex Lockwood; 05.07.2012
comment
Вместо этого вы можете создать два отдельных макета с одинаковыми именами (один для ландшафтного режима и один для портретного режима). Затем поместите книжную ориентацию в res/layout (это вариант по умолчанию, запасной макет) и поместите альбомную ориентацию в res/layout-land. Это гарантирует, что во время выполнения будет выбран правильный макет. - person Alex Lockwood; 05.07.2012
comment
Программная обработка изменения конфигурации обычно требует гораздо больше работы, чем вам нужно. Использование статических макетов/папок xml значительно упрощает задачу. - person Alex Lockwood; 05.07.2012
comment
Итак, я не хочу лениться и обрабатывать изменения ориентации. Я много искал полезного учебника, но ничего не нашел. Мне нужно простое понимание метода onConfigurationChanged. Вносит ли он изменения в текущую активность или дает программисту возможность сохранить свои данные перед перезапуском активности? - person Srht; 05.07.2012
comment
Вносит ли он какие-либо изменения в что? - person Alex Lockwood; 05.07.2012
comment
Может быть, вам следует объяснить, что именно вы пытаетесь сделать. - person Alex Lockwood; 05.07.2012
comment
Я не могу перейти в чат из-за своей репутации и не хочу заморачиваться здесь комментариями. Может быть, мы можем общаться по электронной почте или как-то еще (если вы не возражаете). - person Srht; 05.07.2012
comment
давайте продолжим это обсуждение в чате - person Alex Lockwood; 05.07.2012