java 内部类(整理)
2016-03-14 17:28
288 查看
内部类四种形式:
1.静态内部类(加了static修饰的内部类):也叫嵌套内部类
1.最简单的内部类的形式
2.可以使用所有的private,public,protected,default修饰。
3.可以直接访问外部类的所有静态成员。包含私有的 成员
4.在内部类中如果定义和外部类同名的变量,在内部类中的访问采用就近原则。如果想要访问外部类同名的变量可以使用OuterClass.staticMember;
5.在外部类的外部构建内部类的实例:
new OuterClass.InnerClass();
6.在外部类所有的方法中(包含static和非static)构建内部类的实例:
new InnerClass();
(内部类是非静态的,除了要依靠外部类实例,还要依赖内部类实例,静态常量是不要类实例的,在编译时就分配好了内存,且final类型可以离开实例存活一段时间。所以非静态内部类只能有静态的final型常量。)
内部类总结
内部类不是很好理解,但说白了其实也就是一个类中还包含着另外一个类
如同一个人是由大脑、肢体、器官等身体结果组成,而内部类相当于其中的某个器官之一,例如心脏:它也有自己的属性和行为(血液、跳动)
显然,此处不能单方面用属性或者方法表示一个心脏,而需要一个类
而心脏又在人体当中,正如同是内部类在外部内当中
运行结果:12
从上面的例子不难看出,内部类其实严重破坏了良好的代码结构,但为什么还要使用内部类呢?
因为内部类可以随意使用外部类的成员变量(包括私有)而不用生成外部类的对象,这也是内部类的唯一优点
如同心脏可以直接访问身体的血液,而不是通过医生来抽血。
程序编译过后会产生两个.class文件,分别是Out.class和Out$In.class
其中$代表了上面程序中Out.In中的那个 .
Out.In in = new Out().new In()可以用来生成内部类的对象,这种方法存在两个小知识点需要注意
1.开头的Out是为了标明需要生成的内部类对象在哪个外部类当中
2.必须先有外部类的对象才能生成内部类的对象,因为内部类的作用就是为了访问外部类中的成员变量
实例2:内部类中的变量访问形式
运行结果:
局部变量:14
内部类变量:13
外部类变量:12
从实例1中可以发现,内部类在没有同名成员变量和局部变量的情况下,内部类会直接访问外部类的成员变量,而无需指定Out.this.属性名
否则,内部类中的局部变量会覆盖外部类的成员变量
而访问内部类本身的成员变量可用this.属性名,访问外部类的成员变量需要使用Out.this.属性名
实例3:静态内部类
运行结果:12
可以看到,如果用static 将内部内静态化,那么内部类就只能访问外部类的静态成员变量,具有局限性
其次,因为内部类被静态化,因此Out.In可以当做一个整体看,可以直接new 出内部类的对象(通过类名访问static,生不生成外部类对象都没关系)。
实例4:私有内部类
运行结果:12
如果一个内部类只希望被外部类中的方法操作,那么可以使用private声明内部类
上面的代码中,我们必须在Out类里面生成In类的对象进行操作,而无法再使用Out.In in = new Out().new In() 生成内部类的对象
也就是说,此时的内部类只有外部类可控制
如同是,我的心脏只能由我的身体控制,其他人无法直接访问它
实例5:方法内部类
运行结果:
3
12
在上面的代码中,我们将内部类移到了外部类的方法中,然后在外部类的方法中再生成一个内部类对象去调用内部类方法
如果此时我们需要往外部类的方法中传入参数,那么外部类的方法形参必须使用final定义
至于final在这里并没有特殊含义,只是一种表示形式而已
1.静态内部类(加了static修饰的内部类):也叫嵌套内部类
1.最简单的内部类的形式
2.可以使用所有的private,public,protected,default修饰。
3.可以直接访问外部类的所有静态成员。包含私有的 成员
4.在内部类中如果定义和外部类同名的变量,在内部类中的访问采用就近原则。如果想要访问外部类同名的变量可以使用OuterClass.staticMember;
5.在外部类的外部构建内部类的实例:
new OuterClass.InnerClass();
6.在外部类所有的方法中(包含static和非static)构建内部类的实例:
new InnerClass();
2.实例内部类(成员内部类): 1.定义在类体部,并且没有static修饰的类 2.可以访问外部类所有的成员,包含私有的成员。 3.实例内部类对象的存在依赖外部类实例对象的存在才能存在。 4.实例内部类中不允许声明static的成员。除非是static final常量。
(内部类是非静态的,除了要依靠外部类实例,还要依赖内部类实例,静态常量是不要类实例的,在编译时就分配好了内存,且final类型可以离开实例存活一段时间。所以非静态内部类只能有静态的final型常量。)
5.可以使用所有的private,public,protected,default修饰。 6.在内部类的实例方法中访问和外部同名的实例变量。采用就近原则。如果想访问外部类同名的实例变量可以使用 OuterClass.this.varName; 7.在外部类的外部构建内部类的实例,使用 new OuterClass().new Inner(); 8.在外部类的实例方法中构建实例内部类的对象: this.new InnerClass(); 在外部类的static方法中构建实例内部类的对象:
3.局部内部类: 1.定义在方法体部。或者比方法提更小的代码块的内部。 2.内部类中最少使用的一种格式。 3.类似于局部变量的声明,不能使用private,protected,public,static修饰。 4.局部内部类能够访问的外部类的成员,根据局部内部类所在方法体决定。如果时在static方法体中 ,可以访问外部类static的成员。如果时在非static方法体中,可以访问外部类所有的成员。 5.如果局部内部类要访问所在方法提的局部变量,那么只能访问final修饰的变量。 4.匿名内部类: 1.没有名字的局部内部类 2.没有class,extends,implements关键字 3.没有构造器 4.匿名内部类隐式的继承了某一个父类或者实现了某一个接口
内部类总结
内部类不是很好理解,但说白了其实也就是一个类中还包含着另外一个类
如同一个人是由大脑、肢体、器官等身体结果组成,而内部类相当于其中的某个器官之一,例如心脏:它也有自己的属性和行为(血液、跳动)
显然,此处不能单方面用属性或者方法表示一个心脏,而需要一个类
而心脏又在人体当中,正如同是内部类在外部内当中
实例1:内部类的基本结构 //外部类 class Out { private int age = 12; //内部类 class In { public void print() { System.out.println(age); } } } public class Demo { public static void main(String[] args) { Out.In in = new Out().new In(); in.print(); //或者采用下种方式访问 /* Out out = new Out(); Out.In in = out.new In(); in.print(); */ } }
运行结果:12
从上面的例子不难看出,内部类其实严重破坏了良好的代码结构,但为什么还要使用内部类呢?
因为内部类可以随意使用外部类的成员变量(包括私有)而不用生成外部类的对象,这也是内部类的唯一优点
如同心脏可以直接访问身体的血液,而不是通过医生来抽血。
程序编译过后会产生两个.class文件,分别是Out.class和Out$In.class
其中$代表了上面程序中Out.In中的那个 .
Out.In in = new Out().new In()可以用来生成内部类的对象,这种方法存在两个小知识点需要注意
1.开头的Out是为了标明需要生成的内部类对象在哪个外部类当中
2.必须先有外部类的对象才能生成内部类的对象,因为内部类的作用就是为了访问外部类中的成员变量
实例2:内部类中的变量访问形式
class Out { private int age = 12; class In { private int age = 13; public void print() { int age = 14; System.out.println("局部变量:" + age); System.out.println("内部类变量:" + this.age); System.out.println("外部类变量:" + Out.this.age); } } } public class Demo { public static void main(String[] args) { Out.In in = new Out().new In(); in.print(); } }
运行结果:
局部变量:14
内部类变量:13
外部类变量:12
从实例1中可以发现,内部类在没有同名成员变量和局部变量的情况下,内部类会直接访问外部类的成员变量,而无需指定Out.this.属性名
否则,内部类中的局部变量会覆盖外部类的成员变量
而访问内部类本身的成员变量可用this.属性名,访问外部类的成员变量需要使用Out.this.属性名
实例3:静态内部类
class Out { private static int age = 12; static class In { public void print() { System.out.println(age); } } } public class Demo { public static void main(String[] args) { Out.In in = new Out.In(); in.print(); } }
运行结果:12
可以看到,如果用static 将内部内静态化,那么内部类就只能访问外部类的静态成员变量,具有局限性
其次,因为内部类被静态化,因此Out.In可以当做一个整体看,可以直接new 出内部类的对象(通过类名访问static,生不生成外部类对象都没关系)。
实例4:私有内部类
class Out { private int age = 12; private class In { public void print() { System.out.println(age); } } public void outPrint() { new In().print(); } } public class Demo { public static void main(String[] args) { //此方法无效 /* Out.In in = new Out().new In(); in.print(); */ Out out = new Out(); out.outPrint(); } }
运行结果:12
如果一个内部类只希望被外部类中的方法操作,那么可以使用private声明内部类
上面的代码中,我们必须在Out类里面生成In类的对象进行操作,而无法再使用Out.In in = new Out().new In() 生成内部类的对象
也就是说,此时的内部类只有外部类可控制
如同是,我的心脏只能由我的身体控制,其他人无法直接访问它
实例5:方法内部类
class Out { private int age = 12; public void Print(final int x) { class In { public void inPrint() { System.out.println(x); System.out.println(age); } } new In().inPrint(); } } public class Demo { public static void main(String[] args) { Out out = new Out(); out.Print(3); } }
运行结果:
3
12
在上面的代码中,我们将内部类移到了外部类的方法中,然后在外部类的方法中再生成一个内部类对象去调用内部类方法
如果此时我们需要往外部类的方法中传入参数,那么外部类的方法形参必须使用final定义
至于final在这里并没有特殊含义,只是一种表示形式而已
相关文章推荐
- 转:java 重定向和转发的区别
- Java虚拟机工作原理详解 (一)
- Java多线程(五)、多线程其他知识简要介绍
- 礼拜一log~tag lib标签 & java ArrayList去重方式 & Mac本下对类声明的查询方法
- Java多线程(四)、线程池
- Java多线程(三)、线程同步
- Java多线程(二)、线程的生命周期和状态控制
- SpringBoot附录
- java 内部类
- poi利用反射机制封装导出方法
- Java多线程(一)、多线程的基本概念和使用
- 一个极简单的Retrofit和RxJava的小例子
- java实现邮件发送(带附件)
- 1、Java服务端JPush的应用【带源代码】
- java 实现图的宽度优先遍历
- Java RESTful Web Service相关概念
- Reverse Linked List II | Java最短代码实现
- MyEclipse连接MySQL数据库详细步骤
- spring-jar包详解整理(大合集)
- SpringMvc