2014-02-05 6 views
4

My DialogFragment содержит невидимую кнопку «ОК» и список элементов для клики. Когда нажимается какое-либо из элементов списка ListView, я устанавливаю видимость кнопки для VISIBLE.Почему этот Java-код работает? компилятор не жалуется на закрытие

Это делается через анонимный OnItemClickListener. Код ниже работает, но я не понимаю почему. Поскольку Java не поддерживает закрытие, я ожидаю, что компилятор будет жаловаться на то, что кнопка не является окончательной.

Это не типичный случай закрытия? Почему код ниже не создает ошибку компилятора?

Спасибо

public class AlternativeRoomsDialog extends DialogFragment { 

private Button okButton; 

static AlternativeRoomsDialog newInstance(String name) { 
    AlternativeRoomsDialog f = new AlternativeRoomsDialog(); 
    return f; 
} 

@Override 
public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) { 
    View view = inflater.inflate(R.layout.fragment_alternative_rooms, container); 
    getDialog().setTitle("Change Room"); 

    ListView lv = (ListView) view.findViewById(R.id.alternative_rooms_list); 
    final adapter = /*some initialization*/; 
    lv.setAdapter(adapter); 
    lv.setOnItemClickListener(new OnItemClickListener() { 
     @Override 
     public void onItemClick(AdapterView<?> arg0, View linlay, int position, long id) { 
      okButton.setVisibility(View.VISIBLE); 
      ListView lv = (ListView) linlay.getParent(); 
      int total = lv.getChildCount(); 
      for (int i=0; i< total; i++){ 
       lv.getChildAt(i).setBackgroundColor(Color.BLUE); 
      } 
      linlay.setBackgroundColor(Color.GREEN); 
     } 
    }); 

    // setup OK button 
    okButton = (Button) view.findViewById(R.id.btn_ok); 
    okButton.setOnClickListener(new OnClickListener() { 
     @Override 
     public void onClick(View v) { 
      Toast.makeText(AlternativeRoomsDialog.this.getActivity(), "ok button clicked", Toast.LENGTH_SHORT).show(); 
     } 
    }); 
    return view; 
} 

}

+4

Вы закрываете 'this', что (фактически)' final'. – SLaks

+0

привет @SLaks, спасибо за ваш ответ. Я не уверен, что я следую. Итак, вы говорите, что квалифицированный. Это ведет себя как final, потому что это экземпляр класса? – kouretinho

+0

Как локальная переменная 'final',' this' не может измениться. – SLaks

ответ

5

Требование переменных объявляется окончательным относится только к локальным переменным, поскольку они являются те, которые выходят за рамки, когда метод возвращается. На вашу кнопку ссылается переменная-член, и нет необходимости, чтобы она была окончательной: член не будет идти нигде, пока объект существует.

2

Вам не нужно указывать переменную экземпляра final, то есть только в случае локальных переменных. Локальные переменные должны быть final, они копируются в анонимный класс, который вы используете внутри метода, в качестве переменных экземпляра этого класса. Это делается потому, что анонимный внутренний экземпляр класса может оставаться в памяти кучи после возврата метода. Таким образом, может потребоваться доступ к переменной даже после того, как кадр стека метода будет удален.

Теперь, поскольку есть 2 копии локальных переменных, они могут выйти из синхронизации, если локальная переменная изменяется вне анонимного внутреннего класса. Это будет действительно странно, поэтому необходимо сделать локальную переменную final.

Как и в случае с переменными экземпляра, они вместо этого распределяются между экземпляром класса экземпляра и анонимным экземпляром внутреннего класса. Таким образом, вам не нужно делать их final.

Смежные вопросы