java 内部类的学习总结

内部类
参考文档:https://www.runoob.com/w3cnote/java-inner-class-intro.html

在 Java 中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。

1.成员内部类

//1.成员内部类地位和类的成员等同 
//2.成员内部类可以访问外部类的任意成员方法。 
//3.如果外部类需要访问内部类,则需要先创建内部类对象。直接创建。 
//4.其他类获取成员内部类对象为 new People().new Man(); 
//5.成员内部类的访问修饰符可以控制他的访问权限。这一点和类的成员是一致的。
成员内部类为什么不可以声明静态属性或方法?
三点说明:
static类型的属性和方法,在类加载的时候就会存在于内存中。
要想使用某个类的static属性和方法,那么这个类必须要加载到虚拟机- 中。
非静态内部类并不随外部类一起加载,只有在实例化外部类之后才会加载。
 结论:
非静态内部类不能有静态成员!
成员内部类必须先实例化外部类对象然后再实例化成员内部类;

定义类
package java_basic.stuInnerClass;
import org.junit.Test;
public class People {
    private  String name="小明";
    public int age=29;
    @Test
    public void method01() {
        System.out.println( new Man().showMessage());
        System.out.println(new People().age);//29
        System.out.println(People.this.age);//29
    }
    public People(){}
    //成员内部类
    //这个修饰符,用途类成员的用法非常一致。
    //搞不清访问修饰符,你需要参考:https://blog.csdn.net/qq_40331861/article/details/106949169
   public class Man{
        public int age =28;
        public String showMessage(){
            return "hello world"+"  name="+name;
        }
        public int getAge(){
            return age;
        }
    }
}
访问成员内部类
package java_basic.other;//注意这个包名 和People如果不在同一包里,就需要注意People中内部类前的访问修饰符了。
import java_basic.stuInnerClass.People;
public class Main extends People {
    public void method01() {
       People.Man man = new People().new Man();//其他类获取成员内部类的方式
    }
}

2.局部内部类

参考文档:https://www.cnblogs.com/SQP51312/p/6108848.html

定义在方法中的内部类。

1、局部内部类不能被public、private、static修饰;

2、在外部类中不能创建内部类的实例;

3、创建局部内部类的实例只能在包含他的方法中;

4、内部类访问包含他的方法中的变量必须有final修饰,版本升级就不需要了;

5、外部类不能访问局部内部类,只能在方法体中访问局部内部类,且访问必须在内部类定义之后。

假如现在有这样的一个需求:

要通过方法method(),获取某个类OBJ的重写后的对象。而这个需求只在这个地方(OBJ的某个方法)用。

可以用局部内部类来实现。

package java_basic.stuInnerClass;
import org.junit.Test;
//局部内部类
public class People01 {
    @Test
    public void method() {
        showMessage().getNumber();

    }
    private String name = "小明";
//该方法要求返回一个OBJ对象,业务需要该对象被重写。由于改重写对象只会在此处用,因此可以使用局部内部类。
//更加简便地是使用匿名内部类
    public  OBJ showMessage(){
         int num = 10;
         //
        class inner extends OBJ{
            //String name = "小军";//局部内部类的属性如果和外部类同名,则会隐藏外部属性。
            //static String abc="";//报错
            int score  = 100;
            public void getNumber(){
                String name = "小军1";//
                int mark = 23;
                //局部内部类可以直接访问外部类的属性,即便是private
                // 局部内部类的属性如果和外部类同名,则会隐藏外部属性。
                System.out.println(name);//小军1
                //解决办法
                System.out.println(People01.this.name);//小明
                System.out.println(num);//版本更新,访问已经可以不用final修饰的外部方法了。
                System.out.println(score);
                System.out.println(mark);

            }
        }
        return new inner();
    }
    public class OBJ{
        public void getNumber(){
            System.out.println("number is 20");
        }
    }
}

3.静态内部类

静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,这点很好理解,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。

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

4.匿名内部类

//匿名内部类的测试
public class People02 {
    @Test
    public void method() {
        subject.showMessage();//how are you.  //xxx
        animal.method();//method
        fruit.getNumber();//fruit geNumber
        fruit.toString();
    }


    //1.获取Subject类的匿名子类
    Subject subject = new Subject(){
        //里面的内容和子类集成一样的协防,可以重写父类方法,也可以不重写;
        @Override
        public void showMessage() {
            super.showMessage();
            method();
        }
        public void method(){//非重写方法外部无法访问,仅提供内部使用
            System.out.println("xxx");
        }
    };

    //2.获取接口的匿名实现
    Animal animal = new Animal() {
        @Override
        public void showMessage() {
            System.out.println("ShowMessae");
        }

        @Override
        public void method() {
            System.out.println("method");

        }

    };
    //3.当接口中只有一个方法时,可以通过lambda表达式简写
    //为了限制该接口只有一个自定义抽象方法,可以在该接口上加@FunctionalInterface
    //参考文档https://blog.csdn.net/qq_27610859/article/details/88861988
    Fruit fruit = () -> {
        System.out.println("fruit geNumber");
    };


}
//接口
interface Animal{
    void showMessage();//接口中的属性的默认是public static final 、方法是public abstract
    void method();
}

@FunctionalInterface
interface Fruit  {

    void getNumber();
    int hashCode();//Object 中的抽象方法可以再次出现 此处仅作为测试 并不知道其作用
    String toString();//Object 中的抽象方法可以再次出现 此处仅作为测试 并不知道其作用
    //String getName();//由于注解的缘故 在加一个自定义抽象方法,会报错。
}

//普通类
class Subject{
    public void showMessage(){
        System.out.println("how are you.");
    }
}

5,进一步有匿名内部类的知识(lambda缩写),结合函数是几口Function<T,R>的相关知识,

我们可以实现想js中那样的回调函数。

import java.util.function.Function;
public  class People03 {
    //回掉函数
    public static <T,R> void CallBack(Function<T,R> fnt,T t){
        fnt.apply(t);
    }
    @Test
    public void method() {
      //调用回掉函数
        People03.CallBack((Function<Integer, Object>) o ->(o*10),23);
    }
}

Function<T,R> 是有系统提供的,当然自己也可以编写,非常简单。只要注意里面有一个抽象方法apply().

People03.CallBack((Function<Integer, Object>) o ->(o*10),23); 调用时(Function<Integer, Object>) o ->(o*10)为产生一个Function<Integer, Object>对象,o ->(o*10)就是他的实现。o 是实现方法的传入形参。

回掉函数的实际调用在回调函数的的内部,即fnt.apply(t); t 为接收实参。

函数式接口的相关只是不了解的可以查一下。

参考文档:https://www.cnblogs.com/rever/p/9725173.html

这样做的好处:体味一下函数式编程以及实现,以及回调的一些思想。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值