Java学习 第十三天

第一章 权限修饰符

1.1 概述

Java当中提供了四种权限修饰符,使用不同的访问权限修饰符修饰时,被修饰的内容会有不同的访问权限。

  • public:公共的
  • protected:受保护的
  • (default):默认的,意思是什么都不写,不是默认方法里的关键字default
  • private:私有的

1.2 不同权限的访问能力

\publicprotected(default)private
同一个类(我自己)yesyesyesyes
同一个包中(我邻居)yesyesyesno
不同包的子类(我儿子)yesyesnono
不同包的非子类(陌生人)yesnonono

注意: 包和它的子包并不是同一个包,只要不是同一个包,在引用其中的类时候,就必须写import导入语句。

第二章 内部类

2.1 概述

将一个类A定义在另一个类B里面,里面的那个类A就称之为内部类,B则称为外部类

分类:
1.成员内部类
2.普通局部内部类
3.匿名局部内部类

2.2 成员内部类

定义在类中方法外的类就是成员内部类

成员内部类的定义格式:

修饰符 class  外部类名称{
	修饰符 class 内部类名称{
		//	...
	}
	//	...
}

访问规则:
内部类使用外部类时,无论外部类使用什么修饰符,都可以随意访问。
外部类访问内部类时,必须有内部类对象才可以。

访问方式:
如何使用成员内部类?有两种方式:
1.在外部类的方法当中,使用内部类;然后main只是调用外部类的方法。【间接方式】
2.直接方式(直接创建内部类对象):公式:
外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称()

代码演示:

//	定义外部类Body
public class Body { //  外部类
    public class Heart{ //  成员外部类

        //  内部类的方法
        public void beat(){
            System.out.println("心脏跳动,蹦蹦蹦!");
            System.out.println("我叫:" + name);   //  正确写法!
        }

    }

    //  外部类的成员变量
    private String name;

    //  外部类的方法
    public void methodBody(){
        System.out.println("外部类的方法");
        new Heart().beat();	//	这是匿名对象的调用格式
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

//	定义测试类
public class Demo01InnerClass {

    public static void main(String[] args) {
        Body body = new Body();
        //  通过外部类的对象,调用外部类的方法,里面间接使用内部类Heart
        body.methodBody();
        System.out.println("====================");

        //  按照公式写:
        Body.Heart heart = new Body().new Heart();
        heart.beat();
    }
}

如果出现了成员内部类和外部类成员变量重名现象,那么格式是:外部类名称.this.外部类成员变量名

public class Outer {
    int num = 10;   //  外部类的成员变量
    public void method(){
        System.out.println("外部方法");
    }

    public class Inner{
        int num = 20;   //  内部类的成员变量

        public void methodInner(){
            int num = 30;   //  内部类的局部变量
            System.out.println(num);    //  局部变量,就近原则
            System.out.println(this.num);   //  内部类的成员变量
            System.out.println(Outer.this.num); //  外部类的成员变量
        }
        public void method(){
            System.out.println("内部方法");
        }
    }
}

2.3 普通局部内部类

局部内部类,如果希望访问所在方法的局部变量,那么这个局部变量必须是 【有效final的】
备注:
从Java 8+开始,只要局部变量事实上没有进行改变,那么final关键字可以省略。
原因:
1.new出来的对象在堆内存当中。
2.但是局部变量是跟着方法走的,在栈内存当中。
3.方法运行结束之后,立刻出栈,局部变量就会立刻消失。
4.new出来的对象会在堆内存当中持续存在,直到垃圾回收为止。

注意: 当局部内部类和外部类变量重名时,会发生覆盖现象,将无法在局部内部类访问到外部类的同名变量

代码演示:

public class MyOuter {
    public void methodOuter(){
        final int num = 10;   //  所在方法的局部变量

        class MyInner{
        	//  此处定义num和上面的局部变量重名
        	//	相当于num不再指向10,重新指向了20,发生了覆盖现象
            //  无法再访问到10
            //  Java当中不允许发生方法的嵌套
        	int num = 20;
            public void methodInner(){
                System.out.println(num);	//	20
            }
        }
    }
}

2.4 外内部类的权限修饰符规则

public > protected > (default) > private
定义一个类的时候,权限修饰符规则:
1.外部类只能是这两种: public / (default)
2.成员内部类: public / protected / (default) / private
3.局部内部类必须无修饰符; 什么都不能写(这并不是(default)修饰符的意思)

2.5 匿名内部类(局部内部类的一种)【重点】

2.5.1 概述

是内部类的简化写法。它的本质是一个带具体实现的父类或者父接口的,匿名的子类对象
匿名内部类:匿名内部类也是局部内部类,这意味着匿名内部类是出现在一个方法的内部的,如果它要访问这个方法的参数或者方法中定义的变量,则这些参数和变量必须被修饰为final

2.5.2 使用格式

匿名内部类的定义格式:

接口名称 对象名 = new 接口名称(){
    //  覆盖重写所有的抽象方法
};

对上述格式“new 接口名称(){…}”进行解析:
1.new代表创建对象的动作
2.接口名称就是匿名内部类需要实现的那个接口
3.{…}这才是匿名内部类的内容

注意: 匿名内部类是指匿名了实现类的名称,而不是匿名了实现类对象的名称

2.5.3 使用演示

//	定义接口MyInterface
public interface MyInterface {
    void method1();   //  抽象方法,这里省略了public abstract
    void method2();
}


//	定义实现类MyInterfaceImpl
public class MyInterfaceImpl implements MyInterface{
    @Override
    public void method1() {
        System.out.println("实现覆盖重写了方法!");
    }

    @Override
    public void method2() {

    }
}


//	定义测试类
public class DemoMain {
    public static void main(String[] args) {
        //  多态写法
        //  这里只用了一次实现类MyInterfaceImpl创建对象,但是需要创建一个MyInterfaceImpl类作为实现类,太过麻烦,因此可以使用匿名内部类写法
//        MyInterface obj = new MyInterfaceImpl();
//        obj.method();

//        MyInterface some = new MyInterface();   //  错误写法

        //  使用匿名内部类,但不是匿名对象
        MyInterface objA = new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-A");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-A");
            }
        };
        objA.method1();
        objA.method2();
        System.out.println("======================");

        //  不仅使用了匿名内部类,而且省略了对象名称,也是匿名对象
        new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-B");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-B");
            }
        }.method1();
        //  因为匿名对象无法调用第二次方法,所以需要再创建一个匿名内部类的匿名对象
        new MyInterface() {
            @Override
            public void method1() {
                System.out.println("匿名内部类实现了方法!111-B");
            }

            @Override
            public void method2() {
                System.out.println("匿名内部类实现了方法!222-B");
            }
        }.method2();
    }
}

注意: 要注意上述代码当中匿名对象和匿名内部类的结合使用情况

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值