面向对象的编程容许从已经存在的类中定义新的类,这称为继承。java
面向过程的范式重点在于方法的设计,而面向对象的范式将数据和方法结合在对象中。面向对象范式的软件设计着重于对象以及对象上的操做。面向对象的方法结合了面向过程范式的强大之处,而且进一步将数据和操做集成在对象中。编程
继承可使得你能够定义一个通用的类(即父类),以后扩充该类为一个更加特定的类(即子类)。在Java术语中,若是类C1扩展自另外一个类C2,那么就将C1称为次类,将C2称为超类。 超类也称为父类或基类,次类又称为子类、拓展类或派生类。数组
子类从它的父类中继承可访问的数据域和方法,还能够添加新数据域和新方法。安全
若是父类中有私有的数据域,这个类被继承后,父类中的私有数据域是不能被子类访问的,惟一能读取和改变它们的方法就是经过它们的get和set方法。测试
和传统的理解不一样,子类并非父类的一个子集。实际上,一个子类一般比它的父类包含更多的信息和方法。ui
父类中的私有数据域在该类以外是不可访问的。所以,不能在子类中直接使用。可是,若是父类中定义了公共的访问器/修改器方法,那么能够经过这些公共的访问器/修改器来访问和修改它们。设计
不是全部的“是一种”关系都该用继承来建模。继承是用来为“是一种”关系建模的。不要仅仅为了重用方法这个缘由而盲目地扩展一个类。对象
某些程序设计语言是容许从几个类派生出一个子类的。这种能力成为多重继承。可是在Java中是不容许多重继承的。一个Java类只可能直接继承自一个父类。这种限制称为单一继承。若是使用extends关键字来定义一个子类,它只容许有一个父类。然而,多重继承是能够经过接口来实现的。继承
关键字super指代父类,能够用于调用父类中的普通方法和构造方法。接口
构造方法用于构建一个类的实例。不一样于属性和普通方法,父类的构造方法不会被子类继承。它们只能使用关键字super从子类的构造方法中调用。调用父类构造方法的语法是:super(),或者super(parameters);语句super()调用父类的无参构造方法,而语句super(arguments)调用与参数匹配的父类的构造方法。语句super()和super(arguments)必须出如今子类构造方法的第一行,这是显示调用父类构造方法的惟一方式。
要调用父类的构造方法就必须使用关键字super,并且这个调用必须是构造方法的第一条语句。在子类中调用父类构造方法的名字会引发一个语法错误。
构造方法能够调用重载的构造方法或父类的构造方法。若是它们都没有被显示地调用,编译器就会自动地将super()做为构造方法的第一条语句。
子类必须使用super调用父类的构造方法,若是没有显示调用的话,就会使用super(),调用父类不带参数的构造方法,若是父类没有不带参数的构造方法,则必须显示地用带参数的super(parameters)调用父类的构造方法。
通常状况下,最好能为每一个类提供一个无参构造方法,以便于对该类进行扩展,同时避免错误。
在任何状况下,构造一个类的实例时,将会调用沿着继承链的全部父类的构造方法。当构造一个子类的对象时,子类构造方法会在完成本身的任务以前,首先调用它的父类的构造方法。若是父类继承自其余类,那么父类构造方法又会在完成本身的任务以前,调用它本身的父类的构造方法。这个过程持续到沿着这个继承体系结构的最后一个构造方法被调用为止。这就是构造方法链。
关键字super不只能够引用父类的构造方法,还能够引用父类的方法。
子类从父类中继承方法。有时,子类须要修改父类中定义的方法的实现,这称做方法重写。
方法重写:要重写一个方法,须要在子类中使用和父类同样的签名以及同样的返回值类型来对该方法进行定义。
仅当实例方法是可访问的,它才能被覆盖。由于私有方法在它自己之外是不能访问的,因此它不能被覆盖。若是子类中定义的方法在父类中是私有的,那么这两个方法彻底没有关系。
与实例方法同样,静态方法也能被继承。可是,静态方法不能被覆盖。若是父类中定义的静态方法在子类中被从新定义,那么在父类中定义的静态方法将被隐藏。可使用语法:父类名.静态方法名调用隐藏的静态方法。
重载意味着使用一样的名字可是不一样的签名来定义多个方法。重载意味着在子类中提供一个对方法的新的实现。
方法重写发生在经过继承而相关的不一样类中;方法重载能够发生在同一个类中。也能够发生在因为继承而相关的不一样类中。
方法重写具备一样的签名和返回值类型;方法重载具备相同的名字,可是不一样的参数列表。
重载方法使得可使用一样的名字来定义不一样方法,只要它们的签名是不一样的。(方法名和参数列表共同构成方法签名,这里方法名相同,则重载方法要求参数列表不一样。)
被重载的方法必须具备不一样的参数列表。不能基于不一样修饰符或返回值类型来重载方法。
Java中的全部类都继承自java.lang.Object类。
若是在定义一个类时没有指定继承性,那么这个类的父类就被默认为是Objec。
多态意味着父类的变量能够指向子类对象。
面向对象程序设计的三大支柱是封装、继承和多态。
一个类实际上定义了一种类型。子类定义的类型称为子类型,而父类定义的类型称为父类型。
继承关系使一个子类继承父类的特征,而且附加一些新特征。子类是它的父类的特殊化,每一个子类的实例都是其父类的实例,可是反过来就不成立。例如:每一个圆都是一个几何对象,但并不是每一个几何对象都是圆。
使用父类对象的地方均可以使用子类对象,这就是一般所说的多态。简单来讲,多态意味着父类型的变量能够引用子类型的对象。
动态绑定:方法能够在沿着继承链的多个类中实现,JVM决定运行时调用哪一个方法。
声明类型和实际类型:一个变量必须被声明为某种类型。变量的这个类型称为它的声明类型。一个引用类型变量能够是一个null值或者是一个对声明类型实例的引用。实例可使用声明类型或它的子类型的构造方法建立。变量的实际类型是被变量引用的对象的实际类。
动态绑定工做机制以下:假设对象o是类C1的实例,存在类C1,C2,...,Cn-1,Cn,其中C1是C2的子类,C2是C3的子类,...,Cn-1是Cn的子类,也就是说,Cn是最通用的类,C1是最特殊的类。在Java中,Cn是Object类。若是对象o调用一个方法p,那么JVM会依次在类C1,C2,...,Cn-1,Cn中查找方法p的实现,直到找到为止,一旦找到一个实现,就中止查找,而后调用这个首先找到的实现。
匹配方法的签名和绑定方法的实现是两个不一样的问题。引用变量的声明类型决定了编译时匹配哪一个方法。在编译时,编译器会根据参数类型、参数个数和参数顺序找到匹配的方法。一个方法可能在沿着继承链的多个类中实现。Java虚拟机在运行时动态绑定方法的实现,这是由变量的实际类型决定的。
对象的引用能够类型转换为对另外一种对象的引用,这称为对象转换。
Object o = new Student();
因为Student的实例也是Object的实例,因此,语句Object o=new Student()是合法的,它称为隐式转换。
假设通过上面的一步想使用下面的语句把对象引用o赋值给Student类型的变量:
Student b=o;
这种状况下,将会发生编译错误。由于Student对象老是Object的实例,可是,Object对象不必定是Student的实例。即便能够看到o其实是一个Student对象,可是编译器尚未聪明到知道这一点。为了告诉编译器o就是一个Student对象,就要使用显示转换。它的语法与基本类型转换的语法很相似,用圆括号把目标对象的类型括住,而后放到要转换的对象前面:
Student b = (Student)o;
老是能够将一个子类的实例转换为一个父类的变量,称为向上转换,由于子类的实例永远是它的父类实例。
当把一个父类的实例转换为它的子类变量(称为向下转换)时,必须使用转换记号“(子类名)”进行显示转换,向编译器代表意图。为使转换成功,必须确保要转换的对象是子类的一个实例。若是父类对象不是子类的一个实例,就会出现一个运行异常ClassCastException。例如:若是一个对象不是Student的实例,它就不能转换成Student类型的变量。所以,一个好的经验是,在尝试转换以前确保该对象是另外一个对象的实例。这是能够利用instanceof来实现的。
声明类型决定了在编译时匹配哪一个方法。
为了更好地理解类型转换,能够认为它们相似于水果、苹果、橘子之间的关系,其中水果类Fruit是苹果类Apple和橘子类Orange的父类。苹果是水果,因此,老是能够将Apple的实例安全地赋值给Fruit变量。可是,水果不必定是苹果,因此必须进行显示转换才能将Fruit的实例赋值给Apple变量。
比较运算符==用来比较两个基本数据类型的值是否相等,或者判断两个对象是否具备相同的引用。若是想让equals方法可以判断两个对象是否具备相同的内容,能够在定义这些对象时,重写equals方法。
ArrayList对象能够用于存储一个对象列表:以前能够建立一个数组存储对象,可是这个数组一旦建立,它的大小就固定了。Java提供ArrayList类来存储不限定个数的对象。ArrayList是一种泛型类,具备一个泛型类型E。建立一个ArrayList时,能够指定一个具体的类型来替换E。
ArrayList<AConcreteType> list = new ArrayList<AConcreteType>();能够简化为:
ArrayList<AConcreteType> list = new ArrayList<>();
要建立一个用于存储整数的ArrayList,因为存储在ArrayList中的元素必须是一种对象,因此不能使用诸如int的基本类型来替代一个泛型类型,可使用以下的声明方式:
ArrayList<Integer> list = new ArrayList<>();
ArrayList的大小是灵活的,因此无须提早给定它的大小。而当建立一个数组时,它的大小必须给定。
ArrayList包含许多有用的方法。好比contains方法来测试某个元素是否在列表中。
一个类中的受保护成员能够从子类中访问。
私有成员只能在类内访问,而共有成员能够被任意的其余类访问。
常常须要容许子类访问定义在父类中的数据域和方法,但不容许非子类访问这些数据域和方法。可使用关键字protected完成该功能。父类中被保护的数据域和方法能够在它的子类中访问。
使用private修饰符能够彻底隐藏类的成员,这样,就不能从类外直接访问他们。不使用修饰符就表示容许同一个包里的任何类直接访问类的成员,可是其余包中的类不能够访问。使用protected修饰符容许任何包中的子类或同一包中的类访问类的成员。使用public修饰符容许任意类访问类的成员。
类能够以两种方式使用,一种是用于建立该类的实例;另外一种是经过扩展该类建立它的子类。若是不想从类的外部使用类的成员,就把成员声明成private。若是想让该类的用户都能使用类的成员,就把成员声明成public。若是想让该类的扩展者使用数据和方法,而不想让该类的用户使用,则把成员声明成protected。
修饰符private和protected只能用于类的成员,public修饰符和默认修饰符既能够用于类的成员,也能够用于类。一个没有修饰符的类是不能被其余包中的类访问的。
子类能够重写它的父类的protected方法,并把它的可见性改成public。可是,子类不能削弱父类中定义的方法的可访问性。例如:若是一个方法在父类中定义为public,在子类中也必须定义为public。