java的内部类、匿名类本来以为自己用的已经很溜了, 结果, 就在昨天晚上12点来钟发生了重大事故。要说事故的严重性呢,那就是导致我一晚上没有睡着觉。
那下面先用一段模拟代码来描述下我出现的问题的:
public class Test { public static void main(String[] args) throws InterruptedException { View v = new View(); v.show(1); Thread.sleep(500); v.mTextView.execute(); Thread.sleep(1000); v.show(2); Thread.sleep(500); v.mTextView.execute(); } } class View { public TextView mTextView; public void show(int position) { if(mTextView == null) { mTextView = new TextView(); mTextView.setListener(new Listener() { @Override public void click() { System.out.println("position = " + position); } }); } mTextView.show(); } } class TextView { private Listener mListener; public void setListener(Listener l) { mListener = l; } public void execute() { mListener.click(); } public void show() { System.out.println("textview show..."); } } interface Listener { public void click(); }
从打印的结果上看, 我猜想肯定是在这个内部类的实例中保存了position参数,那带着这个猜想,我们来debug一下程序。
这是第一次执行到的时候, 发现什么问题了吗。 在mListener中竟然有一个和position相关的变量。到这里,我们感觉那个猜测可能是正确的。再往下思考,既然在mListener对象中保存了这个变量,那么下次执行到,同一个对象,所以变量肯定是相同的,这样也就解开我们的疑惑了。
总结一下:
在我们new一个匿名内部类的时候,如果使用了方法中的东西,那么jvm会给我们的匿名类加一个构造方法,并且将这个参数传递进来,例如上面的例子中:
class View$Listener { public View$Listener(int position) { this.Listener$position = position; } public void click() { ... } }
public class Test { public static void main(String[] args) throws InterruptedException { View v = new View(); v.show(1); Thread.sleep(500); v.mTextView.execute(); Thread.sleep(1000); v.show(2); Thread.sleep(500); v.mTextView.execute(); } } class View { public TextView mTextView; public void show(int position) { if(mTextView == null) { mTextView = new TextView(); // mTextView.setListener(new Listener() { // @Override // public void click() { // System.out.println("position = " + position); // } // }); } mTextView.setListener(new Listener() { @Override public void click() { System.out.println("position = " + position); } }); mTextView.show(); } } class TextView { private Listener mListener; public void setListener(Listener l) { mListener = l; } public void execute() { mListener.click(); } public void show() { System.out.println("textview show..."); } } interface Listener { public void click(); }
原文地址:http://blog.csdn.net/qibin0506/article/details/44257309