摘要
- package关键字
- 权限修饰符
- 匿名内部类
package关键字的概述及作用
-
A:为什么要有包
- 将字节码(.class)进行分类存放
- 包其实就是文件夹
B:包的概述
-
举例:
学生:增加,删除,修改,查询
老师:增加,删除,修改,查询
...方案1:按照功能分 com.heima.add AddStudent AddTeacher com.heima.delete DeleteStudent DeleteTeacher com.heima.update UpdateStudent UpdateTeacher com.heima.find FindStudent FindTeacher 方案2:按照模块分 com.heima.teacher AddTeacher DeleteTeacher UpdateTeacher FindTeacher com.heima.student AddStudent DeleteStudent UpdateStudent FindStudent
包的定义及注意事项
- A:定义包的格式
- package 包名;
- 多级包用.分开即可
- B:定义包的注意事项
- A:package语句必须是程序的第一条可执行的代码
- B:package语句在一个java文件中只能有一个
- C:如果没有package,默认表示无包名
- C:案例演示
- 包的定义及注意事项
带包的类编译和运行
- A:如何编译运行带包的类
- a:javac编译的时候带上-d即可
- javac -d . HelloWorld.java
- b:通过java命令执行。
- java 包名.HelloWord
- a:javac编译的时候带上-d即可
不同包下类之间的访问
- A:案例演示
- 不同包下类之间的访问
import关键字的概述和使用
- A:案例演示
- 为什么要有import
- 其实就是让有包的类对调用者可见,不用写全类名了
- 为什么要有import
- B:导包格式
- import 包名;
- 注意:
- 这种方式导入是到类的名称。
- 虽然可以最后写*,但是不建议(即匹配该包中所有的类,当匹配上时导入该类)。
- C:package,import,class有没有顺序关系(面试题)有,从前到后
四种权限修饰符的测试
-
A:案例演示
- 四种权限修饰符
-
B:结论
本类 同一个包下(子类和无关类) 不同包下(子类) 不同包下(无关类) private Y 默认 Y Y protected Y Y Y public Y Y Y Y
类及其组成所使用的常见修饰符
-
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
内部类概述和访问特点
- A:内部类概述
- 在类中定义的类
- B:内部类访问特点
- a:内部类可以直接访问外部类的成员,包括私有。
- b:外部类要访问内部类的成员,必须创建对象。
- 外部类名.内部类名 对象名 = 外部类对象.内部类对象;
- C:案例演示
class Outer{ class Inner{ public void method() { System.out.println("内部类"); } } } class Test { public static void main(String[] args) { Outer.Inner oi=new Outer().new Inner();//外部类访问内部类成员,需要创建对象 oi.method(); } }
成员内部类私有使用
- private
class Outer{
private class Inner{
public void method() {
System.out.println("内部类");
}
}
public void print() { //内部类私有时需要通过成员函数调用
Inner i=new Inner(); //外部类访问内部类成员,需要创建对象
i.method();
}
}
class Test {
public static void main(String[] args) {
Outer o=new Outer();
o.print();
}
}
静态成员内部类
- static
- B:成员内部类被静态修饰后的访问方式是:
- 外部类名.内部类名 对象名 = 外部类名.内部类对象;
class Outer{ static class Inner{ public void method() { System.out.println("method"); } public static void print() { System.out.println("print"); } } } class Test { public static void main(String[] args) { Outer.Inner oi=new Outer.Inner();//静态内部类的调用方法 oi.method(); Outer.Inner.print(); //静态内部类中的静态函数的调用方法 } }
成员内部类的面试题
- A:面试题
要求:使用已知的变量,在控制台输出30,20,10。
class Outer {
public int num = 10;
class Inner {
public int num = 20;
public void show() {
int num = 30;
System.out.println(?);
System.out.println(??);
System.out.println(???);
}
}
}
class InnerClassTest {
public static void main(String[] args) {
Outer.Inner oi = new Outer().new Inner();
oi.show();
}
}
/*
? num
?? this.num
??? Outer.this.num
*/
局部内部类访问局部变量的问题
- A:案例演示
局部内部类访问局部变量必须用final修饰 (注:jdk1.8中不需要显式使用final修饰,但是此变量是effectively final的,同样不可修改)
-
局部内部类在访问他所在方法中的局部变量必须用final修饰,为什么?
因为当调用这个方法时,局部变量如果没有用final修饰,他的生命周期和方法的生命周期是一样的,当方法弹栈,这个局部变量也会消失,那么如果局部内部类对象还没有马上消失想用这个局部变量,就没有了,如果用final修饰会在类加载的时候进入常量池,即使方法弹栈,常量池的常量还在,也可以继续使用但是jdk1.8取消了这个事情,虽然取消,如果在书写代码时候,没有手动添加,系统底层也会默给你final
class Outer {
public void method() {
int num=10; //
class Inner { //局部内部类,只能在该方法内访问
public void print() {
System.out.println(num);
}
}
Inner i=new Inner();
i.print();
}
}
class Test {
public static void main(String[] args) {
Outer o=new Outer();
o.method();
}
}
匿名内部类的格式和理解
- A:匿名内部类
- 就是内部类的简化写法。
- B:前提:存在一个类或者接口
- 这里的类可以是具体类也可以是抽象类。
- C:格式:
new 类名或者接口名(){ 重写方法; }- D:本质是什么呢?
- 是一个继承了该类或者实现了该接口的子类匿名对象。
- E:案例演示
- 按照要求来一个匿名内部类
interface Inter{
public void print();
}
class Outer {
// class Inner implements Inter{
// public void print() {
// System.out.println("print");
// }
// }
// public void metohd() {
// Inner i=new Inner();
// i.print();
// }
// 以上类和方法可以组合为以下匿名内部类
public void method() {
new Inter() { //实现Inter接口
public void print() { //重写抽象方法
System.out.println("print");
}
}.print();
}
}
class Test {
public static void main(String[] args) {
Outer o=new Outer();
o.method();
}
}
匿名内部类重写多个方法调用
- 需要调用多个方法时每次都要重写所有方法,过于复杂,因此只推荐重写一个方法时使用匿名内部类。匿名内部类没有子类类名,不能向下转型。
匿名内部类在开发中的应用
-
A:代码如下
//这里写抽象类,接口都行 abstract class Person { public abstract void show(); } class PersonDemo { public void method(Person p) { p.show(); } } //创建一个子类继承抽象类 class Student extends Person{ public void show() { System.out.println("show"); } } class PersonTest { public static void main(String[] args) { //如何调用PersonDemo中的method方法呢? PersonDemo pd = new PersonDemo (); //通过调用子类来实例化 pd.method(new Student()); } } /* 第二种方法:直接在调用方法时使用匿名内部类 pd.method(new Person() { public void show() { System.out.println("show"); } });
匿名内部类的面试题
-
A:面试题
//按照要求,补齐代码 interface Inter { void show(); } class Outer { //补齐代码 } class OuterDemo { public static void main(String[] args) { Outer.method().show(); } } //要求在控制台输出”HelloWorld”interface Inter { void show(); } class Outer { //补齐代码 public static Inter method() { return new Inter() { public void show() { System.out.println("hello world"); } }; } } class Test { public static void main(String[] args) { Outer.method().show(); //链式编程,调用方法后还能调用方法,说明返回的是对象 } }