大家都知道,Java是一门面向对象的设计语言,只有存在类,才能体现面向对象编程这一特征。那么,究竟什么是类呢?
1.从概念上区分:类、对象、属性
类:①通过对多个相同类型的对象进行分析,把对象抽象成类,也就是说具有相同特性和行为的对象的抽象就是类;
②类是一种引用数据类型;
③并不是说所有的类都具有属性与方法;
④类可以创建多个对象。
对象:①对象是类的示例,类是对象的模板;
②要想使用类中定义的方法或属性,首先要声明类的对象。当然,被static修饰的可以直接用类名来调用;
③通过类的对象调用类的属性与方法;
④类的对象具有类中定义的所有方法与属性,对象有可以成为实例;
⑤对象的属性的初始值与类的属性的默认值是一致的。
属性:①用于区分对象,它是对象具有的特征;
②类的属性又成为实例变量;
③类的属性在没有赋值的情况下存在默认值。
1)对于 char、short、byte、int、long、float、double等基本数据类型的变量来说会默认初始化为0(float、double默认初始化为0.0,boolean变量默认会被初始化为false);
2)对于引用类型的变量(String类型、数组、自定义的类等),会默认初始化为null。
关于构造方法
如果在类的内部没有显示定义一个构造方法,则编译器会在运行时自动创建一个无参(参数列表为空、方法体为空)的构造方法。但是,如果显示定义了构造方法,编译器就不会自动添加构造方法。并且,所有的构造方法默认为static的。
构造方法存在的意义
创建类的对象并初始化成员变量。构造方法唯一的调用方式就是在创建类的对象的时候,当前是通过new关键字来创建对象。
下面具体说明一下类的初始化对象:
在程序执行时,需要生产某个类的对象,JVM首先会先检查是否加已经加载了这个类。如果已经加载,则直接生成类的对象;如果尚未加载,则会先执行类的加载再去生成对象。
在类的加载过程中,类的static成员变量会被初始化。另外,如果类中有static语句块,则会执行static语句块,并且static语句块在第一次使用类之前自动执行而且只执行一次。
static成员变量和static语句块的执行顺序同代码中的顺序一致。记住,在Java中,类是按需加载,只有当需要用到这个类的时候,才会加载这个类,并且只会加载一次。
public class Test {
public static void main(String[] args) throws ClassNotFoundException {
Bread bread1 = new Bread();
Bread bread2 = new Bread();
}
}
class Bread {
static{
System.out.println("Bread is loaded");
}
public Bread() {
System.out.println("bread");
}
}
执行结果:
Bread is loaded
bread
bread
"Bread is loaded"只会被打印一次。
在生成对象的过程中,会先初始化对象的成员变量,然后再执行构造器。也就是说类中的变量会在任何方法(包括构造器)调用之前得到初始化,即使变量散步于方法定义之间。
public class Test {
public static void main(String[] args) {
new Meal();
}
}
class Meal {
public Meal() {
System.out.println("meal");
}
Bread bread = new Bread();
}
class Bread {
public Bread() {
System.out.println("bread");
}
}
执行结果:
bread
meal
2.有关继承
特征:①继承就是在已存在的类的基础上扩展产生新的类;
②继承的目的是实现代码的复用;
③继承可以多层继承,但继承只能有一个直接父类;
④继承具有传递性与单根性;
⑤一旦实现继承,那么子类具有父类与间接父类的属性和方法;
⑥使用extends关键字实现继承;
⑦子类可以使用父类的属性和方法,但父类不能使用子类的属性和方法;
⑧在Java中如果没有使用extends关键字来显示声明一个类的父类时,那么当前类的父类时java.lang.Object;
⑨子类可以根据需要重写父类中的方法,重写要求方法签名(访问修饰符、返回类型、方法名、参数列表)相同;
⑩子类中通过super关键字调用父类的属性与方法;
创建子类对象时会优先创建父类对象,默认调用无参无方法体的构造方法。
如果父类没有无参数的构造方法,那么子类构造方法中必须指明调用父类哪个构造方法来创建父类的对象。
通过super(参数列表)的形式在子类构造方法中调用父类的构造方法,该代码必须是子类构造方法的第一行。
举个例子:
class Person {
public Person() {
}
}
class Man extends Person {
public Man() {
}
}
类Man继承于Person类,这样一来的话,Person类称为父类(基类),Man类称为子类(导出类)。如果两个类存在继承关系,则子类会自动继承父类的方法和变量,在子类中可以调用父类的方法和变量。在java中,只允许单继承,也就是说 一个类最多只能显示地继承于一个父类。但是一个类却可以被多个类继承,也就是说一个类可以拥有多个子类。
1.子类继承父类的成员变量
当子类继承了某个类之后,便可以使用父类中的成员变量,但是并不是完全继承父类的所有成员变量。具体的原则如下:
1)能够继承父类的public和protected成员变量;不能够继承父类的private成员变量;
2)对于父类的包访问权限成员变量,如果子类和父类在同一个包下,则子类能够继承;否则,子类不能够继承;
3)对于子类可以继承的父类成员变量,如果在子类中出现了同名称的成员变量,则会发生隐藏现象,即子类的成员变量会屏蔽掉父类的同名成员变量。如果要在子类中访问父类中同名成员变量,需要使用super关键字来进行引用。
2.子类继承父类的方法
同样地,子类也并不是完全继承父类的所有方法。
1)能够继承父类的public和protected成员方法;不能够继承父类的private成员方法;
2)对于父类的包访问权限成员方法,如果子类和父类在同一个包下,则子类能够继承;否则,子类不能够继承;
3)对于子类可以继承的父类成员方法,如果在子类中出现了同名称的成员方法,则称为覆盖,即子类的成员方法会覆盖掉父类的同名成员方法。如果要在子类中访问父类中同名成员方法,需要使用super关键字来进行引用。
注意:隐藏和覆盖是不同的。隐藏是针对成员变量和静态方法的,而覆盖是针对普通方法的。
3.构造器
子类是不能够继承父类的构造器,但是要注意的是,如果父类的构造器都是带有参数的,则必须在子类的构造器中显示地通过super关键字调用父类的构造器并配以适当的参数列表。如果父类有无参构造器,则在子类的构造器中用super关键字调用父类构造器不是必须的,如果没有使用super关键字,系统会自动调用父类的无参构造器。看下面这个例子就清楚了:
class Shape {
protected String name;
public Shape(){
name = "shape";
}
public Shape(String name) {
this.name = name;
}
}
class Circle extends Shape {
private double radius;
public Circle() {
radius = 0;
}
public Circle(double radius) {
this.radius = radius;
}
public Circle(double radius,String name) {
this.radius = radius;
this.name = name;
}
}
这样的代码是没有问题的,如果把父类的无参构造器去掉,则下面的代码必然会出错:
改成下面这样就行了:
4.super关键字
super主要有两种用法:
1)super.成员变量/super.成员方法;
2)super(parameter1,parameter2....)
第一种用法主要用来在子类中调用父类的同名成员变量或者方法;第二种主要用在子类的构造器中显示地调用父类的构造器,要注意的是,如果是用在子类构造器中,则必须是子类构造器的第一个语句。