Java成员内部类、静态嵌套类、方法内部类、匿名内部类

你好!

内部类解释

Java内部类也是一种独立的类,但是不能用普通的方式进行访问

内部类分类

成员内部类、静态嵌套类、方法内部类、匿名内部类

内部类共性

  • 内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号。
  • 内部类不能用普通的方式访问。内部类是外部类的一个成员,因此内部类可以自由地访问外部类的成员变量,无论是否是private的
  • 内部类声明成静态的,就不能随便访问外部类的成员变量了,此时内部类只能访问外部类的静态成员变量。

内部类实例展示

  • 成员内部类
class TestClass {
    class MemberInnerClass{

    }
}

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

  • 方法内部类 :顾名思义,在方法内部的类
class TestClass {

    public void doSomethingI(){

        class MemberInnerClass{

            public void seeMemberInnerClass(){
                
            }
            
        }
        
    }

}
  1. 方法内部类只能在定义该内部类的方法内实例化,不可以在此方法外对其实例化
  2. 方法内部类对象不能使用该内部类所在方法的非final局部变量。

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

class TestClass {

    public void doSomethingI(){

        final int a = 0;

        class MemberInnerClass{

            public void seeMemberInnerClass(){
                System.out.println(a);
            }

        }

        MemberInnerClass memberInnerClass = new MemberInnerClass();
        memberInnerClass.seeMemberInnerClass();

    }

    public static void main(String[] args){
        TestClass testClass = new TestClass();
        testClass.doSomethingI();
    }
    
}
  • 匿名内部类
    顾名思义,没有名字的内部类。表面上看起来它们似乎有名字,实际那不是它们的名字。
    当程序中使用匿名内部类时,在定义匿名内部类的地方往往直接创建该类的一个对象。匿名内部类的声明格式如下:
 new TestClass(){
            ...//内部类的定义
        }

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

  • 只用到类的一个实例 。
  • 类在定义后马上用到。
  • 类非常小(SUN推荐是在4行代码以下)
  • 给类命名并不会导致你的代码更容易被理解。

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

  • 匿名内部类不能有构造方法。
  • 匿名内部类不能定义任何静态成员、静态方法。
  • 匿名内部类不能是public,protected,private,static。
  • 只能创建匿名内部类的一个实例。
  • 一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类。
  • 因匿名内部类为局部内部类,所以局部内部类的所有限制都对其生效。

A、继承式的匿名内部类

class TestClass {

    public void sayTest(){
        System.out.println("继承式匿名内部类");
    }

    public static void main(String[] args){

        TestClass testClass = new TestClass(){
            public void sayTest(){
                System.out.println("继承式__匿名内部类");
            }
        };

        testClass.sayTest();

    }

}

结果输出了:继承式__匿名内部类, Car引用变量不是引用Car对象,而是Car匿名子类的对象。
B、接口式的匿名内部类。

class TestClass {
    
    public static void main(String[] args){

       Phone phone = new Phone() {
           @Override
           public void callPaone() {
               System.out.println("打电话");
           }
       };

       phone.callPaone();

    }

}

interface Phone{
    void callPaone();
}

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

 class Bar{
        void doStuff(Foo f){
            f.foo();
        }
    }
    interface Foo{
        void foo();
    }
    class Test{
        static void go(){
            Bar b = new Bar();
            b.doStuff(new Foo(){
                public void foo(){
                    System.out.println("foofy");
                }
            });
        }
    }
  • 静态嵌套类
class ParenntClass {
    public static void sayParent(){
        System.out.println("parent");
    }
    static class SonClass {
        public void saySon() {
            System.out.println("son");
            //外部类的静态成员方法
            sayParent();
        }
    }
}

class TestClass {

    public static void main(String[] args) {

        ParenntClass.SonClass sonClass = new ParenntClass.SonClass();

        sonClass.saySon();
    }
}

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

为什么需要内部类

典型的情况是,内部类继承自某个类或实现某个接口,内部类的代码操作创建其他外围类的对象。所以你可以认为内部类提供了某种进入其外围类的窗口。使用内部类最吸引人的原因是:
每个内部类都能独立地继承自一个(接口的)实现,所以无论外围类是否已经继承了某个(接口的)实现,对于内部类都没有影响。如果没有内部类提供的可以继承多个具体的或抽象的类的能力,一些设计与编程问题就很难解决。从这个角度看,内部类使得多重继承的解决方案变得完整。接口解决了部分问题,而内部类有效地实现了“多重继承”。

我是入梦,谢谢你的观看我的博客,如果有什么错误,请随时联系我,QQ:897589417

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值