博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
继承,多态,抽象,接口
阅读量:6996 次
发布时间:2019-06-27

本文共 9648 字,大约阅读时间需要 32 分钟。

-- 继承:

1、子类继承父类中所有属性和方法。
2、子类只能重写父类中实例方法。
3、super关键,如果放到子类构造函数第一行,如果出现子类方法中可通过该关键字调用子类覆写的方法
4、使用final关键字修饰的方法不能重写,类不能被继承。
5、static关键字修改的方法只属于所在类,不属于类的实例,但是所有实例共享静态成员,static修饰的方法中必须调用静态的方法和使用静态变量。
在static中不能使用this和super关键字。

1、静态代码块

  ①、格式

  在java类中(方法中不能存在静态代码块)使用static关键字和{}声明的代码块:

public class CodeBlock {    static{        System.out.println("静态代码块");    }}

  ②、执行时机

  静态代码块在类被加载的时候就运行了,而且只运行一次,并且优先于各种代码块以及构造函数。如果一个类中有多个静态代码块,会按照书写顺序依次执行。后面在比较的时候会通过具体实例来证明。

  ③、静态代码块的作用

  一般情况下,如果有些代码需要在项目启动的时候就执行,这时候就需要静态代码块。比如一个项目启动需要加载的很多配置文件等资源,我们就可以都放入静态代码块中。

  ④、静态代码块不能存在任何方法体中

  这个应该很好理解,首先我们要明确静态代码块是在类加载的时候就要运行了。我们分情况讨论:

  对于普通方法,由于普通方法是通过加载类,然后new出实例化对象,通过对象才能运行这个方法,而静态代码块只需要加载类之后就能运行了。

  对于静态方法,在类加载的时候,静态方法也已经加载了,但是我们必须要通过类名或者对象名才能访问,也就是说相比于静态代码块,静态代码块是主动运行的,而静态方法是被动运行的。

  不管是哪种方法,我们需要明确静态代码块的存在在类加载的时候就自动运行了,而放在不管是普通方法还是静态方法中,都是不能自动运行的。

  ⑤、静态代码块不能访问普通变量

  这个理解思维同上,普通变量只能通过对象来调用,是不能放在静态代码块中的。

2、构造代码块

  ①、格式

  在java类中使用{}声明的代码块(和静态代码块的区别是少了static关键字):

public class CodeBlock {    static{        System.out.println("静态代码块");    }    {        System.out.println("构造代码块");    }}

  ②、执行时机

  构造代码块在创建对象时被调用,每次创建对象都会调用一次,但是优先于构造函数执行。需要注意的是,听名字我们就知道,构造代码块不是优先于构造函数执行,而是依托于构造函数,也就是说,如果你不实例化对象,构造代码块是不会执行的。怎么理解呢?我们看看下面这段代码:

public class CodeBlock {    {        System.out.println("构造代码块");    }         public CodeBlock(){        System.out.println("无参构造函数");    }    public CodeBlock(String str){        System.out.println("有参构造函数");    }}

  我们反编译生成的class文件:

  

  如果存在多个构造代码块,则执行顺序按照书写顺序依次执行。

  ③、构造代码块的作用

   和构造函数的作用类似,都能对对象进行初始化,并且只要创建一个对象,构造代码块都会执行一次。但是反过来,构造函数则不一定每个对象建立时都执行(多个构造函数情况下,建立对象时传入的参数不同则初始化使用对应的构造函数)。

  利用每次创建对象的时候都会提前调用一次构造代码块特性,我们可以做诸如统计创建对象的次数等功能。

3、构造函数 

  1.构造函数的命名必须和类名完全相同。在java中普通函数可以和构造函数同名,但是必须带有返回值;

  2.构造函数的功能主要用于在类的对象创建时定义初始化的状态。它没有返回值,也不能用void来修饰。这就保证了它不仅什么也不用自动返回,而且根本不能有任何选择。而其他方法都有返回值,即使是void返回值。尽管方法体本身不会自动返回什么,但仍然可以让它返回一些东西,而这些东西可能是不安全的;

  3.构造函数不能被直接调用,必须通过new运算符在创建对象时才会自动调用;而一般的方法是在程序执行到它的时候被调用的;

  4.当定义一个类的时候,通常情况下都会显示该类的构造函数,并在函数中指定初始化的工作也可省略,不过Java编译器会提供一个默认的构造函数.此默认构造函数是不带参数的。而一般的方法不存在这一特点;

4、普通代码块

  普通代码块和构造代码块的区别是,构造代码块是在类中定义的,而普通代码块是在方法体中定义的。且普通代码块的执行顺序和书写顺序一致。

public void sayHello(){    {        System.out.println("普通代码块");    }}

5、执行顺序

静态代码块>构造代码块>构造函数>普通代码块

public class CodeBlock {
static{ System.out.println("静态代码块"); } { System.out.println("构造代码块"); } public CodeBlock(){ System.out.println("无参构造函数"); } public void sayHello(){ { System.out.println("普通代码块"); } } public static void main(String[] args) { System.out.println("执行了main方法"); new CodeBlock().sayHello();; System.out.println("---------------"); new CodeBlock().sayHello();; }}

反编译生成的class文件:

 

执行结果:

创建了两个匿名对象,但是静态代码块只是调用了一次。 

6、父类和子类执行顺序

  对象的初始化顺序:

  首先执行父类静态的内容,父类静态的内容执行完毕后,接着去执行子类的静态的内容,当子类的静态内容执行完毕之后,再去看父类有没有构造代码块,如果有就执行父类的构造代码块,父类的构造代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有构造代码块,如果有就执行子类的构造代码块。子类的构造代码块执行完毕再去执行子类的构造方法。

  总之一句话,静态代码块内容先执行,接着执行父类构造代码块和构造方法,然后执行子类构造代码块和构造方法。

  父类:SuperClass.java

package com.ys.extend; public class SuperClass {    static{        System.out.println("父类静态代码块");    }    {        System.out.println("父类构造代码块");    }    public SuperClass(){        System.out.println("父类构造函数");    }}

  子类:SubClass.java

package com.ys.extend; public class SubClass extends SuperClass{    static{        System.out.println("子类静态代码块");    }         {        System.out.println("子类构造代码块");    }         public SubClass(){        System.out.println("子类构造函数");    }     }

  测试:

public static void main(String[] args) {    SubClass sb = new SubClass();    System.out.println("------------");    SubClass sb1 = new SubClass();}

  打印结果:

  

-- 多态: 就是一个事物多种表现形态

表现形式:
重写(子类中重写父类中的方法)
重载(在同一个类中定义参数类型和参数个数不同且方法名称相同的方法就是重载,它与返回值无关)。

重写: package com.extend.demo;public class Student {    public void study() {        System.out.println("学习爱学习");    }}------------------------------------------------package com.extend.demo;public class LittleStudent extends Student {    @Override    public void study() {        System.out.println("小学生正在学习.....");    }}---------------------------------------------------package com.extend.demo;public class MiddleStudent extends Student {    @Override    public void study() {        System.out.println("中学生正在学习......");    }}----------------------------------------------------------package com.extend.demo;public class TestStudent {    public static void main(String[] args) {        LittleStudent little = new LittleStudent();        MiddleStudent middle = new MiddleStudent();        little.study();        middle.study();        System.out.println("--------------");        Student little1 = new LittleStudent();        Student middle1 = new MiddleStudent();        study(little1);        study(middle1);    }        public static void study(Student stu) {        stu.study();    }} -------------------------------------------------- 执行结果:

  小学生正在学习.....

  中学生正在学习......
  --------------
  小学生正在学习.....
  中学生正在学习......

重载: package com.overload.demo;public class Operator {    public int sum(int num1, int num2) {        return num1+num2;    }        public float sum(float num1, float num2) {        return num1+num2;    }        public float sum(int num1, float num2) {        return num1+num2;    }    }

-- 抽象就是一种理论,思想,不能用具体语言描述的事物。

UFO-> 使用什么驱动-> 能飞,形状: 椭圆形。
抽象类: 使用abstract关键字修饰的类,就是抽象类。
语法: public abstract class 抽象类名称 {}
总结:
1、抽象类不能被实例化
2、抽象类中可以定义抽象方法
3、抽象方法不能使用private修饰
4、抽象类可以继承抽象类并且可以不实现父类中抽象方法

package com.abstracts.demo;/** * 1、抽象类不能被实例化 * 2、抽象类中可以定义抽象方法 * 3、抽象方法不能使用private修饰 * 注意: 抽象的方法决定类必须是抽象类 * @author Administrator * */public abstract class AbstractUFO {        protected String name = "zhangssna";        public AbstractUFO() {        System.out.println("========AbstractUFO=======");    }        public void fly() {        System.out.println("UFO能飞");    }        // 抽象方法    public abstract void engine();}-------------------------------------------------------package com.abstracts.demo;public class UFO extends AbstractUFO {    /**     * 必须实现抽象类中所定义方法     */    public void engine() {        System.out.println("UFO使用太阳能驱动"+this.name);    }        public static void main(String[] args) {        UFO ufo = new UFO();        ufo.fly();        ufo.engine();    }}-----------------------------------------------------------package com.abstracts.demo;/** * 抽象类可以继承抽象类并且可以不实现父类中抽象方法 * @author Administrator * */public abstract class Abstract1 extends AbstractUFO {}

 

抽象类: 模板设计模式

package com.abstracts.demo.template;/** * 模板设计模式 * @author Administrator * */public abstract class JDBCTemplate {        /**     * 连接数据库的方法     */    public void connectionDB() {        loadDriver();        getConn();        execSql();    }        // 加载驱动    public abstract void loadDriver();        // 获取连接    public abstract void getConn();        // 执行SQL指令    public abstract void execSql();    }---------------------------------------------------------package com.abstracts.demo.template;public class MySQLConnectionDB extends JDBCTemplate {    @Override    public void loadDriver() {        System.out.println("加载mysql驱动");    }    @Override    public void getConn() {        System.out.println("mysql连接");    }    @Override    public void execSql() {        System.out.println("执行mysql的sql语句");    }}------------------------------------------------------package com.abstracts.demo.template;public class OracleConnectionDB extends JDBCTemplate {    @Override    public void loadDriver() {        System.out.println("加载oracle驱动");    }    @Override    public void getConn() {        System.out.println("oracle连接");    }    @Override    public void execSql() {        System.out.println("执行oracle的sql语句");    }}------------------------------------------------------------package com.abstracts.demo.template;public class TestTemplate {    public static void main(String[] args) {        // 调用方法        ConnectDB(new MySQLConnectionDB());        System.out.println("----------------");        ConnectDB(new OracleConnectionDB());    }        public static void ConnectDB(JDBCTemplate template) {        // 调用模板中的方法        template.connectionDB();    }}

-- 接口: 使用interface关键字修饰的类.

* 1、使用interface关键字修饰的类
* 2、接口是特殊的抽象类,不能有构造函数,不能成员变量
* 3、接口中所有成员,常量,方法都是public
* 4、接口中定义声明都是抽象的。
* 5、接口必须让其实现类使用implements关键字实现接口中所有定义抽象声明
* 6、jdk1.8版本及以上能够哎接口中定义使用static和default关键定义实现方法
* 作用: 限定其实现类必须具有的方法

package com.interfaces;/** * 1、使用interface关键字修饰的类 * 2、接口是特殊的抽象类,不能有构造函数,不能成员变量 * 3、接口中所有成员,常量,方法都是public * 4、接口中定义声明都是抽象的。 * 5、接口必须让其实现类使用implements关键字实现接口中所有定义抽象声明 * 6、jdk1.8版本及以上能够哎接口中定义使用static和default关键定义实现方法 * 作用: 限定其实现类必须具有的方法 * @author Administrator * */public interface IUserDAO {    // 接口定义的变量都是静态的常量    public static final String NAME = "zhangsan";        public abstract void delete();    void update();    void add();        /**     * 接口中可以定义静态方法     */    static void say1() {        System.out.println("定义静态方法");    }        /**     * jdk1.8版本及以上     */    default public void say() {        System.out.println("接口中默认实现的方法");    }}----------------------------------------------------------------------package com.interfaces;public class UserDAOImpl implements IUserDAO {    @Override    public void delete() {        // TODO Auto-generated method stub    }    @Override    public void update() {        // TODO Auto-generated method stub    }    @Override    public void add() {        // TODO Auto-generated method stub    }        public static void main(String[] args) {        IUserDAO.say1();        new UserDAOImpl().say();        String string = IUserDAO.NAME;        System.out.println(string);    }}

 

转载于:https://www.cnblogs.com/sunBinary/p/10463435.html

你可能感兴趣的文章