07day
1.继承
继承:提高了代码的复用性,让类与类之间产生了关系,有了这个关系,才有了多态的特性。
Java语言中,Java只支持单继承,不支持多继承。
因为多继承容易带来安全隐患。当多个类中定义了相同的功能,当功能内容不同时,子类对象不确定要运行哪一个父类的功能。
但Java保留了这种机制 ,并且用一种体现形式来完成表示,多实现。
/* class A { void show(); } class B { void show(); } class C extends A,B { } C c = new C(); c.show(); //如果上面继承关系成立,这是c.show()不知道运行哪个父类的show()方法。 */
Java支持多层继承。
/* class A { void show(); } class B extends A { void show(); } clsss C extends B { } B b = new B(); b.show(); C c = new C(); c.show(); */
2.子父类出现后,类成员的特点。
2.1 变量
如果子类中出现非私有的同名变量时,子类访问本类中的变量用this。子类要访问父类中的同名变量,用super。
class Fu
{
int num = 6;
}
class Zi extends Fu
{
int num = 42;
void show()
{
System.out.println(super.num);
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
Zi z = new Zi();
System.out.println("z.num");
z.show();
}
}
2.2 子父类中的函数
当子类出现和父类一模一样的函数时,子类对象调用该函数,会调用子类函数的内容。这就是函数的另一个特性,重写(覆盖)。
当子类继承父类,就沿袭了父类的功能。当子类功能内容和父类功能不一致时,子类不必定义新功能,使用覆盖特性,保留父类的功能定义,并重写功能。
继承时的覆盖:
1.子类覆盖父类,必须保证子类的权限大于等于父类权限,才可以覆盖,否则编译失败。
2.静态只能覆盖静态
class Fu
{
void show()
{
System.out.println("fu show");
}
}
class Zi extends Fu
{
void show()
{
System.out.println("zi show");
}
}
class ExtendsDemo
{
public static void main(String[] args)
{
Zi z = new Zi();
z.show();
}
}
2.3 子父类中的构造函数
在子类对象进行初始化时,子类的构造函数默认第一行有一条隐式语句super(); 所以父类的空参数构造函数也会运行。并且子类所有的构造函数第一行默认都是super();
super():会访问父类中的空参数构造函数。
为什么子类一定要访问父类中的构造函数
因为父类中的数据子类可以直接获取,子类对象在建立时,需要先查看父类是如何对这些数据进行初始化的。
如果需要访问父类中指定的构造函数,可以通过手动定义super()语句的方式来指定。
当父类中没有空参数的构造函数时,子类必须通过手动super()语句来指定访问父类中的构造函数。
当然,子类的构造函数也可以通过手动指定 this(); 语句访问本类构造函数
class Fu
{
Fu()
{
System.out.println("fu run");
}
}
class Zi extends Fu()
{
Zi()
{
//super();
System.out.println("zi run");
}
Zi(String s)
{
//super();
System.out.println("zi run :"+s);
}
class ExtendsDemo
{
public static void main(String args)
{
Zi z = new Zi();
Zi z1 = new Zi();
}
}
3.final 关键字
3.1 final 可以修饰类、函数、变量。
3.2 final修饰的类不可以被继承。
/* final class A { void show(); } class B extends A //编译错误,不可继承 { } */
3.3 final修饰的函数不可以被覆盖。
/* class A { final void show(); } class B extends A { void show();//编译错误,不可覆盖 } */
3.4 final修饰的变量是一个常量,只能被赋值一次。
/* class A { final double PI = 3.1415926; void show() { PI = 3.14; //编译错误,只能被赋值一次。 } } */
3.5 内部类只能访问被final修饰的局部变量。
4.抽象类 abstract
1.含有抽象方法的类都是抽象类
2.抽象类不能用new创建对象,因为调用抽象方法没意义。
3.抽象类中的方法被使用,必须由子类复写所有的抽象方法后,建立子类对象调用。
抽象类和一般类的区别
抽象类练习
假如我们在开发一个系统时需要对员工进行建模,员工包含3个属性(姓名、工号、工资)。
经理也是员工,除了含有员工的属性外,另外还有一个奖金属性。请使用继承的思想设计
出员工类和经理类。要求类中提供必要的方法进行属性访问。
class Employee
{
private String name = "";
private String employeeId = "";
private double pay = -1;
Employee(String name, String employeeId,double pay)
{
this.name = name;
this.employeeId = emoloyeeId;
this.pay = pay;
}
public abstract void work();
}
class ManagerEmployee extends Employee
{
private double bonus = -1;
ManagerEmployee(String name, String employeeId,double pay,double bonus)
{
super(name,emoloyeeId,pay);
this.bonus = bonus;
}
public void work()
{
System.out.printle("Do Manager Work");
}
}
class ProfessionalEmployee extends Employee
{
ProfessionalEmployee(String name, String employeeId,double pay)
{
super(name,emoloyeeId,pay);
}
public void work()
{
System.out.printle("Do Professional Work");
}
}
/**
模板方法设计模式
在定义功能室,功能的一部分是确定的,但是有一部分是不确定的。
而确定的部分在使用不确定的部分。解决这个问题可以使用模块方法
设计模式
将不确定的部分暴露出去,由该类的子类去完成
*/
abstract class GetTime
{
public final void getTime()
{
long statTiem = System.currentTimeMillis();
runcode();
long endTiem = System.currentTimeMillis();
System.out.println("该程序使用了"+(endTiem-statTiem)+"毫秒");
}
public abstract void runcode();
}
class SubTiem extends GetTime
{
public vodi runcode()
{
for (int x=0;x<10000 ;x++ )
{
System.out.print(X);
}
}
}
5.接口
接口:可以认为接口是特殊的抽象类。当抽象类中方法都是抽象的,那个可以通过接口的形式来完成。
接口的出现将"多继承"通过另外一种新式体现出来。
class 用于定义类
interface 用于定义接口
implements 用于引用接口
extends 用于继承父类
接口的定义是,格式特点
1.接口中常见定义:常量,抽象方法。
2.接口中的成员都有固定修饰符
常量:public static final
方法: public abstract
接口中的成员都是public的,因为接口就是对外暴露原则的。
接口不可被创建对象,因为有抽象方法。
需要被子类实现后才可以实例化。
接口可以被多个实现
class A extends B implements InterA,InterB
{
}
接口可以被接口继承
interface InterA
{
void methodA();
}
interface InterB extends InterB
{
void methodA();
}
abstract class Student
{
abstract void study();
void sleep()
{
System.out.println("sleep");
}
}
interface Smoking
{
void smoke();
}
class ZhangSan extends Student implements Smoking
{
void study(){}
public void smoke(){}
}
class Lisi extends Student
{
void study(){}
}
//张三抽烟,李四不抽烟,他们都是学生
6.多态:可以理解为事物存在的多种形态
6.1多态的体现:
父类的引用指向了自己的子类对象
父类的引用也可以接收自己的子类对象
6.2多态的前提:
类于类之间有关系,要么继承,要么实现
还有一个前提:存在覆盖关系
6.3多态的好处:
提高程序扩展性
6.4多态的弊端:
只能使用父类的引用访问父类的成员
6.5多态的转型
Animal a = new Animal();
Cat c = (Cat)a; //错误
Animal a1 = new Cat(); //这是向上转型
Cat a2 = (Cat)a1; //可行 , 这是向下转型
6.6多态中成员函数的特点:
在编译时期:参阅引用类型变量所属的类是否含有调用的方法,如果有,编译通过,如果没有就编译失败
在运行时期:参阅对象所属的类中是否有调用的方法
总的来说:就是成员函数在多态调用时,编译看左边,运行看右边。
6.7多态中成员变量的特点
无论编译和运行,都参考左边
6.8在多态中,静态成员函数的特点:
无论编译和运行,都参考左边
7.内部类
class Inner
{
}
内部类的访问规则:
1、内部类可以直接访问外部类中的成员,包括私有.
原因是内部类中持有外部类的引用: 外部类名.this
2、外部类要访问内部类,必须建立对象
访问格式:
1.当内部类定义在外部类的成员位置上,而且非私有化时
可以再外部其他类中直接建立内部类对象
格式如下:
外部类名.内部类名 变量名 = 外部类对象.内部类对象
Outer.Inner in = new Outer().new Inner();
2.当内部类在成员位置上,就可以被成员修饰符所修饰。
比如:private,将内部类在外部类中进行封装
static ,内部类具备静态属性,只能直接访问外部类中的静态成员。出现了访问局限
在外部其他类中,直接访问static内部类的非静态成员格式如下
new Outer.Inner().function();
在外部其他类中,直接访问static内部类的静态成员格式如下
Outer.Inner().function();
注意:当内部类中定义了静态成员,该内部类必须是static的
当外部类中的静态方法访问内部类时,内部类也必须是static的
当描述事物时,事物的内部还有事物,该事物用内部类来描述,
因为内部事务在使用外部事物对的内容。
内部类定义在局部时
1.不可以被成员修饰符修饰
2.可以直接访问外部类中的成员,因为还有特有的外部中的引用。
但是不额可以王文它所在的局部中的变量,只能访问被final修饰的局部变量。