JavaSE知识-10(面向对象_权限修饰符&匿名内部类)

package关键字的概述及作用

  • A:为什么要有包

    • 将字节码(.class)进行分类存放
    • 包其实就是文件夹
  • 举例:
    学生:增加,删除,修改,查询
    老师:增加,删除,修改,查询
    ...

      方案1:按照功能分
      	com.hwh.add
      		AddStudent
      		AddTeacher
      	com.hwh.delete
      		DeleteStudent
      		DeleteTeacher
      	com.hwh.update
      		UpdateStudent
      		UpdateTeacher
      	com.hwh.find
      		FindStudent
      		FindTeacher
      
      方案2:按照模块分
      	com.hwh.teacher
      		AddTeacher
      		DeleteTeacher
      		UpdateTeacher
      		FindTeacher
      	com.hwh.student
      		AddStudent
      		DeleteStudent
      		UpdateStudent
      		FindStudent
    

包的定义及注意事项

  • A:定义包的格式
    • package 包名;
    • 多级包用.分开即可
  • B:定义包的注意事项
    • A:package语句必须是程序的第一条可执行的代码
    • B:package语句在一个java文件中只能有一个
    • C:如果没有package,默认表示无包名

带包的类编译和运行

  • A:如何编译运行带包的类
    • a:javac编译的时候带上-d即可
      • javac -d . HelloWorld.java
    • b:通过java命令执行。
      • java 包名.HelloWorld

会在当前路径下生成com>hwh>Demo1_Package

不同包下类之间的访问

先编译Person.java, 在baidu文件夹中生成Person.class
再编译Demo2_Package.java, 在hwh文件夹中生成Demo2_Package.class
再运行Demo2_Package

package com.baidu;
public class Person {
	private String name;
	private int age;
	
	public Person(){}

	public Person(String name,int age){
		this.name = name;
		this.age = age;
	}

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

	public void setAge(int age){
		this.age = age;
	}

	public String getName(){
		return name;
	}

	public int getAge(){
		return age;
	}
}
package com.hwh;
class Demo2_Package {
	public static void main(String[] args) {
		com.baidu.Person p = new com.baidu.Person("张三",23);
		System.out.println(p.getName()+"..."+p.getAge());
	}
}

import关键字的概述和使用

  • 让有包的类对调用者可见,不用写全类名了
  • 导包格式
    • import 包名;
    • 注意:
    • 这种方式导入是到类的名称。
    • 虽然可以最后写*,但是不建议。
package com.hwh;
import com.baidu.Person;
//import java.util.Scanner;		//在开发中我们用的都是导入具体的类
import java.util.*;				//*代表通配符,他会到该包下挨个匹配,匹配上就导入
class Demo2_Package {
	public static void main(String[] args) {
		com.baidu.Person p = new com.baidu.Person("张三",23);
		System.out.println(p.getName()+"..."+p.getAge());

	Scanner sc = new Scanner(System.in);
		System.out.println("请输入一个整数");
		int x = sc.nextInt();
		System.out.println(x);
	}
}

四种权限修饰符

/ 本类 同一个包下(子类和无关类) 不同包下(子类) 不同包下(无关类)
private
默认
protected
public

类及其组成所使用的常见修饰符

  • A:修饰符:

    • 权限修饰符:private,默认的,protected,public
    • 状态修饰符:static,final
    • 抽象修饰符:abstract
  • B:类:

    • 权限修饰符:默认修饰符,public

    • 状态修饰符:final

    • 抽象修饰符:abstract

    • 用的最多的就是:public

  • C:成员变量:

    • 权限修饰符:private,默认的,protected,public

    • 状态修饰符:static,final

    • 用的最多的就是:private

  • D:构造方法:

    • 权限修饰符:private,默认的,protected,public

    • 用的最多的就是:public

  • E:成员方法:

    • 权限修饰符:private,默认的,protected,public

    • 状态修饰符:static,final

    • 抽象修饰符:abstract

    • 用的最多的就是:public

  • F:除此以外的组合规则:

    • 成员变量:public static final 接口
    • 成员方法:
      • public static
      • public abstract
      • public final

内部类概述和访问特点

  • 内部类访问特点
    • 内部类可以直接访问外部类的成员,包括私有。
    • 外部类要访问内部类的成员,必须创建对象。
    • 外部类名.内部类名 对象名 = 外部类对象.内部类对象
class Demo1_Innerclass {
	public static void main(String[] args) {
	    //Inner i = new Inner();
		//i.method();
		//外部类名.内部类名 = 外部类对象.内部类对象
		Outer.Inner oi = new Outer().new Inner();	//创建内部类对象
		oi.method();
	}
}

class Outer{
	private int num = 10;
	class Inner{
		public void method(){
			System.out.println(num);
		}
	}
}

运行结果为10

成员内部类私有使用

class Demo2_InnerClass {
	public static void main(String[] args) {
		//Outer.Inner oi = new Outer().new Inner();
		//oi.method();

		Outer o = new Outer();
		o.print();
	}
}

class Outer {
	private  int num = 10;
	private class Inner {   //内部类私有
		public void method() {
			System.out.println(num);
		}
	}

	public void print() {
		Inner i = new Inner();
		i.method();
	}
}

运行结果为10

静态成员内部类

  • 成员内部类被静态修饰后的访问方式是:
    • 外部类名.内部类名 对象名 = 外部类名.内部类对象;
class Demo3_InnerClass {
	public static void main(String[] args) {
		Outer.Inner oi = new Outer.Inner();
		oi.method();

	Outer.Inner2.print();
	}
}

class Outer{
	static class Inner{
		public void method(){
			System.out.println("method");
		}
	}

	static class Inner2{
		public static void print(){
			System.out.println("print");
		}
	}

}

运行结果为method print

局部内部类访问局部变量

class Demo4_InnerClass {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}
//局部内部类
class Outer {
	public void method() {
		final int num = 10; //int num = 10 错误: 从内部类中访问本地变量num; 需要被声明为最终类型
		class Inner {
			public void print() {
				System.out.println(num);
			}
		}

		Inner i = new Inner();
		i.print();
	}

	/*public void run() {
		Inner i = new Inner();				//局部内部类,只能在其所在的方法中访问
		i.print();
	}*/
}

因为当调用这个方法时,局部变量如果没有用final修饰,他的生命周期和方法的生命周期是一样的,当方法弹栈,这个局部变量也会消失,那么如果局部内部类对象还没有马上消失想用这个局部变量,就没有了,如果用final修饰会在类加载的时候进入常量池,即使方法弹栈,常量池的常量还在,也可以继续使用
但是jdk1.8取消了这个事情,虽然取消,如果在书写代码时候,没有手动添加,系统底层也会默给你final

匿名内部类的格式

  • 匿名内部类
    • 就是内部类的简化写法。
  • 前提:存在一个类或者接口
    • 这里的类可以是具体类也可以是抽象类。
  • 格式:
    new 类名或者接口名(){
    重写方法;
    }
  • 本质:
    • 是一个继承了该类或者实现了该接口的子类匿名对象。
class Demo1_NoNameInnerClass {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}

interface Inter {
	public void print();
}

class Outer {
	class Inner implements Inter {
		public void print() {
			System.out.println("print");
		}
	}

	public void method(){
		Inner i = new Inner();
		i.print();
		
	}
}

有名字的内部类
运行结果为print

修改method()

public void method(){
		//Inner i = new Inner();
		//i.print();
		//new Inner().print();
		//Inter i = new Inner();			//父类引用指向子类对象
		
		new Inter() {						//实现Inter接口
			public void print() {			//重写抽象方法
				System.out.println("print");
			}
		}.print();     //整个代表Inter()的子对象,调用print()方法
	}

运行结果为print

匿名内部类重写多个方法调用

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

interface Inter {
	public void show1();
	public void show2();
}
//匿名内部类只针对重写一个方法时候使用, 多个方法时为避免冗余使用有名字类
class Outer {
	public void method() {
		/*new Inter(){
			public void show1() {
				System.out.println("show1");
			}

			public void show2() {
				System.out.println("show2");
			}
		}.show1();

		new Inter(){
			public void show1() {
				System.out.println("show1");
			}

			public void show2() {
				System.out.println("show2");
			}
		}.show2();*/

		Inter i = new Inter(){
			public void show1() {
				System.out.println("show1");
			}

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

			/*public void show3() {
				System.out.println("show3");
			}*/
		};

		i.show1();    //编译时看父类接口中的show1(), 运行时看子类对象中的show2()方法
		i.show2();
		//i.show3();						//匿名内部类是不能向下转型的,因为没有子类类名
	}
}

运行结果为show1 show2

匿名内部类在开发中的应用

有名字类

		//这里写抽象类,接口都行
		abstract class Person {
			public abstract void show();
		}
	
		class PersonDemo {
			public void method(Person p) { //Person p = new Student() 
				p.show();
			}
		}
	
		class PersonTest {
			public static void main(String[] args) {
				//如何调用PersonDemo中的method方法呢?
				PersonDemo pd = new PersonDemo ();
				pd.method(new Student());
			}
		}

匿名类

	//这里写抽象类,接口都行
		abstract class Person {
			public abstract void show();
		}
	
		class PersonDemo {
			/*
			Person p = new new Person(){     //父类引用指向子类对象
					public void show(){
						System.out.println("show");
					}
				}
			*/
			public void method(Person p) {  
				p.show();
			}
		}
	
		class PersonTest {
			public static void main(String[] args) {
				//如何调用PersonDemo中的method方法呢?
				PersonDemo pd = new PersonDemo ();
				pd.method(new Person(){     //匿名内部类当作参数传递(本质把匿名内部类看作一个对象)
					public void show(){
						System.out.println("show");
					}
				});
			}
		}
posted @ 2020-01-30 12:03  20145232  阅读(140)  评论(0编辑  收藏  举报