多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。
多个类可以称为子类,单独这个类称为父类、超类或者基类。
子类可以直接访问父类中的非私有的属性和行为。
通过 extends 关键字让类与类之间产生继承关系。
1
|
class
SubDemo
extends
Demo{}
//SubDemo是子类,Demo是父类
|
1.Java只支持单继承,不支持多继承。
1
2
3
|
//一个类只能有一个父类,不可以有多个父类。
class
SubDemo
extends
Demo{}
//ok
class
SubDemo
extends
Demo1,Demo2...
//error
|
2.Java支持多层(重)继承(继承体系)。
1
2
3
|
class
A{}
class
B
extends
A{}
class
C
extends
B{}
|
super是一个关键字,代表父类的存储空间标识。(可以理解为父亲的引用)
super和this的用法相似。
this代表对象的引用(谁调用就代表谁);
super代表当前子类对父类的引用。
使用场景
区别
1.成员变量
1
2
|
this
.变量 -- 本类的
super
.变量 -- 父类的
|
2.构造方法
1
2
|
this
(...) -- 本类的
super
(...) -- 父类的
|
3.成员方法
1
2
|
this
.方法名() -- 本类的
super
.方法名() -- 父类的
|
super();和this();都是在构造函数的第一行,不能同时出现。
子类中出现与父类一模一样的方法时(除了权限修饰符,权限修饰符大于等于不包括private,返回值类型,方法名和参数列表相同),会出现覆盖操作,也称为重写或者复写。
父类私有方法,子类看不到,因此父类私有方法的重写也就无从谈起。
覆盖注意事项:
覆盖的使用场景:
当子类需要父类的功能,而功能主体子类有自己特有内容时,可以复写父类中的方法,这样,既沿袭了父类的功能,又定义了子类特有的内容。
方法重写和重载有什么区别?
方法的重写用在子类方法与父类方法一模一样时,除权限修饰符,返回值类型,方法名和参数列表都是相同的。
重载用在同一个类中各方法方法名相同,参数列表不同(与返回值类型没有关系)的情况。
子父类中构造方法的用法:
静态代码块、构造代码块,构造方法的执行顺序:
父类静态代码块→子类静态代码块→父类构造代码块→父类构造方法→子类构造代码块→子类构造方法
final是一个关键字,可以用于修饰类,成员变量,成员方法。
特点:
final修饰的常量定义一般都有书写规范,被final修饰的常量名称,所有字母都大写。
final修饰成员变量,必须初始化,初始化有两种
final和private的区别:
概念:
对象在不同时刻表现出来的不同状态。
多态的前提:
程序中的体现:
父类或者接口的引用指向或者接收自己的子类对象。
好处和作用:
多态的存在提高了程序的扩展性和后期可维护性。
弊端:
父类调用的时候只能调用父类里的方法,不能调用子类的特有方法,因为你并不清楚将来会有什么样的子类继承你。
多态的成员特点:
一定不能够将父类的对象转换成子类类型!
父类的引用指向子类对象,该引用可以被提升,也可以被强制转换。
多态自始至终都是子类对象在变化!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
//多态向下转型和向上转型的例子,多态转型解决了多态中父类引用不能使用子类特有成员的弊端。
class
PolymorphicTest2 {
public
static
void
main(String[] args) {
Phone p1 =
new
Nokia();
//向上转型,类型提升
Nokia no = (Nokia)p1;
//向下转型,强制将父类的引用转换成子类类型,不能将Nokia类型转成Moto或Nexus类型
no.print();
//输出结果为Phone---null---0,因为继承了父类的方法
Phone p2 =
new
Moto();
Moto m = (Moto)p2;
m.print();
//输出结果为Moto---yellow---1599,方法重写,子类方法覆盖父类方法
Phone p3 =
new
Nexus();
Nexus ne = (Nexus)p3;
ne.print();
}
}
class
Phone{
String color;
int
price;
public
void
print(){
System.out.println(
"Phone---"
+ color +
"---"
+ price );
}
}
class
Nokia
extends
Phone{
String color =
"red"
;
int
price =
1009
;
//public void print(){
// System.out.println("Nokia---" + color + "---" + price);
//}
}
class
Moto
extends
Phone{
String color =
"yellow"
;
int
price =
1599
;
public
void
print(){
System.out.println(
"Moto---"
+ color +
"---"
+ price);
}
}
class
Nexus
extends
Phone{
String color =
"black"
;
int
price =
1999
;
public
void
print(){
System.out.println(
"Nexus---"
+ color +
"---"
+ price);
}
}
}
|
抽象就是从多个事物中将共性的,本质的内容抽象出来。
抽象类:
Java中可以定义没有方法体的方法,该方法的具体实现由子类完成,该方法称为抽象方法,包含抽象方法的类就是抽象类。
由来:
多个对象都具备相同的功能,但是功能具体内容有所不同,那么在抽取过程中,只抽取了功能定义,并未抽取功能主体,那么只有功能声明,没有功能主体的方法称为抽象方法。
抽象类特点:
抽象类的成员特点:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
abstract
class
葵花宝典 {
public
abstract
void
自宫();
}
class
岳不群
extends
葵花宝典 {
public
void
自宫(){
System.out.println(
"剪刀"
);
}
}
class
林平之
extends
葵花宝典{
public
void
自宫(){
System.out.println(
"指甲刀"
);
}
}
class
AbstractTest {
public
static
void
main(String[] args) {
岳不群 岳 =
new
岳不群();
岳.自宫();
林平之 林 =
new
林平之();
林.自宫();
}
}
|
抽象类注意事项:
抽象类不能被实例化,为什么还有构造函数?
只要是class定义的类里面就肯定有构造函数。抽象类中的函数是给子类实例化的。
一个类没有抽象方法,为什么定义为抽象类?
不想被继承,还不想被实例化。
抽象关键字abstract不可以和哪些关键字共存?
接口(interface)
接口是抽象方法和常量值的集合。从本质上讲,接口是一种特殊的抽象类,这种抽象类只包含常量和方法的定义,而没有变量和方法的实现。
格式:interface 接口名{}
接口的出现将”多继承“通过另一种形式体现出来,即”多实现“。
实现(implements)
格式:class 类名 implements 接口名 {}
特点:
接口的成员特点:
接口中的成员修饰符是固定的!
继承与实现的区别:
抽象类和接口的区别:
成员变量
成员方法
构造方法
-抽象类有构造方法
-接口没有构造方法
类与抽象类和接口的关系
接口的思想特点:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
55
56
57
58
59
60
61
62
63
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
|
//运动员和教练的案例(下图是思路分析)
/*
篮球运动员和教练
乒乓球运动员和教练
现在篮球运动员和教练要出国访问,需要学习英语
请根据你所学的知识,分析出来哪些是类,哪些是抽象类,哪些是接口
*/
interface
SpeakEnglish {
public
abstract
void
speak();
}
interface
GoAboard{
public
abstract
void
aboard();
}
abstract
class
Person {
private
String name;
private
int
age;
public
Person(){}
public
Person(String name,
int
age){
this
.name = name;
this
.age = age;
}
public
void
setName(String name){
this
.name = name;
}
public
String getName(){
return
name;
}
public
void
setAge(
int
age){
this
.age = age;
}
public
int
getAge(){
return
age;
}
//吃饭
public
abstract
void
eat();
//睡觉
public
void
sleep(){
System.out.println(
"Zzz..."
);
}
}
//运动员
abstract
class
Player
extends
Person {
public
abstract
void
study();
}
//教练
abstract
class
Coach
extends
Person {
public
abstract
void
teach();
}
//篮球运动员
class
BasketballPlayer
extends
Player
implements
SpeakEnglish,GoAboard{
public
void
eat(){
System.out.println(getAge() +
"岁的"
+ getName() +
"吃鸡腿"
);
}
public
void
study(){
System.out.println(getAge() +
"岁的"
+ getName() +
"学扣篮"
);
}
public
void
speak(){
System.out.println(getAge() +
"岁的"
+ getName() +
" Say Hello World"
);
}
public
void
aboard(){
System.out.println(getAge() +
"岁的"
+ getName() +
" Go Aboard"
);
}
}
//乒乓运动员
class
PingPangPlayer
extends
Player{
public
void
eat(){
System.out.println(getAge() +
"岁的"
+ getName() +
"吃鸡蛋"
);
}
public
void
study(){
System.out.println(getAge() +
"岁的"
+ getName() +
"学扣球"
);
}
}
//篮球教练
class
BasketballCoach
extends
Coach
implements
SpeakEnglish {
public
void
eat(){
System.out.println(getAge() +
"岁的"
+ getName() +
"啃鸡爪"
);
}
public
void
teach(){
System.out.println(getAge() +
"岁的"
+ getName() +
"教扣篮"
);
}
public
void
speak(){
System.out.println(getAge() +
"岁的"
+ getName() +
" Say Hello Java"
);
}
public
void
aboard(){
System.out.println(getAge() +
"岁的"
+ getName() +
" Go Aboard"
);
}
}
//乒乓球教练
class
PingPangCoach
extends
Coach{
public
void
eat(){
System.out.println(getAge() +
"岁的"
+ getName() +
"吃鸡蛋皮"
);
}
public
void
teach(){
System.out.println(getAge() +
"岁的"
+ getName() +
"教扣球"
);
}
}
class
PlayerAndCoach {
public
static
void
main(String[] args) {
//篮球运动员
BasketballPlayer bp =
new
BasketballPlayer();
bp.setName(
"郭艾伦"
);
bp.setAge(
33
);
bp.eat();
bp.sleep();
bp.study();
bp.speak();
bp.aboard();
System.out.println(
"***********************"
);
//篮球教练
BasketballCoach bc =
new
BasketballCoach();
bc.setName(
"波波维奇"
);
bc.setAge(
65
);
bc.eat();
bc.sleep();
bc.teach();
bc.speak();
bc.aboard();
System.out.println(
"***********************"
);
//多态
Person p =
new
BasketballPlayer();
p.setName(
"Kobe Bryant"
);
p.setAge(
33
);
p.eat();
p.sleep();
//p.study();
//p.speak();
BasketballPlayer bp2 = (BasketballPlayer)p;
bp2.study();
bp2.speak();
bp2.aboard();
System.out.println(
"***********************"
);
}
}
|
将一个类定义在另一个类里面,里面的那个类就称为内部类。内部类的出现,再次打破了Java单继承的局限性。
访问特点:
共性:
成员内部类
在外部类中有成员变量和成员方法,成员内部类就是把整个一个类作为了外部类的成员;
成员内部类是定义在类中方法外的类;
创建对象的格式为:外部类名.内部类名 对象名 = 外部类对象.内部类对象;
成员内部类之所以可以直接访问外部类的成员,那是因为内部类中都持有一个外部类对象的引用:外部类名.this;
成员内部类可以用的修饰符有final,abstract,public,private,protected,static.
静态内部类
静态内部类就是成员内部类加上静态修饰符static,定义在类中方法外。
在外部类中访问静态内部类有两种场景:
局部内部类
局部内部类是定义在方法中的类。
可以用于方法内部类的修饰符有final,abstract;
静态方法中的方法内部类只能访问外部的静态成员。
匿名内部类
匿名内部类是内部类的简化写法,是建立一个带内容的外部类或者接口的子类匿名对象。
前提:
内部类可以继承或实现一个外部类或者接口。
格式:
new 外部类名或者接口名(){重写方法};
通常在方法的形式参数是接口或者抽象类,并且该接口中的方法不超过三个时,可以将匿名内部类作为参数传递。
类 | 成员变量 | 成员方法 | 构造方法 | |
---|---|---|---|---|
private | Y | Y | Y | |
默认 | Y | Y | Y | Y |
protected | Y | Y | Y | |
public | Y | Y | Y | Y |
abstract | Y | Y | ||
static | Y | Y | Y | |
final | Y | Y | Y |
注意,常见规则如下:
本类 | 同包(无关类或子类) | 不同包(子类) | 不同包(无关类) | |
---|---|---|---|---|
private | Y | |||
默认 | Y | Y | ||
protected | Y | Y | Y | |
public | Y | Y | Y | Y |
推荐:
因篇幅问题不能全部显示,请点此查看更多更全内容
Copyright © 2019- stra.cn 版权所有 赣ICP备2024042791号-4
违法及侵权请联系:TEL:199 1889 7713 E-MAIL:2724546146@qq.com
本站由北京市万商天勤律师事务所王兴未律师提供法律服务