(一) 概述java
把类定义在另外一个类的内部,该类就被称为内部类。安全
举例:把类Inner定义在类Outer中,类Inner就被称为内部类。学习
class Outer { class Inner { } }
A:能够直接访问外部类的成员,包括私有测试
B:外部类要想访问内部类成员,必须建立对象this
A:成员内部类spa
B:局部内部类code
C:静态内部类对象
D:匿名内部类继承
成员内部类——就是位于外部类成员位置的类
特色:可使用外部类中全部的成员变量和成员方法(包括private的)
class Outer { private int age = 20; //成员位置 class Inner { public void show() { System.out.println(age); } } } class Test { public static void main(String[] ages) { //成员内部类是非静态的演示 Outer.Inner oi = new Outer().new Inner(); oi.show(); } }
//成员内部类不是静态的: 外部类名.内部类名 对象名 = new 外部类名.new 内部类名(); //成员内部类是静态的: 外部类名.内部类名 对象名 = new 外部类名.内部类名();
A:private接口
若是咱们的内部类不想轻易被任何人访问,能够选择使用private修饰内部类,这样咱们就没法经过建立对象的方法来访问,想要访问只须要在外部类中定义一个public修饰的方法,间接调用。这样作的好处就是,咱们能够在这个public方法中增长一些判断语句,起到数据安全的做用。
class Outer { private class Inner { public void show() { System.out.println(“密码备份文件”); } } public void method() { if(你是管理员){ Inner i = new Inner(); i.show(); }else { System.out.println(“你没有权限访问”); } } }
下面咱们给出一个更加规范的写法
class Outer { private class Inner { public void show() { System.out.println(“密码备份文件”); } } //使用getXxx()获取成员内部类,能够增长校验语句(文中省略) public Inner getInner() { return new Inner(); } public static void main(String[] args) { Outer outer = new Outer(); Outer.Inner inner = outer.getInner(); inner.show(); } }
这种被 static 所修饰的内部类,按位置分,属于成员内部类,但也能够称做静态内部类,也常叫作嵌套内部类。具体内容咱们在下面详细讲解。
D:成员内部类经典题(填空)
请在三个println 后括号中填空使得输出25,20,18
class Outer { public int age = 18; class Inner { public int age = 20; public viod showAge() { int age = 25; System.out.println(age);//空1 System.out.println(this.age);//空2 System.out.println(Outer.this.age);//空3 } } }
局部内部类——就是定义在一个方法或者一个做用域里面的类
特色:主要是做用域发生了变化,只能在自身所在方法和属性中被使用
A 格式:
class Outer { public void method(){ class Inner { } } }
B:访问时:
//在局部位置,能够建立内部类对象,经过对象调用和内部类方法 class Outer { private int age = 20; public void method() { final int age2 = 30; class Inner { public void show() { System.out.println(age); //从内部类中访问方法内变量age2,须要将变量声明为最终类型。 System.out.println(age2); } } Inner i = new Inner(); i.show(); } }
C: 为何局部内部类访问局部变量必须加final修饰呢?
由于局部变量是随着方法的调用而调用,使用完毕就消失,而堆内存的数据并不会当即消失。
因此,堆内存仍是用该变量,而该变量已经没有了。为了让该值还存在,就加final修饰。
缘由是,当咱们使用final修饰变量后,堆内存直接存储的是值,而不是变量名。
(即上例 age2 的位置存储着常量30 而不是 age2 这个变量名)
咱们所知道static是不能用来修饰类的,可是成员内部类能够看作外部类中的一个成员,因此能够用static修饰,这种用static修饰的内部类咱们称做静态内部类,也称做嵌套内部类.
特色:不能使用外部类的非static成员变量和成员方法
解释:非静态内部类编译后会默认的保存一个指向外部类的引用,而静态类却没有。
简单理解:
即便没有外部类对象,也能够建立静态内部类对象,而外部类的非static成员必须依赖于对象的调用,静态成员则能够直接使用类调用,没必要依赖于外部类的对象,因此静态内部类只能访问静态的外部属性和方法。
class Outter { int age = 10; static age2 = 20; public Outter() { } static class Inner { public method() { System.out.println(age);//错误 System.out.println(age2);//正确 } } } public class Test { public static void main(String[] args) { Outter.Inner inner = new Outter.Inner(); inner.method(); } }
一个没有名字的类,是内部类的简化写法
A 格式:
new 类名或者接口名() { 重写方法(); }
本质:实际上是继承该类或者实现接口的子类匿名对象
这也就是下例中,能够直接使用 new Inner() {}.show(); 的缘由 == 子类对象.show();
interface Inner { public abstract void show(); } class Outer { public void method(){ new Inner() { public void show() { System.out.println("HelloWorld"); } }.show(); } } class Test { public static void main(String[] args) { Outer o = new Outer(); o.method(); } }
若是匿名内部类中有多个方法又该如何调用呢?
Inter i = new Inner() { //多态,由于new Inner(){}表明的是接口的子类对象 public void show() { System.out.println("HelloWorld"); } };
B:匿名内部类在开发中的使用
咱们在开发的时候,会看到抽象类,或者接口做为参数。
而这个时候,实际须要的是一个子类对象。
若是该方法仅仅调用一次,咱们就可使用匿名内部类的格式简化。
-----------------------------------------------------------------------------
2019-8-17更新补充
做为一个类的编写者,咱们很显然须要对这个类的使用访问者的访问权限作出必定的限制,咱们须要将一些咱们不肯意让别人看到的操做隐藏起来,
若是咱们的内部类不想轻易被任何人访问,能够选择使用private修饰内部类,这样咱们就没法经过建立对象的方法来访问,想要访问只须要在外部类中定义一个public修饰的方法,间接调用。
public interface Demo { void show(); } class Outer { private class test implements Demo { public void show() { System.out.println("密码备份文件"); } } public Demo getInner() { return new test(); } }
咱们来看其测试
public static void main(String[] args) { Outer outer = new Outer(); Demo d = outer.getInner(); i.show(); } //运行结果 密码备份文件
这样作的好处之一就是,咱们能够在这个public方法中增长一些判断语句,起到数据安全的做用。
其次呢,咱们的对外可见的只是getInner()这个方法,它返回了一个Demo接口的一个实例,而咱们真正的内部类的名称就被隐藏起来了
咱们以前的学习知道,java是不能够实现多继承的,一次只能继承一个类,咱们学习接口的时候,有提到能够用接口来实现多继承的效果,即一个接口有多个实现,可是这里也是有一点弊端的,那就是,一旦实现一个接口就必须实现里面的全部方法,有时候就会出现一些累赘,可是使用内部类能够很好的解决这些问题
public class Demo1 { public String name() { return "BWH_Steven"; } } public class Demo2 { public String email() { return "xxx.@163.com"; } } public class MyDemo { private class test1 extends Demo1 { public String name() { return super.name(); } } private class test2 extends Demo2 { public String email() { return super.email(); } } public String name() { return new test1().name(); } public String email() { return new test2().email(); } public static void main(String args[]) { MyDemo md = new MyDemo(); System.out.println("个人姓名:" + md.name()); System.out.println("个人邮箱:" + md.email()); } }
咱们编写了两个待继承的类Demo1和Demo2,在MyDemo类中书写了两个内部类,test1和test2 二者分别继承了Demo1和Demo2类,这样MyDemo中就间接的实现了多继承
咱们用通俗讲解就是说在Java中,一般就是编写一个接口,而后你来实现这个接口,而后把这个接口的一个对象做以参数的形式传到另外一个程序方法中, 而后经过接口调用你的方法,匿名内部类就能够很好的展示了这一种回调功能
public interface Demo { void demoMethod(); } public class MyDemo{ public test(Demo demo){ System.out.println("test method"); } public static void main(String[] args) { MyDemo md = new MyDemo(); //这里咱们使用匿名内部类的方式将接口对象做为参数传递到test方法中去了 md.test(new Demo){ public void demoMethod(){ System.out.println("具体实现接口") } } } }
编写一个接口 Demo
public interface Demo { void test(); }
编写一个类 MyDemo
public class MyDemo { public void test() { System.out.println("父类的test方法"); } }
编写一个测试类
public class DemoTest extends MyDemo implements Demo { public void test() { } }
这样的话我就有点懵了,这样如何区分这个方法是接口的仍是继承的,因此咱们使用内部类解决这个问题
public class DemoTest extends MyDemo { private class inner implements Demo { public void test() { System.out.println("接口的test方法"); } } public Demo getIn() { return new inner(); } public static void main(String[] args) { //调用接口而来的test()方法 DemoTest dt = new DemoTest(); Demo d = dt.getIn(); d.test(); //调用继承而来的test()方法 dt.test(); } } //运行结果 接口的test方法 父类的test方法