首页 > 代码库 > Java学习-第三天

Java学习-第三天

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修饰的局部变量。

Java学习-第三天