Java中有三大特性,分别是封装继承多态,其理念十分抽象,并且是层层深入式的.
一.封装
概念:封装,即隐藏对象的属性和实现细节,仅对外公开接口,控制在程序中属性的读和修改的访问级别;将抽象得到的数据和行为(或功能)相结合,形成一个有机的整体,也就是将数据与操作数据的源代码进行有机的结合,形成“类”,其中数据和函数都是类的成员。在电子方面,封装是指把硅片上的电路管脚,用导线接引到外部接头处,以便与其它器件连接。(来自百度)
在将成员属性封装之后,我们应该提供相应的get/set方法进行访问此成员属性.
封装的好处是:属性进行了隐藏,不再是暴露在外的数据,使数据更加的安全可靠.
1 class A{
2 private int a;//此变量进行了封装
3 //提供给外部访问的接口,实现了只读和只写
4 public int getA(){
5 return a;
6 }
7 public void setA(int a){
8 this.a=a;
9 }
10 }
提到封装就必须要提到private关键字,如上代码,private是一个权限修饰符,它可以将成员方法和成员属性私有化,使其只有类内部能够访问到.
权限修饰符有4种,分别是public->protected->default->private 其中private权限最严格
public:可以在全工程访问到 protected:只有这个类的子类可以访问 defaule:为默认,无须写出,类所在包下可以访问 private:仅仅只有本省类可以访问
提到了封装就不得不提到this关键字
概念:this关键字指这个当前对象的引用
this关键字有三种用法:
1.this.成员方法,这种方法用于给成员变量的赋值.
1 class A{
2 private int a;//此变量进行了封装
3 //提供给外部访问的接口,实现了只读和只写
4 public int getA(){
5 return a;
6 }
7 public void setA(int a){
8 this.a=a;
9 }
10 }
因为就近原则,所以直接a=a;实际上并没有对成员属性进行赋值,所以需要使用this关键字对其进行访问.
2.调用构造方法,如:可以在空参构造方法中调用有参构造方法
1 class A{
2 private int a;//此变量进行了封装
3 //提供给外部访问的接口,实现了只读和只写
4 public int getA(){
5 return a;
6 }
7 public void setA(int a){
8 this.a=a;
9 }
10 //空参构造
11 public A(){
12 this(1);
13 }
14 //全参构造
15 public A(int a){
16 this.a=a;
17 }
18 }
这个例子使用this在空参构造中调用了全参构造
使用this调用构造方法时需要注意:
1.this()必须在构造方法中的第一行,且只能够使用一次
2.this()不能够互相调用,因为会陷入死循环,如下:
1 class A{
2 private int a;//此变量进行了封装
3 //提供给外部访问的接口,实现了只读和只写
4 public int getA(){
5 return a;
6 }
7 public void setA(int a){
8 this.a=a;
9 }
10 //空参构造
11 public A(){
12 this(1);
13 //this();错误
14 }
15 //全参构造
16 public A(int a){
17 //this();错误
18 this.a=a;
19 }
20 }
3.this可以当作返回值使用,this是当前对象的引用,所以返回的同样是一个A类型,可以使用一个新的A类型对象b进行接受,b会继承所有a的属性值,使用方法如下:
class A{
private int a;//此变量进行了封装
//提供给外部访问的接口,实现了只读和只写
public int getA(){
return a;
}
public void setA(int a){
this.a=a;
}
//使用this当作返回值
A b(){
return this;
}
//空参构造
public A(){
this(1);
}
//全参构造
public A(int a){
this.a=a;
}
}
二.继承
概念:继承即从已有的类中派生一个新的类,已有的类成为父类(基类,超类),而派生的类成为子类(派生类).若类B继承了类A,则我们把类A称为类B的子类,反之同理,子类可以拥有自己本身的特有方法以及特有属性,而父类则必须是子类的共性抽取.如动物都有颜色,年龄,其颜色和年龄就是其共性
1 class A{
2 int a;
3 void a(){}
4 }
5 class B extends A{
6 //子类特有方法
7 void b(){
8 }
9 }
1. 继承需要注意的点
(1)继承是类在继承,而不是对象在继承
(2)子类无条件继承父类的所有成员属性及成员方法
(3)若方法出现重写,属性出现重复,则优先使用子类的方法和属性(就近原则).
(4)继承不继承父类的构造方法
2.java中继承的特点
(1).java不支持多继承,但是可以通过接口实现多继承 多继承,即子类继承自多个父类
(2).java支持多重继承 多重继承,即子类继承父类,父类继承他的父类,即子类相当于其孙子辈
class A{
int a;
public A(){
//super();这里有一个默认的super()方法
}
public A(int a){
//super();这里有一个默认的super()方法
}
void a(){}
}
class B extends A{
//子类的特有属性
int b;
//子类特有方法
void b(){
//调用父类的成员属性
System.out.println(super.a);
//调用父类的成员方法
super.a();
}
public B(){
//调用父类的有参构造
super(1);
}
}
提到继承就必须提到super关键字,super关键字指父类的引用,super关键字最重要的是每个类的构造方法都有其默认的一个super()方法,我们知道所有的类都是继承自Object类,而super()这个方法其实是调用到Object处进行空间的申请.
super关键字的使用
1.访问父类成员方法和成员属性 super.成员方法() super.成员属性
2.访问父类的构造方法 super(参数)
1 abstract class Animal{
2 private String color;
3 private int numOfLegs;
4 abstract void eat();
5 public String getColor() {
6 return color;
7 }
8 public void setColor(String color) {
9 this.color = color;
10 }
11 public int getNumOfLegs() {
12 return numOfLegs;
13 }
14 public void setNumOfLegs(int numOfLegs) {
15 this.numOfLegs = numOfLegs;
16 }
17 public Animal(String color, int numOfLegs) {
18 super();
19 this.color = color;
20 this.numOfLegs = numOfLegs;
21 }
22 public Animal() {
23 super();
24 }
25
26 }
27 class Dog extends Animal{
28 Dog(){
29 super();
30 }
31 Dog(String color,int numOfLegs){
32 super(color,numOfLegs);
33 }
34 @Override
35 void eat(){
36 System.out.println(super.getNumOfLegs()+"条腿"+super.getColor()+"的狗在啃骨头");
37 }
38 void lookHome(){
39 System.out.println(super.getNumOfLegs()+"条腿"+super.getColor()+"的狗在看家");
40 }
41 }
42 class Porrot extends Animal{
43 Porrot(){
44 super();
45 }
46 Porrot(String color,int numOfLegs){
47 super(color,numOfLegs);
48 }
49 @Override
50 void eat(){
51 System.out.println(super.getNumOfLegs()+"条腿的"+super.getColor()+"鹦鹉在吃小米");
52 }
53 void say(){
54 System.out.println(super.getNumOfLegs()+"条腿的"+super.getColor()+"鹦鹉在说你好,丑八怪");
55 }
56 }
提到继承当然也不能不提到抽象类,定义一个抽象类使用是这样的:public abstract class Animal(){}
当你想要创建一个动物类时,你知道动物都会吃,但是你不知道动物的子类吃是如何具体实现的,如:猫吃鱼,但是狗吃的是骨头同样的子类,但是其方法的实现却是不同的,这个时候就需要使用抽象类了.
抽象类的注意点:
1.抽象类必须在类中使用abstract关键字进行修饰,
2.抽象类中可以没有抽象方法,
3.抽象类不能够实例化,如果想要实例化的话则需要子类将其所有的抽象方法重写
1 interface A{
2 int b=0;//默认有public static final;
3 void a(){}//默认有public abstract;
4 }
5 interface B{
6 }
7 class Father{
8 }
9 class Son extends Father implements A,B{
10 //方法的实现
11 public void a(){
12 }
13 }
提到抽象类就一定要提到接口
接口:接口其实相当于一个规范,当你定义一个接口时,实现接口的类必须实现接口中的所有的方法.举个现实中的例子,比如我们的手机充电借口,基本很少发生改变,因为他已经是规定死了的,使用接口等于制定一个规范,等于是所有的实现此接口的类都必须实现此规范,这样就可以用接口来实现不同类的比较等操作.
接口的注意点:
1.接口中的成员属性必须都是public static final的,不修饰会自动加上,接口中的成员方法必须都是public static修饰的,且不可以使用其他的权限修饰符
2.接口不可以被实例化,必须使用子类 explements 接口,然后实现接口中的方法.
3.一个类可以实现多个接口,可以在实现抽象类的同时实现多个接口或单个接口
三.多态
概念:同一个行为,对于传入不同的对象,实现不同的动作 多态需要记住的一个概念:父类引用指向子类对象
多态的好处:提高了代码的可复用性.
多态的坏处:无法使用子类所特有的方法.因为编译时看的是右边.
多态的实现必须有三要素 1.继承(实现) 2.重写 3.向上转型(缺一不可!!!)
1 public class Test{
2 public static void main(String[] args){
3 show(new Dog());
4 show(new Cat());
5 }
6 public static void show(Animal a){
7 a.eat();
8 }
9 }
10 abstract class Animal{
11 abstract void eat(){
12 }
13 }
14 class Dog extends Animal{
15 public void eat(){
16 System.out.println("在吃骨头");
17 }
18 }
19 class Cat extends Animal{
20 public void eat(){
21 System.out.println("在吃鱼");
22 }
23 }
多态基于继承的实现如上.
public class Test01{
public static void main(String[] args){
show(new Dog());
show(new Cat());
}
public static void show(Animal a){
a.eat();
}
}
interface Animal{
abstract void eat();
}
class Dog implements Animal{
public void eat(){
System.out.println("在吃骨头");
}
}
class Cat implements Animal{
public void eat(){
System.out.println("在吃鱼");
}
}
接口的实现如上
提到多态就需要提到一个例子:
public class Test01{
public static void main(String[] args){
A a1=new A();
A a2=new B();
B b=new B();
C c=new C();
D d=new D();
a1.show(b);// a and a
a1.show(c);//a and a
a1.show(d);//a and d
a2.show(b);//b and a
a2.show(c);//b and a
a2.show(d);//a and d
}
/*public static void show(Animal a){
a.eat();
}*/
}
class A{
void show(A a){
System.out.println("a and a");
}
void show(D d){
System.out.println("a and d");
}
}
class B extends A{
void show(A a){
System.out.println("b and a");
}
void show(B b){
System.out.println("b and b");
}
}
class C extends B{
}
class D extends C{
}
我们先看a2.show(c),首先a2是一个a类型的引用类型,在a2种寻找show(c)参数为c类型的方法,没有找到c类型的方法,于是到a2的超类寻找其方法,由于a类的超类是Object,也并没有show(c)这个方法,于是将c类型进行了提升,变成了b类型,b类型在a中也不存在,于是便转换成了a类型,a类型与A类中找到了方法,并由于A类型的show方法于B类型中进行了重写,于是便输出了b and a
其寻找的顺序是:this.show(O) super.show(O) this.show(super(O)) super.show(super(O))
当使用了上述方法进行了多态的实现之后,其子类就会自动将其的特有方法给丢弃掉了,然而我们现在想调用这个对象的特有方法怎么办?
答案是使用使用强制类型转换,而使用强制类型转换则会有可能出现类型转换异常,这个时候就需要使用instanceof运算符,使用方法:父类对象 instanceof 子类,若父类引用时子类类型时返回true,则可以使用强制类型转换
这种实现过程叫做向下转型