java内部类的一些看法
java内部类, 我在看《thinking in java》的时候总感觉模棱两可的, 挣扎了好几天之后, 感觉有一部分的问题想的清楚了, 写一个随笔记录一下, 以备以后修改和查看
什么是内部类?
内部类说白了就是类套类, 在一个类里面写一个类, 内部类分为两种, 一种是普通内部类(non-static nested class), 第二种是静态内部类(static nested class),也叫嵌套类.
普通内部类包含另外两种特殊的内部类, 一种是局部内部类, 另一种是匿名内部类.
内部类有什么作用?
- 最简单的功能 : 一种有效的代码隐藏机制, 将某些功能放置在一起组合形成一个新的类, 而且这个新类可以任意访问类的所有成员. 有效地提高了代码的封装型性, 使写出来的代码更加便于理解
- 实现了类逻辑上的分组: 比如 一个类A仅仅只对类B有用处, A是B的辅助类, 或者说仅希望A对B起作用, 那么在逻辑上A已经属于了B类, 内部类就恰好实现了这个逻辑
- 和interface一起真正实现了多重继承 : 因为一个类允许存在多个内部类, 而内部类可以随意访问类的所有成员, 那么就可以通过在不同的内部类继承不同的类, 实现不同的接口在真正意义上完成多重继承的任务.
- 通过内部类提供闭包的功能, 或者说模拟闭包的功能(这部分暂时没有完全弄明白)
什么是闭包?
《thinking in java》有一段原话感觉很贴切:
闭包(closure)是一个可调用的对象, 这个对象记录着一些信息, 这些信息来自于创建他的作用域.
通过回调, 对象可以携带一些信息, 这些信息允许它在将来某个时刻调用初始的对象.
内部类就是面向对象的闭包, 创建一个内部类, 就会得到一个外部类的引用, 通过这个引用, 我们可以调用外部类对象的所有成员, 当然这些成员就是所谓的“创建他的域的信息”。除此之外,使用内部类完成闭包更加安全,因为返回的是一个引用,而不是一个指针,指针和引用可操作的权限差异不止一点点。
为什么普通内部类中不能包含static方法、static字段和嵌套类?
因为static所修饰的属性、方法都是和类相关,而非类对象相关。static成员的访问可以直接通过classname.staticField进行,而内部类的创建是和外部类相关联的,也就是要使用内部类,必须创建一个相关的外部类以提供引用,如果允许static成员的存在,那么就可以通过Outer.Inner.staticField进行访问,与是否存在外部类无关,所以不允许包含static成员。
但是普通内部类却可以包含static final类型的成员, 因为static final类型的成员是静态常量,在编译期就会将所有用到该成员的地方替换成相对应的子面值常量。所以static final类型的成员只能使用常量表达式进行初始化。
例如:
class SS {
public class Test1
{
public static final String str ="";
public static final Integer d = new Integer(25) ; //报错:The field d cannot be declared static in a non-static inner type, unless initialized with a constant expression
}
}
为什么局部内部类访问的局部变量必须是final类型?
什么是局部内部类?
局部内部类是一种特殊的内部类, 只能被定义在代码段Block中 ,也就是由"{"和"}"围成的作用域中. 比如一个代码块中, 一个方法中,一个for循环中,甚至是一个if(){...}中
局部内部类只能在它定义的代码段中被访问,外界是无法访问到的, 所以在外界定义一个相同名字的类,甚至是interface
定义在静态方法中的局部内部类只能访问外部类的静态成员, 而定义在普通方法中的则没有限制.
public class OutClass
{
private int a = 0 ;
private static int b = 1 ; public static void c1()
{
class InClass
{
{
System.out.println(a); //由于这个内部类是定义在静态方法中的,所以只能访问外部类的静态成员,所以这行会报错
System.out.println(b) ;
}
}
} public void c2()
{
//完全正确
class InClass
{
{
System.out.println(a) ;
System.out.println(b) ;
}
}
} public class InClass
{ }
} class InClass
{ }
当局部内部类访问局部变量或形参时时, 这些变量和参数必须是final类型, 如果你的JDK是java8的话, 形参和变量允许是effectively final类型, 所谓effectively final类型就是虽然声明时没有声明成final类型, 但是他的值(基本数据类型)或引用(引用数据类型)从来没有改变过.(估计是java8做了一些优化吧,太深奥的就不懂了)
局部类使用局部变量和参数之所以必须是final类型的, 和他们的生命周期有关. 局部类是一个类, 他的生命周期和外部类是一样的, 而局部变量和参数则不同, 一旦退出了他们所在的代码段, 就会被销毁, 如果允许一个内部类持有一个这样的变量或者说成员, 是十分不安全的, 所以为了“延续”这些局部变量和参数的生命周期(毕竟不是真正的延续), 所采取的办法就是将他们复制一份, 也就是基本数据类型拷贝一份, 引用数据类型拷贝引用. 那么如何保证内外一致呢?就必须将他们定义成final类型.
//Local classes : http://docs.oracle.com/javase/tutorial/java/javaOO/localclasses.html
//Nested Classes : http://docs.oracle.com/javase/tutorial/java/javaOO/nested.html
//有关于final的解释: http://bbs.csdn.net/topics/300068105 , http://blog.csdn.net/onisland/article/details/5807637
//Java中普通内部类为何不能有static数据和static字段,也不能包含嵌套类 :http://www.dewen.io/q/13793
//thinking in java笔记,如果有不对的地方,还望指正^_^
java内部类的一些看法的更多相关文章
- Java内部类final语义实现
本文描述在java内部类中,经常会引用外部类的变量信息.但是这些变量信息是如何传递给内部类的,在表面上并没有相应的线索.本文从字节码层描述在内部类中是如何实现这些语义的. 本地临时变量 基本类型 fi ...
- Java内部类详解
Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...
- 黑马----JAVA内部类
黑马程序员:Java培训.Android培训.iOS培训..Net培训 黑马程序员--JAVA内部类 一.内部类分为显式内部类和匿名内部类. 二.显式内部类 1.即显式声明的内部类,它有类名. 2.显 ...
- java 内部类 *** 最爱那水货
注: 转载于http://blog.csdn.net/jiangxinyu/article/details/8177326 Java语言允许在类中再定义类,这种在其它类内部定义的类就叫内部类.内部类又 ...
- java内部类和匿名内部类
内部类即是包含在类里面的又一个类. java内部类分为: 成员内部类.静态嵌套类.方法内部类.匿名内部类 . 内部类的共性 (1).内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.clas ...
- Java内部类小程序(成员内部类,静态内部类,匿名内部类)
/** * 测试java内部类(成员内部类,静态内部类,匿名内部类) * 局部内部类不常用,就不写了. * @package :java05 * @author shaobn * @Describe ...
- [转] Java内部类详解
作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置 ...
- java内部类的作用分析
提起Java内部类(Inner Class)可能很多人不太熟悉,实际上类似的概念在C++里也有,那就是嵌套类(Nested Class),关于这两者的区别与联系,在下文中会有对比.内部类从表面上看,就 ...
- 9)Java内部类(Inner Class)
内部类:不可以有静态数据,静态方法或者又一个静态内部类 内部类的优点:隐藏类的细节,内部类可以声明为私有.内部类可以访问外部类的对象(包括private) 静态内部类:可以有静态数据,静 ...
随机推荐
- Ubuntu14.04使用root登陆帐户
http://jingyan.baidu.com/article/27fa73268144f346f8271f83.html 1.输入sudo gedit /usr/share/lightdm/lig ...
- Python中你不知道的特性
内置函数print(*objects, sep=' ', end='\n', file=sys.stdout, flush=False) 本函数是实现对象以字符串表示的方式格式化输出到流文件对象fil ...
- iOS长按控件
前言 网上看到一个button的长按控件效果不错,一个菱形从中间向两边增大,研究了下 原理 上图红色是控件上面放了视图,从上到下分别是view,normalLable,highlightLabel,b ...
- [LOJ#2270][BZOJ4912][SDOI2017]天才黑客
[LOJ#2270][BZOJ4912][SDOI2017]天才黑客 试题描述 SD0062 号选手小 Q 同学为了偷到 SDOI7012 的试题,利用高超的黑客技术潜入了 SDOI 出题组的内联网的 ...
- 雅礼培训 Problem A 【线段树】
题意 维护一段区间,支持求区间最大值,区间且,区间或 \(n,q<=2*10^5\) 题解 我们用线段树维护区间最大值 对于and和or运算, and实质就是强行把一些位改为0 or实质就是强行 ...
- 【霍夫曼树】 poj 1521 Entropy
poj.org/problem?id=1521 注意只有特殊情况:只有一种字母 #include<iostream> #include<cstdio> #include< ...
- 创建mavenweb项目
1.创建maven项目 2.创建maven-web项目 3.将webapp文件夹复制到maven项目下,src路径下
- hdu 1104 数论+bfs
Remainder Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total ...
- LeetCode OJ--Valid Parentheses
http://oj.leetcode.com/problems/valid-parentheses/ 对栈的考察,看括号的使用方式是否合法. class Solution { public: bool ...
- (46)C#注册表及读写
启动注册表:regedit 结构: 注册表一共有7个配置单元用regedit只能看到5个 HKEY_CLASSES_ROOT 包含系统上文件类型的细节(.txt,.doc)等.以及使用那些应用程序可以 ...