- 继承:简单通俗的来讲,继承就是一个类继承另一个类,通常用extends表示继承。
- 继承的类叫子类,被继承的类叫父类。
- 子类可以使用父类的变量和方法,同时也可以重写父类的方法。
- 在Java中没有多继承这一概念,但是有类似多继承的方法!(通常来讲一个父类可以有多个子类,而一个子类只能有一个父类)
//继承简单表示
public class Son extends Father{
//将特殊的方法放在子类中
}
class Father {
//我们通常将一般的方法放在父类中,因为父类的方法子类都可以用。
}
二、super、this、instanceof关键字
this
- 上一篇文章已经说过了方法重写的概念,在这里就不再重复了。
- this关键字也在上一篇讲过,我们可以在构造器中通过this()来调用本类的不同参数构造器。通常我们都写在第一行!
- 还可以通过
this.变量
的方式表示使用的是本类的变量
- super关键字就是在继承中应用的,我们可以通过super()调用父类的不同参数构造器
- 同时还可以通过
super.方法
,调用父类的方法。 - 同时它也是写在第一行!
- 通常使用super是因为子类继承了父类,这样子类就不必要在写一些成员变量,直接在构造器中通过super()调用父类的构造器,将参数初始化即可。
public class Son extends Father{
Son(String name, int age) {
super(name, age);//如果子类的构造器没有显示的调用父类的构造器,则将默认的调用父类的无参构造。
}
}
class Father {
String name;
int age;
Father(String name,int age) {
this.name = name;
this.age = age;
}
}
//若父类中没有无参构造,而子类构造器中又没有调用父类的其它构造器,则Java编译器会报错。
instanceof对于继承简言之就是:父类有的子类都有,父类没有的子类也可以有
public void instanceOf(Animal T) {
if(T instanceof Dog) { //我们传进来的dog就是此时的T,通过instanceof检测它是否属于Dog或者Animal,可以判断对象的类型。
//此时的语句应该是这样的 Animal T = new Dog(); 而我们应该将T向下转型
t = (Dog) T;
System.out.println("yes");
}
}
Animal dog = new Dog();
Animal cat = new Cat();
cat.instanceOf(dog);//程序将输出"yes"
三、多态
- 多态是在继承的基础上实现的。也称之为(向上转型)
- 大家只需要记住对于成员变量:编译看左边,运行也看左边。
- 而对于方法:则是编译看左边,运行看右边。
public class Animal {
public void play() {
System.out.println("玩");
}
public void eat() {
System.out.println("吃");
}
}
public class Cat extends Animal{
public void eat() {
System.out.println("猫吃饭,亲密度+8");
}
public void play() {
System.out.println("撸猫,体力值-9");
}
}
public class Test {
public static void main(String[] args) {
Animal cat = new Cat();//此时cat被看成是Animal的对象,但实际上本质是Cat的。
//在编译阶段我们看左边,它是Animal骗过编译器,但真正运行的时候它会看右边。
cat.eat();
cat.play();
//最终输出“猫吃饭,亲密度+8”和“撸猫,体力值-9”。这就是多态的应用
//看右边,就是先去寻找Cat中是否有重写的父类方法,如果有则调用自己的。如果没有则用父类的。
//可以记为:先调子类,再调父类。
}
}
Animal cat = new Cat();这也是向上转型,将Cat类的转成了Animal
- 抽象类:使用abstract关键字
- 抽象类中的方法不需要实现,只需要声明占一个位置就行。我们可以在子类中去具体实现这个方法。
- 这样更有灵活性,就像我们定义了一个抽象方法eat(),而不同的子类可以去实现成不同的方法,猫可以实现吃鱼,狗可以实现吃翔,使程序更加简单化。
- 抽象类无法实例化,也就不能造对象了。
- 再次总结权限修饰范围
- 仅对本类可见————private
- 对外部完全可见————public
- 对本包和所有子类可见————protected
- 对本包可见————缺省,也就使默认的
- Object是所有类的父类
- 既是没有明显写出继承Object,但也默认认为Object是父类
- Object类中的equals()方法比较的是内存地址,通常我们都会重写equals()方法,达到值比较的目的。
- getClass方法将返回一个对象所属的类。我们可以通过它检测两个对象是否属于同一个类。
Java规范要求equals方法具有下面的特性:
1.自反性:对于任何非空引用x,x.equals(x)应该返回true。
2.对称性:对于任何引用x和y,当且仅当y.equals(x)返回true时,x.equals(y)返回true。
3.传递性:对于任何引用x、y和z,如果x.equals(y)返回true,y.equals(z)返回true,则x.equals(z)也应该为true。
4.一致性:如果x和y引用的对象没有发生变化,反复调用x.equals(y)应该返回同样的结果。
5.对于任意非空引用x,x.equals(null)应该返回false。
- 但是对于一个e是Employee对象,m是Manager对象,并且两个对象的值都是相同的。如果调用e.equals(m)则返回true。若反过来调用m.equals(e)则返回false。这就违反了我们的对称性。
- 因为我们用的是instanceof检测的,但是子类instanceof父类是无法进行的。所以这也是instanceof的缺点
- 我们可以通过getClass来比较两个对象是否属于同一个类,但是这样也有限制,就是多态的情况下返回false。
如何正确重写equals方法: - 比较常见的两种重写方法:
- 用
instanceof
实现重写equals
方法 - 用
getClass
实现重写equals
方法
- 用
class Son extends Father{
public static void main(String[] args) {
Son son = new Son();
Father father = new Father();
Father sss = new Son();
System.out.println(father.equals(son));
}
}
class Father{
public boolean equals(Object obj) {
if(this == obj) return true;//判断是否同一个对象
//如果两个类不是同一个类
// if(obj == null || obj.getClass() != this.getClass()) {
// return false;
// }
//传进来的对象不属于这个类或者它的父类,返回false
System.out.println(obj.getClass());
if(obj == null || !(obj instanceof Father)) {//左边是子类或者本类的对象
return false;
}
Father obj1 = (Father) obj;//强转然后判断
return true;
}
}
- 这里instanceof的缺点就是,父类调用比较子类为true,子类调用比较父类为false,所以我们可以尽量调用父类的equals来避免这个问题。
- 同时也可以使用getClass进行判断,是否为同一个类对象。
- String类计算散列码算法:
点击查看代码
int hash = 0;
for(int i = 0; i < length(); i++) {
hash = 31*hash + charAt(i);
}
- 其实这个方法非常简单,就是把我们的字段以字符串的形式,好看的输出出来。
@Override
public String toString() {
return getClass().getName()
+"{"
+ "tili="
+ tili
+ ", qimi="
+ qimi
+ '}';
}
- 在使用的时候我们可以写成
System.out.println(x.toString());
System.out.println(x);这样会默认调用toString()方法,可以简略不写
但是在有时候我们用toString方法是会出现输出java.io.PrintStream@2f6684和[I@1a46e30的情况
1.这是因为Objcet类定义了toString方法,可以打印对象的类名和散列码,所以需要我们对toString方法进行重写
2.数组也继承了Objcet类的toString,如果我们使用时不重写toString,则会出现[I@1a46e30
,补救办法就时使用Arrays.toString()
3.若是二维数组,则可以使用Arrays.deepToString()方法
4.强烈建议在每一个类中都重写toString方法。
- 在做题时,必须确定数组的长度以后才能够使用,这样使我们非常的不方便!
- 所以我们就可以使用一个动态扩容的数组,这样就不必定义它的初始化长度。
- ArraysList
list = new ArrayList<>(100) 这是初始化100的数组,可动态扩容 - 如果说空间过多浪费了,可以使用trimToSize()方法,将没用过的空间释放掉。
- 具体详细的ArrayList用法,我们在日后的集合学习。
int x = Integer.parseInt(s);
String s = String.ValueOf(x);
System.out.println(sum(1,2,3,4,5));
public static int sum(int... a) {
int total = 0;
for(int i:a) {
total += i;
}
return total;
}
六、枚举类
- 枚举用enum表示
- public enum Size{1,2,3,4,5};一次自增1
- 枚举的构造器总是私有的
- 对于继承、多态内容就总结这么多,希望大家可以多多练习。如果有不足之处,希望大家多多包涵,多多支持。如果有不懂的地方可以直接私信问我,欢迎来访!
- 我将会继续更新关于Java的学习知识,感兴趣的小伙伴可以关注一下。
- 文章写得比较走心,用了很长时间,绝对不是copy过来的!
- 尊重每一位学习知识的人,同时也尊重每一位分享知识的人。