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
- a:javac编译的时候带上-d即可
会在当前路径下生成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");
}
});
}
}