Java 高级开发必修知识---内部类

Java 内部类分为:

  1)成员内部类

  2)静态嵌套类

  3)方法内部类

  4)匿名内部类

内部类的共性

1、内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号 。

2、内部类不能用普通的方式访问。内部类是外部类的一个成员,因此内部类可以自由地访问外部类的成员变量,无论是否是private的 。

3、内部类声明成静态的,就不能随便的访问外部类的成员变量了,此时内部类只能访问外部类的静态成员变量

成员内部类

成员内部类就是将类放在类中

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        成员内部类
 */

public class Member1 {
    class Inner {
    }
}

编译上述代码会产生两个文件:Member.class和Member$Inner.class。

方法内部类

把类放在方法内

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        方法内部类
 */

public class Member2 {
    public void doSomething() {
        class Inner {
            public void seeOuter() {
            }
        }
    }

}

(1)、方法内部类只能在定义该内部类的方法内实例化,不可以在此方法外对其实例化。

(2)、方法内部类对象不能使用该内部类所在方法的非final局部变量

因为方法的局部变量位于栈上,只存在于该方法的生命期内。当一个方法结束,其栈结构被删除,局部变量成为历史。但是该方法结束之后,在方法内创建的内部类对象可能仍然存在于堆中!例如,如果对它的引用被传递到其他某些代码,并存储在一个成员变量内。正因为不能保证局部变量的存活期和方法内部类对象的一样长,所以内部类对象不能使用它们。

完整栗子:

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        方法内部类 实例
 */

public class Member3 {
    public void doSomething() {
        final int a = 10;
        class Inner {
            public void seeOuter() {
                System.out.println(a);
            }
        }
        Inner in = new Inner();
        in.seeOuter();
    }

    public static void main(String[] args) {
        Member3 out = new Member3();
        out.doSomething();
    }

}

匿名内部类

顾名思义,没有名字的内部类。表面上看起来它们似乎有名字,实际那不是它们的名字。

当程序中使用匿名内部类时,在定义匿名内部类的地方往往直接创建该类的一个对象。匿名内部类的声明格式如下:

new ParentName(){
...// 内部类的定义
}

匿名内部类就是没有名字的内部类。什么情况下需要使用匿名内部类?如果满足下面的一些条件,使用匿名内部类是比较合适的:

·只用到类的一个实例 。

·类在定义后马上用到。

·类非常小(SUN推荐是在4行代码以下)

·给类命名并不会导致你的代码更容易被理解。

在使用匿名内部类时,要记住以下几个原则:

·匿名内部类不能有构造方法

·匿名内部类不能定义任何静态成员、静态方法。

·匿名内部类不能是public,protected,private,static。

·只能创建匿名内部类的一个实例。

·一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类。

·因匿名内部类为局部内部类,所以局部内部类的所有限制都对其生效。

A、继承式的匿名内部类

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *
 *        继承式匿名内部类
 */

public class Member4 {

    public void drive() {
        System.out.println("Driving a car!");
    }

    public static void main(String[] args) {
        Member4 car = new Member4() {
            public void drive() {
                System.out.println("Driving another car!");
            }
        };
        car.drive();
    }

}

结果输出了:Driving another car! Car引用变量不是引用Car对象,而是Car匿名子类的对象。

B、接口式的匿名内部类。

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        接口式匿名内部类
 */

public class Member5 {
    public static void main(String[] args) {
        Vehicle v = new Vehicle() {
            public void drive() {
                System.out.println("Driving a car!");
            }
        };
        v.drive();
    }
}

interface Vehicle {
    public void drive();
}

上面的代码很怪,好像是在实例化一个接口。事实并非如此,接口式的匿名内部类是实现了一个接口的匿名类。而且只能实现一个接口

C、参数式的匿名内部类。

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        参数式匿名内部类
 */

public class Member6 {
    static void go() {
        Bar b = new Bar();
        b.doStuff(new Foo() {
            public void foo() {
                System.out.println("foofy");
            }
        });
    }

}

interface Foo {
    void foo();
}

class Bar {
    void doStuff(Foo f) {
        f.foo();
    }
}

静态嵌套类

静态内部类中可以定义静态或者非静态的成员。

从技术上讲,静态嵌套类不属于内部类。因为内部类与外部类共享一种特殊关系,更确切地说是对实例的共享关系。而静态嵌套类则没有上述关系。它只是位置在另一个类的内部,因此也被称为顶级嵌套类。

静态的含义是该内部类可以像其他静态成员一样,没有外部类对象时,也能够访问它。静态嵌套类仅能访问外部类的静态成员和方法。

package cn.internal;

/**
 * 
 * @author: 房上的猫
 * 
 * @time: 下午9:08:33
 * 
 * @博客地址: https://www.cnblogs.com/lsy131479/
 *
 *        静态嵌套内部类
 */

public class Member7 {
    static class Inner {
    }
}

class Test {
    public static void main(String[] args) {
        Member7.Inner n = new Member7.Inner();
    }

}

在静态方法中定义的内部类也是StaticNested Class,这时候不能在类前面加static关键字,静态方法中的StaticNested Class与普通方法中的内部类的应用方式很相似,它除了可以直接访问外部类中的static的成员变量,还可以访问静态方法中的局部变量,但是,该局部变量前必须加final修饰符。

为什么需要内部类?

典型的情况是,内部类继承自某个类或实现某个接口,内部类的代码操作创建其他外围类的对象。所以你可以认为内部类提供了某种进入其外围类的窗口。使用内部类最吸引人的原因是:

每个内部类都能独立地继承自一个(接口的)实现,所以无论外围类是否已经继承了某个(接口的)实现,对于内部类都没有影响。如果没有内部类提供的可以继承多个具体的或抽象的类的能力,一些设计与编程问题就很难解决。从这个角度看,内部类使得多重继承的解决方案变得完整。接口解决了部分问题,而内部类有效地实现了“多重继承”。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏开发与安全

数据结构:程序加图示分析单链表的插入和删除操作

下图展示了单链表的基本结构: ? head指针是链表的头指针,指向第一个节点,每个节点的next指针域指向下一个节点,最后一个节点的next指针域为NULL,在...

3916
来自专栏DOTNET

C#要点

内容摘要 1 数据类型   1.1表达范围问题   1.2 数组的长度问题   1.3 值类型与引用类型   1.4 匿名类型与隐式类型   1.5硬编码造成的...

2895
来自专栏技术博客

C#委托四(匿名方法)

什么是匿名方法? 匿名方法是C#2.0引入的一个新特性,它允许开发者声明自己的函数代码而无须使用委托函数。 C#为委托提供一种机制,可以为委托定义匿名方...

1172
来自专栏向治洪

Kotlin基础之内联函数

内联函数 使用高阶函数会给运行时带来一些坏处:每个函数都是一个对象,捕获闭包(如:访问函数体内的变量),内存分配(函数对象或Class),虚拟调用引入的运行过载...

2085
来自专栏CodingToDie

Python学习(五):函数

第5 章 函数 Table of Contents 函数调用 数据类型转换 定义函数 空函数 返回多个值 tuple 函数调用 Python内置了很多有用的函数...

4575
来自专栏CodingBlock

正则表达式(一)

  正则表达式是一种强大而灵活的文本处理工具。使用正则表达式,我们能够以编程的方式,构造复杂的文本模式,并对输入的字符串进行搜索。找到匹配这些模式的部分就可以对...

20110
来自专栏向治洪

Swift基础语法

本文来自Swift中文开发组,感谢翻译者的分享。 本文将分几部分对Swift对iOS的语法做讲解。本文为第一节,主要讲解基础语法。 常量和变量 常量和变量把一个...

1946
来自专栏码云1024

C#泛型

2884
来自专栏老马说编程

(91) Lambda表达式 / 计算机程序的思维逻辑

在之前的章节中,我们的讨论基本都是基于Java 7的,从本节开始,我们探讨Java 8的一些特性,主要内容包括: 传递行为代码 - Lambda表达式 函数式...

2048
来自专栏WindCoder

Java中的域与变量

Java中的Field译为“字段”,也译为“域”,Field和成员变量(Member Variable)是相同的。所以域是变量中的一种。

5191

扫码关注云+社区

领取腾讯云代金券