- 浏览: 699331 次
- 性别:
- 来自: 西安
文章分类
最新评论
-
cuqing:
laogao598599 写道匿名内部类不一定是局部内部类,还 ...
匿名内部类访问外部类中的局部变量必须是final属性 -
laogao598599:
匿名内部类不一定是局部内部类,还可以作为成员变量
匿名内部类访问外部类中的局部变量必须是final属性 -
stand_star:
非常给力!
struts2验证规则配置文件命名方式 -
wenlongsust:
嗯, 将返回值修改为html的确就可以了
EasyUI form ajax submit后,在IE下提示下载内容的解决办法 -
勇往直前wwt:
这样是自动增长,但每次id还得插入,如何只插入其他字段而让id ...
把主键定义为自动增长标识符类型
如果定义一个匿名内部类,并且希望它使用一个在其外部定的对象,那么编译器会要求其参数引用是final的。
1.匿名内部类肯定是局部内部类(在一个方法里面定义的内部类),因为在java中,语句必须写在方法里,而匿名内部类其实就是一条特殊的语句;
2.外部给定的对象就是所有外来的对象:外部方法的形参、局部变量、基本类型或自定义类型等。
3.内部类很微妙,它可以直接访问外部类的private field,这打破了类的封装。但是内部类又有它自身的好处,比如简洁,可见性等,于是就把它定位成“只读”,也就是final。不过这个保护也非常脆弱!
4.local inner class访问local var时,那个var必须是final的。因为可以通过enclosing class访问那个local var,也可以通过inner class访问,可能造成问题,所以就必须是final的
5.匿名内部类为什么只能用final.是变量的作用域的问题,因为匿名内部类是出现在一个方法的内部的,如果它要访问这个方法的参数或者方法中定义的变量,则这些参数和变量必须被修饰为final。因为虽然匿名内部类在方法的内部,但实际编译的时候,内部类编译成Outer.Inner,这说明内部类所处的位置和外部类中的方法处在同一个等级上,外部类中的方法中的变量或参数只是方法的局部变量,这些变量或参数的作用域只在这个方法内部有效。因为编译的时候内部类和方法在同一级别上,所以方法中的变量或参数只有为final,内部类才可以引用。
6.例子
public class LocalInnerClassTest{
public static void main(String[] args){
Outer obj=new Outer(); //生成一个外部类对象
//调用外部类中的outer()方法,返回一个SuperInner类型对象赋值给si
SuperInner si=obj.outer();
si.m1(); //调用被覆盖的方法m1(),输出:Inner's m1() 20
}
}
/**
*定义一个接口SuperInner,内部定义一个抽象方法m1(),无返回类型
*/
interface SuperInner{
public void m1();
}
/**
*定义一个类Outer,内部只定义一个方法outer(),返回类型为SuperInner
*/
class Outer{
public SuperInner outer(){
int a=10; //方法中定义一个局部变量a,并赋值为10
final int b=20; //再定义一个final局部变量b,初始化为20
//在outer()方法中定义一个局部内部类Inner,实现接口SuperInner
class Inner implements SuperInner{
public void m1(){ //类中只有一个覆盖接口SuperInner的方法m1()
System.out.println("Inner's m1()"+a); //编译报错
//编译通过,输出:Inner's m1() 20
System.out.println("Inner's m1() "+b);
}
}
return new Inner();
}
}
我们先从主方法开始看代码的执行顺序,先生成一个Outer类对象obj,obj调用本类中方法outer();程序开始跳到outer()方法内执行程序语句,先后生成局部变量a和b,再定义一个局部内部类Inner,返回一个SuperInner类型的对象。将返回的SuperInner类型对象地址传给SuperInner类型对象si。si再调用m1()方法,因为已经在局部内部类中覆盖了接口中的m1()方法,所以将调用局部内部类中的m1() 方法,程序跳到局部内部类中m1()方法内执行程序语句,先输出一段字符串和a,结果编译报错,先 将这条程序语句隐藏,执行下面的语句,你会发现编译通过而且输出Inner's m1() 20!
为什么会这样呢?大家都知道局部变量仅仅在一个范围内有效,在方法调用完就被内存释放,在Outer类对象obj调用outer()方法时,a和b才产生,调用结束后被内存释放,那么b这个值也就不复存在了,为什么还会输出20呢?难道局部变量被final修饰就不会被内存释放而保留?
其实有部分操作对于程序员是透明的,那是JAVA语言开发者的小把戏,在定义a和b 时JVM(JAVA虚拟机)做了程序员看不到的操作,他将b拷贝了一份给局部内部类,也就是说JVM在局部内部类中定义了一个final int b=20;这个操作程序员是不知道的!当调用m1()方法时输出的20并不是原来outer()方法中定义的b,而仅仅是JVM拷贝的一个副本。那么为什么a没被打印出呢?那是因为JVM并没有拷贝它,因为没有final修饰,说明它可以被修改,如果把a 改为 a++,此时JVM就不知道拷贝a还是a++了,所以对于无final修饰的局部变量JVM是不会拷贝传给局部内部类的,自然无法打输出!
请参考这篇 http://www.cnblogs.com/nliao/p/3308690.html。
匿名内部类不一定是局部内部类。
List shippingSrvs = (List) CollectionUtils.select(serviceDetails, new Predicate() {
public boolean evaluate(Object object) {
QuotationDetail serviceDetail = (QuotationDetail) object;
assert serviceDetail.getItemMaster() != null;
return StringUtils.contains(" Shipping Cost ", serviceDetail.getItemMaster().getItemBasic().getFullName()) && serviceDetail.getIncludingShippingFee() == 1;
}
});
//new Predicate(){...}显然不是局部内部类。
局部内部类也不一定是匿名内部类。
class LocalInner{
public void execute(){
final int a = 1;
/**
* 创建局部内部类
*/
class InnerClass{
public void execute(){
System.out.println("LocalInner Class");
//局部内部类只能访问final类型的变量
System.out.println(a);
}
}
//只能在所在方法区域创建
new InnerClass().execute();
//new InnerClass()为匿名内部类 若是创建一个对象引用指向它,则为非匿名内部类。 此种理解是否有待验证? pls correct it.
}
}
1.匿名内部类肯定是局部内部类(在一个方法里面定义的内部类),因为在java中,语句必须写在方法里,而匿名内部类其实就是一条特殊的语句;
2.外部给定的对象就是所有外来的对象:外部方法的形参、局部变量、基本类型或自定义类型等。
3.内部类很微妙,它可以直接访问外部类的private field,这打破了类的封装。但是内部类又有它自身的好处,比如简洁,可见性等,于是就把它定位成“只读”,也就是final。不过这个保护也非常脆弱!
4.local inner class访问local var时,那个var必须是final的。因为可以通过enclosing class访问那个local var,也可以通过inner class访问,可能造成问题,所以就必须是final的
5.匿名内部类为什么只能用final.是变量的作用域的问题,因为匿名内部类是出现在一个方法的内部的,如果它要访问这个方法的参数或者方法中定义的变量,则这些参数和变量必须被修饰为final。因为虽然匿名内部类在方法的内部,但实际编译的时候,内部类编译成Outer.Inner,这说明内部类所处的位置和外部类中的方法处在同一个等级上,外部类中的方法中的变量或参数只是方法的局部变量,这些变量或参数的作用域只在这个方法内部有效。因为编译的时候内部类和方法在同一级别上,所以方法中的变量或参数只有为final,内部类才可以引用。
6.例子
public class LocalInnerClassTest{
public static void main(String[] args){
Outer obj=new Outer(); //生成一个外部类对象
//调用外部类中的outer()方法,返回一个SuperInner类型对象赋值给si
SuperInner si=obj.outer();
si.m1(); //调用被覆盖的方法m1(),输出:Inner's m1() 20
}
}
/**
*定义一个接口SuperInner,内部定义一个抽象方法m1(),无返回类型
*/
interface SuperInner{
public void m1();
}
/**
*定义一个类Outer,内部只定义一个方法outer(),返回类型为SuperInner
*/
class Outer{
public SuperInner outer(){
int a=10; //方法中定义一个局部变量a,并赋值为10
final int b=20; //再定义一个final局部变量b,初始化为20
//在outer()方法中定义一个局部内部类Inner,实现接口SuperInner
class Inner implements SuperInner{
public void m1(){ //类中只有一个覆盖接口SuperInner的方法m1()
System.out.println("Inner's m1()"+a); //编译报错
//编译通过,输出:Inner's m1() 20
System.out.println("Inner's m1() "+b);
}
}
return new Inner();
}
}
我们先从主方法开始看代码的执行顺序,先生成一个Outer类对象obj,obj调用本类中方法outer();程序开始跳到outer()方法内执行程序语句,先后生成局部变量a和b,再定义一个局部内部类Inner,返回一个SuperInner类型的对象。将返回的SuperInner类型对象地址传给SuperInner类型对象si。si再调用m1()方法,因为已经在局部内部类中覆盖了接口中的m1()方法,所以将调用局部内部类中的m1() 方法,程序跳到局部内部类中m1()方法内执行程序语句,先输出一段字符串和a,结果编译报错,先 将这条程序语句隐藏,执行下面的语句,你会发现编译通过而且输出Inner's m1() 20!
为什么会这样呢?大家都知道局部变量仅仅在一个范围内有效,在方法调用完就被内存释放,在Outer类对象obj调用outer()方法时,a和b才产生,调用结束后被内存释放,那么b这个值也就不复存在了,为什么还会输出20呢?难道局部变量被final修饰就不会被内存释放而保留?
其实有部分操作对于程序员是透明的,那是JAVA语言开发者的小把戏,在定义a和b 时JVM(JAVA虚拟机)做了程序员看不到的操作,他将b拷贝了一份给局部内部类,也就是说JVM在局部内部类中定义了一个final int b=20;这个操作程序员是不知道的!当调用m1()方法时输出的20并不是原来outer()方法中定义的b,而仅仅是JVM拷贝的一个副本。那么为什么a没被打印出呢?那是因为JVM并没有拷贝它,因为没有final修饰,说明它可以被修改,如果把a 改为 a++,此时JVM就不知道拷贝a还是a++了,所以对于无final修饰的局部变量JVM是不会拷贝传给局部内部类的,自然无法打输出!
评论
3 楼
cuqing
2016-08-12
laogao598599 写道
匿名内部类不一定是局部内部类,还可以作为成员变量
请参考这篇 http://www.cnblogs.com/nliao/p/3308690.html。
匿名内部类不一定是局部内部类。
List shippingSrvs = (List) CollectionUtils.select(serviceDetails, new Predicate() {
public boolean evaluate(Object object) {
QuotationDetail serviceDetail = (QuotationDetail) object;
assert serviceDetail.getItemMaster() != null;
return StringUtils.contains(" Shipping Cost ", serviceDetail.getItemMaster().getItemBasic().getFullName()) && serviceDetail.getIncludingShippingFee() == 1;
}
});
//new Predicate(){...}显然不是局部内部类。
局部内部类也不一定是匿名内部类。
class LocalInner{
public void execute(){
final int a = 1;
/**
* 创建局部内部类
*/
class InnerClass{
public void execute(){
System.out.println("LocalInner Class");
//局部内部类只能访问final类型的变量
System.out.println(a);
}
}
//只能在所在方法区域创建
new InnerClass().execute();
//new InnerClass()为匿名内部类 若是创建一个对象引用指向它,则为非匿名内部类。 此种理解是否有待验证? pls correct it.
}
}
2 楼
laogao598599
2016-02-19
匿名内部类不一定是局部内部类,还可以作为成员变量
1 楼
清水幽香
2011-06-27
在Java中,方法的局部变量位于栈上,对象位于堆上。因为局部变量的范围被限制在该方法内,当一个方法结束时,栈结构被删除,该变量消失。但是,定义在这个类中的内部类对象仍然存活在堆上,所以内部类对象不能使用局部变量。除非这些局部变量被标识为最终的。
final int i=8;
就可以
感觉这样解释更容易理解,从栈和堆角度来说明
final int i=8;
就可以
感觉这样解释更容易理解,从栈和堆角度来说明
发表评论
-
java.lang.ClassCircularityError
2015-07-27 00:39 1267类循环依赖错误。在初始化一个类时,若检测到类之间循环依赖则抛出 ... -
随记内容
2015-06-09 00:10 0public static CloseableHttpClie ... -
httpclient3.1使用小结
2017-08-18 00:08 638Jakarta的httpclient3.1是 ... -
POI
2014-05-21 16:08 509Apache POI是一个开源的Java读写Excel、WOR ... -
servlet与js 跳转
2014-05-15 17:19 805servlet跳转 1.request.getRequestD ... -
jdk各个版本下载地址
2014-05-12 17:40 891http://www.oracle.com/technetwo ... -
Jackson库
2014-04-20 13:17 903概述 Jackson库(http://jac ... -
生成Webservice客户端的4种方法
2014-04-09 15:39 10451.使用Axis生产客户端代码(另存为bat文件) 之前很多次 ... -
map遍历
2013-12-14 23:41 476Map<String, String> map = ... -
POI处理Excel
2013-12-14 22:44 8441.读取2003 InputStream stream ... -
常用数据库 JDBC URL 格式
2013-12-04 18:35 1086Microsoft SQL Server Microsoft ... -
公共查询接口
2013-12-04 15:19 1134http://www.webmasterhome.cn/ ht ... -
天气预报接口
2013-12-04 11:18 1356天气预报接口(中国天气网) 中国天气weather.com ... -
Java 判断文件夹、文件是否存在
2013-12-04 10:00 50581、判断文件是否存在,不存在创建文件 File file=ne ... -
Jakarta Commons Pool
2013-12-03 14:55 802使用Jakarta Commons Pool可以根据需要快速的 ... -
多线程读写资料利器-ReentrantReadWriteLock
2013-12-03 14:49 1258多线程读写文件利器-ReentrantReadWriteLoc ... -
保持顺序的 Java Properties 类
2013-12-02 16:21 857public class OrderedProperties ... -
java中用poi处理Excel表格之读取
2013-11-18 16:24 1986http://tool.oschina.net/apidocs ... -
科大讯飞语音识别服务
2013-09-03 17:06 13411.官方地址:http://open.voicecloud.c ... -
ibatis数据库字段带下划线(_)映射与java类不一致
2013-08-24 17:00 5264ibatis数据库字段带下划线(_)与java pojo对象映 ...
相关推荐
•对一个final变量来说,不管它是类变量、实例变量,还是局部变量,只要该变量满足3个条件,这个final变量就 不再是一个变量,而是相当于一个直接量。 –使用final修饰符修饰; –在定义该final变量时指定...
240 12.2.1 创建局部内部类 240 12.2.2 在局部内部类中访问外部类成员变量 240 12.2.3 在局部内部类中访问外部类的局部变量 241 12.2.4 静态方法中的局部内部类 243 12.3 静态内部类 244 12.3.1 创建静态内部类 244 ...
7.8.1 发现问题:当实例变量和局部变量重名 177 7.8.2 经常深藏不露的this关键字 178 7.8.3 在方法中调用方法 179 7.9 构造方法(Constructor) 181 7.9.1 构造(Constructor)方法初探 181 7.9.2 如何使用构造...
7.8.1 发现问题:当实例变量和局部变量重名 177 7.8.2 经常深藏不露的this关键字 178 7.8.3 在方法中调用方法 179 7.9 构造方法(Constructor) 181 7.9.1 构造(Constructor)方法初探 181 7.9.2 如何使用构造...
final 用于声明属性,方法和类,分别表示属性不可变,方法不可覆盖,类不可继承。 finally是异常处理语句结构的一部分,表示总是执行。 finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的...
学生提问:既然内部类是外部类的成员,是否可以为外部类定义子类,在子类中再定义一个内部类来重写其父类中的内部类? 211 6.7.4 局部内部类 211 6.7.5 匿名内部类 212 6.7.6 闭包(Closure)和回调 215 6.8 ...
15.2.2 局部变量与局部内部类 329 15.2.3 静态方法中的局部内部类 331 15.3 静态内部类 332 15.3.1 语法规则 332 15.3.2 创建静态内部类的对象 332 15.3.3 静态/非静态内部类的区别 333 15.4 匿名内部...
402.10 final/override控制 442.11 模板函数的默认模板参数 482.12 外部模板 502.12.1 为什么需要外部模板 502.12.2 显式的实例化与外部模板的声明 522.13 局部和匿名类型作模板实参 542.14 本章小结 55第3章 通用为...
ORACLE_SID=oral10g\ --变局部变量 export ORACLE_SID --变全局变量 unset ORACLE_SID --卸载环境变量 ORACLE_HOME=... --安装路径;直接用一句语句也可以,如下 export ORACLE_HOME=/oracledata/.../bin: ...