Java-面向对象-内部类(内置类、嵌套类)

什么是内部类

当A类中的内容要被B类直接访问,而A类还需要创建B的对象,访问B的内容时,
这时,可以将B类定义到A类的内部。这样访问更为便捷。

将B称之为内部类(内置类,嵌套类)。

访问方式
内部类可以直接访问外部类中的所有成员,包含私有的。
而外部类要想访问内部类中的成员,必须创建内部类的对象。

当描述事物时,事物的内部还有事物,这个内部的事物还在访问外部事物中的内容。
这时就将这个事物通过内部类来描述。

class Outer//外部类。
{
    private int num = 4;
    class Inner//内部类。
    {
        void show()
        {
            System.out.println("num="+num);
        }
    }

    void method()
    {  
        //因为在Outer中访问可以省略Outer.
        /*Outer.*/Inner in = new /*Outer.*/Inner();
        in.show();
    }
}
class InnerClassDemo
{
    public static void main(String[] args)
    {
        Outer out = new Outer();
        out.method();
    }
}

内部类被访问的方式

情况一:内部类在成员位置上的被访问方式。
成员是可以被指定的修饰符所修饰的。
public:不多见:因为更多的时候,内部类已经被封装到了外部类中,不直接对外提供。

class Outer//外部类。
{
    private int num = 4;
    public class Inner//内部类。
    {
        void show()
        {
            System.out.println("num="+num);
        }
    }
}
class InnerClassDemo
{
    public static void main(String[] args)
    {
        //测试情况一:直接访问Outer中的Inner内部类的非静态成员。
        //创建内部类的对象就哦了。内部类作为成员,应该先有外部类对象,再有内部类对象。
        Outer.Inner in = new Outer().new Inner();
        in.show();

    }
}

private:只能内部用

class Outer//外部类。
{
    private int num = 4;
    private class Inner//内部类。
    {
        void show()
        {
            System.out.println("num="+num);
        }
    }
    void method()
    {  
        //因为在Outer中访问可以省略Outer.
        /*Outer.*/Inner in = new /*Outer.*/Inner();
        in.show();
    }
}
class InnerClassDemo
{
    public static void main(String[] args)
    {
        Outer out = new Outer();
        out.method();
    }
}

static:

class Outer//外部类。
{
    private static int num = 4;
    public class Inner//内部类。
    {
        void show()
        {
            System.out.println("num="+num);
        }
//      static void show1(){}//非静态内部类中不允许定义静态成员。仅允许在非静态内部类中定义 静态常量 static final。
//      如果想要在内部类中定义静态成员,必须内部类也要被静态修饰。
    }

    /*
    内部类被静态修饰后,随着Outer的加载而加载。可以把一个静态内部类理解为就是一个外部类。

    */
    static class Inner2
    {
        void show2()
        {
            System.out.println("Inner2 show2 run..."+num);
        }
        static void staticShow()
        {
            System.out.println("Inner2 staticShow run");
        }
    }
    void method()
    {
        /*Outer.*/Inner in = new /*Outer.*/Inner();
        in.show();
    }
}
class InnerClassDemo
{
    public static void main(String[] args)
    {
        //对静态内部类中的非静态成员进行调用。
        //因为内部类是静态,所以不需要创建Outer的对象。直接创建内部类对象就哦了。
        Outer.Inner2 in = new Outer.Inner2();
        in.show2();
//      如果静态内部类有静态成员,该如何访问呢?既然静态内部类已随外部类加载,而且静态成员随着类的加载而加载,
//      就不需要对象,直接用类名调用即可。
        Outer.Inner2.staticShow();
    }
}

为什么内部类就能直接访问外部类中的成员

那是因为内部类其实持有了外部类的引用 外部类.this (Outer.this)
对于静态内部类不持有 外部类.this 而是直接使用 外部类名(Outer)。

class Outer
{
    int num = 3;
    class Inner
    {
        int num = 4;

        void show()
        {
            int num = 5;
            System.out.println("num="+num);// 3
            System.out.println("num="+this.num);//4
            System.out.println("num="+Outer.this.num);//5
        }
    }
    
    void method()
    {
//      System.out.println(num);
        new Inner().show();
    }
}


class InnerClassDemo2 
{
    public static void main(String[] args) 
    {
        Outer out = new Outer();
        out.method();
    }
}

内部类其实也可以定义在外部类的局部位置上。

内部类定义在局部时,只能访问被final修饰的局部变量。
为啥呢?因为编译生产的class中直接操作那个最终数值了。

为什么不能访问非最终的局部变量呢?
生命周期太短,比如例子中的y

class Outer
{
    int  num = 3;
    void method()
    {
        final int  x = 10;
//      final int x = 5;//局部变量。
        int y = 2;
        class Inner//局部内部类。不能被成员修饰符修饰。
        {
            void show()
            {
//              System.out.println("y="+y);//访问失败。y的生命周期太短了。
                System.out.println("x="+x);
                System.out.println("inner show run........"+num);
            }
        }
        new Inner().show();
    }
}
class InnerClassDemo3 
{
    public static void main(String[] args) 
    {
        Outer out = new Outer();
        out.method();
    }
}

内部类对象对外提供功能的访问方式

看API,发现类名或者接口名称中有 . 说明是内部类,或者内部接口。

内部类的延伸。
内部类是可以继承或者实现外部其他的类或者接口的。

好处:通过内部类的方式对类进行继承重写,或者接口进行实现。
通过公共的方式对其内部类对象进行访问。因为通常内部类很有可能被外部类封装其中。
我们就可以通过父类或者接口的方式访问到内部类对象。

abstract class AbsDemo
{
    abstract void show();
}

class Outer
{
    int num = 3;
    private class Inner extends AbsDemo
    {
        //重写抽象方法show。
        void show()
        {
            System.out.println("num="+num);
        }
    }
    //获取内部类的对象。 
    public AbsDemo getObject()
    {
        return new Inner();
    }

    public void method()
    {
        new Inner().show();
    }
}

class InnerClassDemo4 
{
    public static void main(String[] args) 
    {
        Outer out = new Outer();
//      out.method();
        //如果Inner对外提供,可以如此获取。
//      Outer.Inner in = out.getObject();
//      in.show();
        //如果Inner被private 。可以通过父类型获取。
        AbsDemo a = out.getObject();//多态。
        a.show();
    }
}

匿名内部类

匿名内部类:其实就是一个带有内容的子类对象。
格式:new 父类or接口(){子类的内容}
匿名内部类就是内部类的简化形式。
别忘了:匿名内部类有前提,内部类必须要继承父类或者实现接口。

abstract class AbsDemo
{
    abstract void show();
}


class Outer
{
    int num = 3;
    /*
    class Inner extends AbsDemo
    {
        void show()
        {
            System.out.println("num="+num);
        }
    }
    */
    public void method()
    {
        
//      new Inner().show();
        /*
        不想创建具体的子类型。还想创建AbsDemo的子类对象。
        怎么实现呢?没有子类型干脆,直接使用父类型就哦了。
        可是在该例子中是抽象类,怎么可以new对象呢?
        抽象类之所以不能new对象是因为抽象方法没重写。直接重写不就哦了吗?
        */
        new AbsDemo()//这就是传说中的一个AbsDemo的子类对象。只不过这个对象有点胖!这是一个带着内容的子类对象。
                    //这种写法一个称呼:匿名内部类。
        {
            //重写抽象的show方法。
            void show()
            {
                System.out.println("num===="+num);
            }
        }.show();

    }
}
class InnerClassDemo5 
{
    public static void main(String[] args) 
    {
        Outer out = new Outer();
        out.method();
    }
}

匿名内部类使用

interface Inter
{
    void show1();
    void show2();
}
class Outer
{
    int num = 4;
    //在一个类使用一个接口的对象。可以通过内部类来实现。
    /*
    class Inner implements Inter
    {
        public void show1()
        {}
        public void show2()
        {}
    }
    */
    public void method()
    {
        /*
        Inner in = new Inner();
        in.show1();
        in.show2();
        */
        //对其简化,写成匿名内部类的方式。
        Inter in = new Inter()//记住:内部类中一般方法不要过多。阅读性会很差。
        {
            public void show1()
            {}
            public void show2()
            {}
        };
        in.show1();
        in.show2();
        
    }
}
class InnerClassDemo6 
{
    public static void main(String[] args) 
    {
        System.out.println("Hello World!");
    }
}
interface Inter
{
    public void show();
}
class Outer
{
    //代码补足。要求使用匿名内部类。
    public static Inter method()
    {
        //既然在Oute类中使用到了Inter的对象。可以使用内部类来完成。
        //需要子类型,只要简化格式即可,因为接口中就只有一个方法。
        return new Inter()
        {
            public void show()
            {
                //code..;
            }
        };
//      return new Inner();
    } 
    
    /*
//  还原成内部类。  当静态方法访问内部类时,内部类必须是静态的。
    static class Inner implements Inter
    {
        public void show(){}
    }
    */
}
class InnerClassDemo7 
{
    public static void main(String[] args) 
    {
        Outer.method().show();
        /*
        Outer.method() //Outer类中有一个method的方法。这个方式静态的。
        Outer.method.show() //能调用show()的必然是对象,说明method方法运算完应该返回一个对象。而且能调用Inter中的show方法,说明这个对象的类型是Inter。
        */
}

匿名内部类和内部类的一个面试题

class Outer2
{
    public void method()
    {
        //以下两个对象有区别吗?
        new Object()
        {
            public void show(){}
        }.show();//这个可以编译通过。

        Object obj = new Object()
        {
            public void show(){}
        };
//      obj.show();//编译失败。为啥呢?因为匿名内部类是子类对象,当Object obj指向时,就被提升了Object。而Object类中没有定义show方法,编译失败。
    }
//  class Inner extends Object
//  {
//      public void show(){}
//  }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,608评论 18 399
  • 一、继承 当两个事物之间存在一定的所属关系,即就像孩子从父母那里得到遗传基因一样,当然,java要遗传的更完美,这...
    玉圣阅读 1,050评论 0 2
  • 专业考题类型管理运行工作负责人一般作业考题内容选项A选项B选项C选项D选项E选项F正确答案 变电单选GYSZ本规程...
    小白兔去钓鱼阅读 8,983评论 0 13
  • 一:java概述: 1,JDK:Java Development Kit,java的开发和运行环境,java的开发...
    慕容小伟阅读 1,778评论 0 10
  • 爸爸给两个人发草莓吃,问,谁先吃?东东抢着说:“我先吃!”。西西不示弱,大声说:“妹妹!”停了两秒,又补充了一句,...
    讲西讲东阅读 258评论 0 0