将一个类的定义放到另一个类的定义内部,这就是内部类。
平凡内部类
平凡内部类就是最普通的内部类,没有static修饰类名,就像这样的类
public class Outer{
class Inner {}
}
普通内部类能访问其外部类的所有成员,拥有所有成员的访问权,而不需要任何特殊条件。这是如何做到的呢?弄白这一点很重要,普通内部类的很多特性都是由此而来。来看一段代码
class Outer {
//内部类
class Inner {
public void f() {
//查看此内部类的所有属性
Field[] fields = Inner.class.getDeclaredFields();
for(Field f : fields) {
System.out.println(f.getName());
}
}
}
}
Inner.f()方法是输出Inner类的所有属性名称,但是在Inner类中没有定义任何属性,那么执行f()方法的结果是什么呢?结果会打印:
this$0
也就是说在Inner类里有一个this$0的属性,通过名字就可以看出这是Java自动给Inner类添加的属性,使用javap反编译Inner.class,果然发现了一个final的this$0域
这就是问题的答案,内部类就是通过this$0来访问其外部类的成员,this$0就是用来保存创建该内部类对象的外部类对象。也就是说,当某个外部类对象创建了一个内部类对象时,内部类对象会拥有那个外部类对象的引用,就是this$0,在访问外部类的成员时,就是使用this$0实现的。所有的这一切都由编译器来处理。由此我们发现,内部类的对象必须依附某个外部类对象,于是平凡内部类的几个特性均可由此而得到解释:
(1)在内部类里面需要外部类的引用时,可以使用.thi语法,.this就是this$0属性,也就是创建该对象的外部类对象,如
public Outer outer {
return Outer.this;
}
(2)在外部类的非静态方法中可以直接创建内部类,但是在外部类的静态方法中则不行,如
//在外部类的非静态方法中可以直接创建
//因为必须要通过外围类的对象才可以调用非静态方法,内部类对象正是由这个外围类对象创建
public Inner getInner() {
Inner in = new Inner();
return in;
}
//错误,因为不需要外围类对象调用静态方法,没有外围类对象就不能创建内部类
public static Inner sgetInner() {
return new Inner();
}
只有外部类的对象才能创建平凡内部类的对象,这句话我们一定要记住。
(3)在外部类的静态方法或者其他类里面,需要使用外围类对象.new创建内部类对象,如
//需要外部类的对象调用.new语法来创建内部类对象,在其他类中也是如此
public static Inner sgetInner(Outer outer) {
Inner in = outer.new Inner();
return in;
}
(4)在平凡内部类不能有static修饰的东西,因为内部类需要依赖外部类对象。
static内部类(嵌套类)
使用static修饰的内部类叫做嵌套类,嵌套类的对象与外部类的对象之间没有联系,它的内部也不会保存对外部类对象的引用。嵌套类和普通的类没有什么分别,它只是置于外部类的命名空间。但是它到底是在外部类的内部,所以还是可以访问外部类的所有成员的,只不过需要一个显式的外部类对象
class Outer {
private String s;
//拥有外部类的私有成员的访问权限
static class SIn {
public void f(Outer outer) {
String ss = outer.s;
}
}
}
内部类可以作为接口的一部分,放在接口的内部类自动都是public static的,它可以作为所有实现该接口的公共代码,甚至可以在内部类实现外围接口
interface A{
//自动是public static
class B implements A{
公共代码
}
}
普通的Java类不可以使用private和pretected修饰,但是内部类可以。在其他类里使用某个类的内部类时,必须显式的使用外部类.内部类这种形式:Outer.Inner。
局部内部类
内部类不仅可以在类里面,还可以定义在方法和任意作用域内。这样做的理由:
(1)方法可以返回某接口,可以在方法内部定义一个类,该类实现了要返回的接口,创建该类的实例,并返回,这样对就对外界完全屏蔽了。
(2)为了解决一个复杂的问题,想创建一个类来辅助你的解决方案,但是又不希望这个类是公共可用的。
这些内部类的作用域跟普通变量一样,都是属于块作用域,出了作用域,它就没有意义了。
{
class Inner {}
//可以在这里创建
Inner in = new Inner();
}
//这里已经出了作用域了
Inner in = new Inner();//错误
关于匿名内部类
a.匿名内部类被自动转型为父类的引用,所以里面创建的方法如果超出了父类的接口,那是无法访问的,包括属性。匿名内部类可以看成是父类的“无名”子类,所以在创建匿名内部类的对象时,首先还是要创建父类的对象,根据父类构造器中参数的不同,来调用不同的父类构造器,当然了,父类的成员定义处的初始化,语句块的执行也是肯定会有的。
//父类
class Base {
{
System.out.println("父类的语句块");
}
Base(int x) {
System.out.println("调用执行父类的构造器:" + x);
}
public Base getBase(int x) {
//匿名内部类,自动调用父类的构造器
return new Base(x) {
//匿名内部类的语句块
{
System.out.println("匿名内部类的语句块");
}
//匿名内部类实现的自己的方法,根本没有任何用处
public void anoyMethod() {
System.out.println("匿名内部类实现的方法");
}
};
}
如果调用Base的getBase(1),则执行结果是:
父类的语句块
调用执行父类的构造器:1
匿名内部类的语句块
超出父类接口的anoyMethod()没有任何机会得到执行。
b.匿名内部类是没有名字呢,怎么创建它的构造器呢?
通过上面的示例,我们可以看到,通过非静态语句块可以达到这个效果,非静态语句块就是匿名内部类的构造器,只是语句块不能重载。
c.传递到匿名内部类的参数和局部变量,必须得是final的;回顾一下:非静态内部类,必须要保存一个外部类的引用。匿名内部类也是内部类,也可以无条件的访问外部类的成员变量,也可以使用.this语句,所以在匿名内部类中使用外部类的成员属性是没有任何限制的,外部类的成员属性不需要是final的。
class Outer {
private String name = "cxy";
//注意label和age都是final,而name则无所谓
public Base getBase(final String label) {
final int age = 25;
return new Base() {
{
System.out.println("label: " + label + ",name: " + name + ",age: " + age);
//使用.this语法
System.out.println(Outer.this.getClass().getName());
}
} ;
}
}
d.匿名内部类和工厂方法结合,会产生非常优美的代码,不需要为每个子类额外的添加一个工厂类,而是在子类内部使用一个匿名内部类代替
//业务接口
interface Service {
定义业务方法
}
//工厂接口,返回逻辑类的实例
interface ServiceFactory {
Service getService();
}
//看看使用了匿名内部类的工厂方法
class ServiceImp implements Service {
实现逻辑
//使用内部类完全就不需要对应的工厂类了,这让类的接口瞬间清晰了很多
public static ServiceFactor factory = new ServiceFactory() {
public Service getService() {
return new ServiceImp();
}
};
}
使用匿名内部类确实可以让工厂方法变得相当优雅,可以减少一半的类
内部类的命名规则
内部类也会生成.class文件,即使是匿名内部类。内部类有自己的一套命名规则,有时候在看反编译出来的代码时或许会用到。最基本的是在外部类的后面加上$,然后紧接内部类的名称。一例胜千言
public class NamedInnderClass {
//静态内部类:NamedInnderClass$StaticInner
static class StaticInner {
//嵌套在内部类中:NamedInnderClass$StaticInner$StaticInnerInner
static class StaticInnerInner {}
}
//普通内部类:NamedInnderClass$InnerClass
class InnerClass { }
//方法内部的类
public void f() {
//NamedInnderClass$1MethodInnder,注意加上了一个数字,数字是递增的
class MethodInner { }
//NamedInnderClass$1,匿名内部类,直接加数字,数字递增
new NamedInnderClass() { };
}
}
内部类还是有些复杂,基本的使用方法上面都做了介绍,但是为什么要使用内部类呢?使用内部类最吸引人的原因是:每个内部类都能独立地继承一个实现(类或者接口),不管外部类是否继承了某个实现,对内部类都不会影响。从这个角度看,内部类使得多重继承的解决方案更加完整。
class A{}
abstract class B{}
class C extends A {
//使用内部类实现"多重继承"
B makeB() {
return new B() {};
}
}
转载请注明出处:喻红叶《Java中的内部类》
分享到:
相关推荐
Java 内部类 实例化 在Outer类的静态方法中实例化内部类 在同一个包的其它类中实例化Outer类中的内部类
在java语言中,有一种类叫做内部类(inner class),也称为嵌入类(nested class),它是定义在其他类的内部。
根据不同的定义位置和作用域,Java内部类可以分为四类:成员内部类、局部内部类、匿名内部类和静态内部类。下面将详细介绍这些内部类的定义和使用方法。 成员内部类 成员内部类是最常见的内部类形式,它定义在外部...
成员内部类是指在另一个类中定义但没有使用 `static` 关键字的类。它的定义格式如下: ```java public class 外部类 { ... class 成员内部类 { ... } } ``` **特点** - **访问权限**:成员内部类可以访问外部...
Java中的内部类与外部类是Java面向对象编程中一个重要的特性,它允许我们在一个类的内部定义另一个类。这种设计模式提供了更高级别的封装,并且能够实现一些特殊的功能,如匿名内部类、事件处理等。以下是对内部类与...
JAVA从JDK1.1开始引入了内部类,可以参见代码,感觉好处就是设计类的时候可以偷懒,呵呵。主要是可以引用类的内部其他元素,差不多是把这个内部类当成原类的元素。还有可以隐藏类的一些设计细节,好处还是很多的。
Java匿名内部类是Java语言中一个独特且强大的特性,它允许我们在不需要定义一个单独的类的情况下创建类的实例。这在处理回调、事件监听、单例模式以及其他需要短时间内定义一个简单类的情况时非常有用。本篇文章将...
Java 中内部类使用方法实战案例分析 Java 中内部类的使用方法是一个非常重要的知识点,它们可以帮助开发者更好地组织代码、提高代码的可读性和维护性。在本文中,我们将通过实战案例来分析 Java 中内部类的使用方法...
14.java局部内部类(方法中类).zip14.java局部内部类(方法中类).zip14.java局部内部类(方法中类).zip14.java局部内部类(方法中类).zip14.java局部内部类(方法中类).zip14.java局部内部类(方法中类).zip14...
### Java内部类详解 #### 一、内部类的分类与概念 Java的内部类机制是其强大特性之一,它允许类作为另一个类的成员存在,从而增强了代码的封装性和复用性。根据定义和作用域的不同,Java内部类主要分为四类: 1. ...
在内部类中访问外部类的成员可以使用 `Outerclass.this` 来指定。 3. 局部内部类(Local Inner Class) 局部内部类定义在方法或代码块内,其作用域仅限于该方法或代码块。它们通常用于实现特定方法的功能,且只能...
Java 语言支持在一个类中定义另一个类,这样的类称做内部类。内部类和外嵌类之间存在着紧密的关系:内部类可以访问外嵌类的成员变量和方法,而外嵌类也可以使用内部类声明的对象作为其成员。内部类的类体中不能声明...
Java内部类是Java语言中的一种特殊类别,它是指定义在另外一个类内部的类。内部类可以访问外部类的所有成员变量和方法,包括私有的变量和方法。内部类可以分为四种:成员内部类、静态嵌套类、方法内部类和匿名内部...
### Java内部类总结 在Java编程语言中,内部类是一个重要的概念,它允许开发者在一个类的内部定义另一个类。这种特性极大地增强了代码的封装性和复用性,同时也为解决特定问题提供了灵活的方法。本文将围绕Java内部...
5. **静态方法调用**:在静态内部类中可以直接访问外部类的静态方法和变量,但是无法直接访问非静态成员。 6. **非静态方法调用限制**:如果需要访问外部类的非静态方法或变量,则必须通过外部类的对象引用来进行...
16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名内部类.zip16.java匿名...
Java内部类是Java语言中一个独特且强大的特性,它允许在一个类的定义内部定义另一个类。内部类可以访问外部类的所有成员,包括私有成员,这使得内部类在实现某些特定设计模式时非常有用。本篇文章将深入探讨Java内部...
成员内部类的使用在Java编程中具有独特的价值,它能够帮助我们更好地封装代码,提高代码的组织性和可读性。以下是关于Java成员内部类的一些关键知识点: 1. **分类**:成员内部类分为两种,静态内部类(Static ...