匿名类和final外部变量的实现

来源:互联网 发布:武汉大学网络管理中心 编辑:程序博客网 时间:2024/06/05 11:32
我们知道,在java中有大量的匿名类使用,而且在匿名类的函数中,我们还可以使用在外部声明的final变量,例如下面的android代码:

public class ArrowPopupWindowDemo extends Activity {    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.arrow_popup_window);        final Button arrowTop = (Button) findViewById(R.id.arrow_top);        arrowTop.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View v) {                final ListPopupWindow popup = new ListPopupWindow(ArrowPopupWindowDemo.this);                popup.setAdapter(createAdapter());                popup.setAnchorView(arrowTop);                popup.setWidth(400);                popup.setModal(true);                popup.setListSelector(getResources().getDrawable(android.R.color.transparent));                popup.setOnItemClickListener(new AdapterView.OnItemClickListener(){                    @Override                    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {                        popup.dismiss();                    }                });                ArrowPopupWindow popupWindow = popup.getPopupWindow();                popupWindow.setTitle(R.string.arrow_top);                popupWindow.setPositiveButton(android.R.string.ok, null);                popupWindow.setNegativeButton(android.R.string.cancel, null);                popup.show();            }        });     }     .....}

我们为Button arrowTop创建了一个listner,在内部类的onClick函数中,又使用了arrowTop这个final变量。


我们知道,内部匿名类也是一个单独独立的类,会别java编译器生成一个 <外部类名>$<N>的方式。这个例子里面,就是ArrowPopupWindowDemo$1。我们来反编译它的class,得到如下的信息

Compiled from "ArrowPopupWindowDemo.java"class com.example.miui.widget.popupwindow.ArrowPopupWindowDemo$1 implements android.view.View$OnClickListener {  final android.widget.Button val$arrowTop;  final com.example.miui.widget.popupwindow.ArrowPopupWindowDemo this$0;  com.example.miui.widget.popupwindow.ArrowPopupWindowDemo$1(com.example.miui.widget.popupwindow.ArrowPopupWindowDemo, android.widget.Button);    Code:       0: aload_0              1: aload_1              2: putfield      #1                  // Field this$0:Lcom/example/miui/widget/popupwindow/ArrowPopupWindowDemo;       5: aload_0              6: aload_2              7: putfield      #2                  // Field val$arrowTop:Landroid/widget/Button;      10: aload_0             11: invokespecial #3                  // Method java/lang/Object."<init>":()V      14: return        ....
根据反编译结果可以看到,java编译生成了两个隐含的域:val$arrowTop和 this$0,对应的就是外部的final arrowTop变量和外部的this变量,而且还为它创建了一个隐含的构造函数。这个构造函数也是隐含调用的,由编译器自动产生。



0 0
原创粉丝点击