java基础之内部类

算法露台
• 阅读 189

内部类

内部类是一种较为特殊的类形式,定义在另一个类中的类,叫做内部类,是主类的一部分,可以把一些逻辑相关的类组织在一起,并控制位于内部的类的可见性。

作用

  • 内部类方法可以访问该类定义所在的作用域中的数据,包括私有数据
  • 内部类可以对同一个包中的其他类隐藏起来

内部类分为

  • 静态内部类
  • 成员内部类
  • 局部内部类
  • 匿名内部类

<!-- more -->

成员内部类

成员内部类可以等同的看做是成员变量,可以有权限修饰符,内部不能有静态声明

public class Outter {

    class Inner {
        public void print(String s) {
            System.out.println(s);
        }
    }

    public Inner buildInner() {
        return new Inner();
    }

    public static void main(String[] args) {
        Outter outter = new Outter();
        Inner inner = outter.buildInner();
        inner.print("输出");
    }
}

成员内部类与外部类联系

成员内部类可以使用外部类的所有成员,而且不需要任何特殊条件,且拥有其对于外部类所有元素的访问权。

在创建成员内部类对象时,它会与创建它的外围对象有某种联系隐式引用(内部类对象会秘密的捕获一个指向外部类的引用,通过这个引用可以用来访问外部类的成员),通过这个隐式引用可以访问外部类的所有成员

public class Sequence {

    private Object[] items;

    private int next;

    public Sequence(int size) {
        items = new Object[size];
    }

    /**
     * 添加元素
     *
     * @param obj
     */
    public void add(Object obj) {
        if (next < items.length) {
            items[next] = obj;
            next++;
        }
    }

    public SequenceSelector selector() {
        return new SequenceSelector();
    }

    public static void main(String[] args) {
        Sequence sequence = new Sequence(10);
        for (int i = 0; i < 10; i++) {
            sequence.add(i);
        }
        SequenceSelector selector = sequence.selector();
        while (!selector.end()) {

            System.out.print(selector.current() + " ");
            selector.next();
        }
    }

    private class SequenceSelector {

        private int current = 0;

        public boolean end() {
            return current == size();
        }

        public int size() {
            return items.length;
        }

        public Object current() {
            return items[current];
        }

        public void previous() {
            if (current > 0) {
                current--;
            }
        }

        public void next() {
            if (current < items.length) {
                current++;
            }
        }
    }
}

使用.this和.new

在成员内部类如何生成外部类对象的引用,使用外部类名.this

public class DoThis {
    class Inner{
        public DoThis outer(){
            return DoThis.this;
        }
    }

    public void f(){
        System.out.println("DoThis.f()");
    }

    public Inner inner(){
        return new Inner();
    }

    public static void main(String[] args) {
        DoThis doThis = new DoThis();
        Inner inner = doThis.inner();
        inner.outer().f();
    }
}

外部类如何创建成员内部类的对象,需要使用外部类实例.new

public class DoNew {

    class Inner {
        public void f(){
            System.out.println("Inner.f()");
        }
    }

    public static void main(String[] args) {
        DoNew doNew = new DoNew();
        Inner inner = doNew.new Inner();
        inner.f();
    }
}

在拥有外部类对象之前是不可能创建成员内部类对象的,成员内部类对象会连接到创建它的外部类对象上,静态内部类不需要外部类对象的引用。

对于私有的成员内部类,外部类是可以访问到它的,其他的类无法访问到,如下

public class DoNew {

    private class Inner {
        public void f(){
            System.out.println("Inner.f()");
        }
    }

    public static void main(String[] args) {
        DoNew doNew = new DoNew();
        // 外部类中可以访问到
        Inner inner = doNew.new Inner();
        inner.f();
    }
}

public class Test {
    public static void main(String[] args) {
        DoNew doNew = new DoNew();
        // 其他类无法访问
        // 'DoNew.Inner' has private access in 'DoNew'
        //doNew.new Inner();
    }
}

成员内部类深度剖析

上面提到在拥有外部类对象之前是不可能创建成员内部类对象的,成员内部类对象会连接到创建它的外部类对象上,这里来说明一下为什么

来看一下这个Sequence.SequenceSelector类的构造器是什么

javap -c Sequence.SequenceSelector

 com.zhanghe.study.inner.Sequence$SequenceSelector(com.zhanghe.study.inner.Sequence, com.zhanghe.study.inner.Sequence$1);
    Code:
       0: aload_0
       1: aload_1
       2: invokespecial #1                  // Method "<init>":(Lcom/zhanghe/study/inner/Sequence;)V
       5: return

发现这个Sequence.SequenceSelector类构造器默认并不是无参构造器,而是要传外部类对象的

局部内部类

在方法作用域内创建的类,称为局部内部类,局部内部类没有访问修饰符,可以等同的看做是局部变量,局部内部类在访问局部变量时,局部变量必须使用final修饰

Variable 'name' is accessed from within inner class, needs to be final or effectively final
public class MethodOuter {

    public Destination desc(String s){
        // 局部内部类
        class Inner implements Destination{
            @Override
            public void f(){
                System.out.println("Inner.f()传入参数"+s);
            }
        }
        return new Inner();
    }

    public static void main(String[] args) {
        MethodOuter methodOuter = new MethodOuter();
        methodOuter.desc("test").f();
    }
}

Inner是desc()方法的一部分,在方法之外访问不到

匿名内部类

匿名内部类,就是没有定义类名的一种嵌套类

如一个接口/类的方法的某个实现方式在程序中只会执行一次,但为了使用它,我们需要创建它的实现类/子类去实现/重写。此时可以使用匿名内部类的方式,可以无需创建新的类,减少代码冗余

public interface Contents {
    void f();
}

public class Outer1 {
    public Contents contents(){
        return new Contents(){
            @Override
            public void f() {
                System.out.println("Contents匿名内部类.f()");
            }
        };
    }

    public static void main(String[] args) {
        Outer1 outer1 = new Outer1();
        outer1.contents().f();
    }
}

静态内部类

静态内部类可以等同的看做是静态变量,可以有权限修饰符

静态内部类,内部类对象与外部类对象之间没有联系,将内部类声明为static,普通的内部类对象隐式的保存了一个引用,指向创建它的外部类对象,当内部类是static的时候,意味着

  • 要创建静态内部类的对象,并不需要其外部类的对象
  • 不能从静态内部类的对象中访问非静态的外围对象

    普通内部类不能有static数据和static字段,也不能包含静态内部类,但是静态内部类可以。

    public class Outer2 {
        private static class Inner2{
            private static int i = 0;
            private static void f(){
                System.out.println("Inner2.f()");
            }
        }
    
        public static void main(String[] args) {
            Inner2.f();
        }
    }

在创建静态内部类时不需要将静态内部类的实例绑定在外部类实例上。

接口内部类

静态内部类可以作为接口的一部分,放到接口中任何类都是public static的

https://zhhll.icu/2020/java基础/面向对象/9.java基础之内部类/

本文由mdnice多平台发布

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
java中的内部类内部接口详解
简介一般来说,我们创建类和接口的时候都是一个类一个文件,一个接口一个文件,但有时候为了方便或者某些特殊的原因,java并不介意在一个文件中写多个类和多个接口,这就有了我们今天要讲的内部类和内部接口。内部类先讲内部类,内部类就是在类中定义的类。类中的类可以看做是类的一个属性,一个属性可以是static也可以是非static的。而内部类
Wesley13 Wesley13
3年前
java基础常见面试题,这是一篇超长的随笔!!!
1\.Java基础部分.......................................................41、一个".java"源文件中是否可以包括多个类(不是内部类)?有什么限制?..42、Java有没有goto?.................................................4
Wesley13 Wesley13
3年前
Java进阶
到现在为止,我们都是在Java文件中直接定义类。这样的类出现在包(package)的级别上。Java允许类的嵌套定义。这里将讲解如何在一个类中嵌套定义另一个类。!(http://static.oschina.net/uploads/img/201601/21115000_D2R5.jpg)嵌套内部类Java允许我们在类的内部
Wesley13 Wesley13
3年前
Java中的静态内部类学习记录
一、静态内部类:1.静态内部类,静态内部类对象可以不依赖于外部类对象,直接创建。 2.静态内部类中,只能直接访问外部类的静态成员,如果需要调用非静态成员,可以通过实例对象实例。3.静态内部类对象实例时,可以不依赖于外部类对象4.可以通过外部类.内部类.静态成员的方式,访问内部类中的静态成员。5.当内部类属性与外部类属性同名时,默认直接
Wesley13 Wesley13
3年前
(翻译)Java SE 8 Lambda 特性与基本原理(下)
6,词法域(Lexical Scoping)确定内部类中变量名字(包括this)的意义要比在顶级类中困难的多,并且很容易出错。继承成员包括类对象中的方法可能不小心就覆盖了外部类的声明, 未加限定的this引用总是指向外部类自身。Lambda表达式更加简单:他们不会从超类中继承任何名字,也不会引入任何新的级别的作用域。相反,他们
Wesley13 Wesley13
3年前
Java 内部类理解
为什么使用内部类?答:每个内部类都能独立地继承一个(接口的)实现,所以无论外围类是否已经继承了某个(接口的)实现,对于内部类都没有影响。内部类有哪些?答:内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类这些内部类需要注意的点?成员内部类1)成员内部类中不能存在任何static的变量和方法2)成员内部类是依附
Wesley13 Wesley13
3年前
Java中匿名类的两种实现方式
使用匿名内部类课使代码更加简洁、紧凑,模块化程度更高。内部类能够访问外部内的一切成员变量和方法,包括私有的,而实现接口或继承类做不到。然而这个不是我说的重点,我说的很简单,就是匿名内部类的两种实现方式:第一种,继承一个类,重写其方法;第二种,实现一个接口(可以是多个),实现其方法。下面通过代码来说明:1.publicclass TestAnonymo
Wesley13 Wesley13
3年前
Java私有内部类实例化对象访问私有内部类中的方法
今天和小伙伴讨论私有类的对象创建问题,大家都知道外部类是不能设置为私有的,但是内部类可以设置为私有。问题是私有内部类是否可以实例化一个对象,并调用私有内部类中的方法和属性呢?答案是:Yes!Java语法规范既然允许创建私有的内部类,私有内部类中的方法肯定会被调用。要不然,私有内部类就失去了它的作用。于是小coder尝试写了如下代码。
Stella981 Stella981
3年前
Scala进阶:扩大内部类作用域的两种方式:伴生对象、类型投影
内部类初始用户,内部类的作用域属于:外部类对象,不同外部类对象中的内部类对象类型不同会报错typemismatch。实例代码:注意:    定义了一个外部类Person 和一个内部类Studentpackagecom.hadoop.ljs.spark.study主函数:这里person2对象里面的
Wesley13 Wesley13
3年前
Java封装:访问控制
1、类的访问控制符有两种:public,default(默认的,什么都不用写)     default就是包内访问控制符。     当然对于内部类还可以有static修饰,而用static修饰的内部类叫作嵌套类。2、属性和方法的访问控制符有四种:private,default,protected,publi
小万哥 小万哥
1年前
Java 抽象类与方法:实现安全性与代码重用
Java内部类简介在Java中,可以嵌套类(即类内部的类),称为内部类。嵌套类的目的是将属于一起的类分组,从而使您的代码更可读和可维护。访问内部类要访问内部类,请创建外部类的对象,然后创建内部类的对象:javaclassOuterClassintx10;c