面向对象3
发布日期:2021-06-27 12:55:45 浏览次数:28 分类:技术文章

本文共 7124 字,大约阅读时间需要 23 分钟。

文章目录

面向对象3

一、继承

继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为。

继承的限制:

Java中只有单继承,多重继承,没有多继承。

格式:	class 父类 {	} 	class 子类 extends 父类{ 	}

当子类实例化时,会首先在堆中创建父类内存,进而创建自身,且在子类中通过super指向父类。

1、super

· 通过super,可以访问父类的构造方法		调用super构造方法的代码,必须写在子类构造方法的第一行。· 通过super,可以访问父类的属性· 通过super,可以访问父类的方法

注意:子类创建多少个对象,自动的也会创建多少个父类的对象

this的对比

super关键字用来访问父类内容,而this关键字用来访问本类内容。this的用法也有三种:1. 在奔雷的成员方法中,访问本类的属性;2. 在本类的成员方法中,访问本类的另一个成员方法;3. 在本类的构造方法中,访问本类的另一个构造方法其中,用法3中要注意:· this(..)调用也必须放在构造方法的第一句;· super和this两种构造调用,不能同时使用。

二、重写与重载

重写规则:

  1. 参数列表必须完全与被重写方法的参数列表相同;
  2. 返回类型必须完全与被重写方法的返回类型相同;
  3. 访问权限不能比父类中被重写的方法的访问权限更低;
    例如:如果父类的一个方法被声明为public,那么在子类中重写该方法就不能生命为protected
  4. 父类的成员方法只能被它的子类重写;
  5. 声明为staticprivate的方法不能被重写,但是能够被再次声明。

重写(Override)与重载(Overload)的区别

  1. 发生的位置不同:
    重载:一个类中
    重写:子父类中
  2. 参数列表限制:
    重载:必须不同的
    重写:必须相同的
  3. 返回值类型:
    重载:与返回值类型无关
    重写:返回值类型必须一致
  4. 访问权限:
    重载:与访问权限无关
    重写:子的方法权限不能小于父的访问权限
  5. 异常处理
    重载:与异常无关
    重写:异常范围可以更小,但是不能抛出新的异常

三、final关键字

final关键字:		final用于修饰属性、变量		·变量成为了常量,无法对其再次进行赋值		·final修饰的局部变量,只能赋值一次(可以先声明后赋值)		·final修饰成员属性,必须在声明时进行赋值。		全局常量(public static final)		常量的命名规范:			由1个或多个单词组成,单词与单词之间必须用下划线隔开,单词中所有字母大写。如:SQL_INSERT				final用于修饰类		final修饰的类不可以被继承			final用于修饰方法		final修饰的方法,不能被子类重写

四、抽象类

1、概念

抽象类必须用abstract class声明

一个抽象类中可以没有抽象方法。抽象方法必须写在抽象类或者接口中。

格式:

abstract class 类名{
//抽象类 }

2、抽象方法

只声明而未实现的方法称为抽象方法(未实现指的是:没有“{}”方法体),抽象方法必须使用abstract关键字声明。

格式:

abstract class 类名{
//抽象类 public abstract void 方法名(); //抽象方法,只声明而未实现}

3、抽象类的使用原则及常见问题

在抽象类的使用中有几个原则:	· 抽象类本身是不能直接进行实例化操作的,即:不能直接使用关键字new完成。	· 一个抽象类必须被子类所继承,被继承的子类(如果不是抽象类)则必须覆写(重写)抽象类中的全部抽象方法。

常见问题

  1. 抽象类能否使用final声明?

    不能,因为final属修饰的类是不能有子类的 , 而抽象类必须有子类才有意义,所以不能。

  2. 抽象类能否有构造方法?

    能有构造方法,而且子类对象实例化的时候的流程与普通类的继承是一样的,都是要先调用父类中的构造方法(默认是无参的),之后再调用子类自己的构造方法。

抽象类与普通类的区别:

  1. 抽象类必须用public或protected修饰(如果为private修饰,那么子类则无法继承,也就无法实现其抽象方法)。默认缺省为 public
  2. 抽象类不可以使用new关键字创建对象, 但是在子类创建对象时, 抽象父类也会被JVM实例化。
  3. 如果一个子类继承抽象类,那么必须实现其所有的抽象方法。如果有未实现的抽象方法,那么子类也必须定义为abstract类

有抽象方法的类一定是抽象类,抽象类中不一定有抽象方法

五、接口

1、概念

如果一个类中的全部方法都是抽象方法,全部属性都是全局常量,那么此时就可以将这个类定义成一个接口。

格式:

interface 接口名称{
全局常量; 抽象方法; }

2、接口的实现 implements

接口能够多实现,格式如下:

class 子类 implements 父接口1,父接口2...{
}

以上的代码称为接口的实现。那么如果一个类即要实现接口,又要继承抽象类的话,则按照以下的格式编写即可:

class 子类 extends 父类 implements 父接口1,父接口2...{
}

a、全局常量和抽象方法在接口中的简写

由于接口本身都是由全局常量和抽象方法组成,所以接口中的成员定义可以简写:

  1. 全局常量,可以成略public static final
public static final int a  = 10;//可以简写如下:int a = 10;
  1. 抽象方法简写,可以省略public abstract
public abstract void print();//可以简写为void print();

b、接口的继承

接口由于都是抽象部分,不存在具体实现,所以允许多继承

interface c extends A,B{
}

**注意:**如果一个接口要想使用,必须依靠子类。子类(如果不是抽象类的话)要实现接口中的所有抽象方法

3、 接口和抽象类的区别

1、抽象类要被子类继承,接口要被子类实现。2、接口只能声明抽象方法,抽象类中可以声明抽象方法,也可以写非抽象方法。3、接口里定义的变量只能是公共的静态的常量,抽象类中的变量是普通变量。4、抽象类使用继承来使用, 无法多继承。 接口使用实现来使用, 可以多实现5、抽象类中可以包含static方法 ,但是接口中不允许(静态方法不能被子类重写,因此接口中不能声明静态方法)6、接口不能有构造方法,但是抽象类可以有

六、多态

多态:就是对象的多种表现形式,(多种体现形态)

多态的体现

对象的多态性,从概念上非常好理解,在类中有子类和父类之分,子类就是父类的一种形态 ,对象多态性就从此而来。

ps: 方法的重载 和 重写 也是多态的一种, 不过是方法的多态(相同方法名的多种形态)。

重载: 一个类中方法的多态性体现

重写: 子父类中方法的多态性体现。

1、多态的使用

类似于基本数据类型的转换:· 向上转型:将子类实例变为父类实例		|- 格式:父类 父类对象 = 子类实例 ;· 向下转型:将父类实例变为子类实例		|- 格式:子类 子类对象 = (子类)父类实例 ;

2、instanceof

作用:	判断某个对象是否是指定类的实例,则可以使用instanceof关键字	格式:	实例化对象 instanceof 类 //此操作返回boolean类型的数据
public class Demo{
public static void main(String[] args){
Nurse n = new Nurse(); say(n); } public static void say(Person p){
//如何判断传入的对象是此类型的哪种形态(哪个子类的对象) if(p instanceof Student){
Student s= (Student) p; s.say(); }else{
System.out.println("传入学生形态方可执行"); }}

七、Object类

Object类是所有类的父类(基类),如果一个类没有明确的继承某一个具体的类,则将默认继承Object类。

多态性: 使用Object可以接收任意的引用数据类型

方法

1、toString

作用: 返回对象的字符串表示形式。

Object的toString方法, 返回对象的内存地址。

建议重写Object中的toString方法。

2、equals

作用: 指示某个其他对象是否“等于”此对象。

Object的equals方法:实现了对象上最具区别的可能等价关系; 也就是说,对于任何非空引用值x和y ,当且仅当x和y引用同一对象( x == y具有值true )时,此方法返回true 。

建议重写Object中的equals(Object obj)方法

equals==的区别:==比较的是内存地址是否相等

equals方法重写时的五个特性:自反性 :对于任何非空的参考值x , x.equals(x)应该返回true 。对称性 :对于任何非空引用值x和y , x.equals(y)应该返回true当且仅当y.equals(x)回报true 传递性 :对于任何非空引用值x , y和z ,如果x.equals(y)回报true个y.equals(z)回报true ,然后x.equals(z)应该返回true 。一致性 :对于任何非空引用值x和y ,多次调用x.equals(y)始终返回true或始终返回false ,前提是未修改对象上的equals比较中使用的信息。非空性 :对于任何非空的参考值x , x.equals(null)应该返回false 。
public boolean equals(Object o) {
if(this == o) {
return true; } if(o == null) {
return false; } if(o instanceof Person) {
Person p2 = (Person)o; if(this.name.equals(p2.name) && this.age == p2.age) {
//相同 return true; } } return false; }

八、内部类

在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。

广泛意义上的内部类一般来说包括这四种:1. 成员内部类2. 局部内部类3. 匿名内部类4. 静态内部类

1、成员内部类

成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:

class Outer {
private double x = 0; public Outer(double x) {
this.x = x;} class Inner {
//内部类 public void say() {
System.out.println("x="+x); } }}
特点: 成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:	外部类.this.成员变量	外部类.this.成员方法

外部使用成员内部类

  • Outter outter = new Outter();
  • Outter.Inner inner = outter.new Inner();

2、局部内部类

局部内部类是定义在一个方法或者一个作用域里面的类

它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。

注意: 局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。

class Person{
public Person() {
} } class Man{
public Man(){
} public People getPerson(){
class Student extends People{
//局部内部类 int age =0; } return new Student(); } }

3、匿名内部类

只使用一次

创建格式:new 父类构造器(参数列表)|实现接口(){	//匿名内部类的类体部分}

使用匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的引用。当然这个引用是隐式的。

注意:

  1. 使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
  2. 匿名内部类中是不能定义构造函数的。
  3. 匿名内部类中不能存在任何的静态成员变量和静态方法。
  4. 匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。
  5. 匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。
  6. 只能访问final型的局部变量
    是因为内部类会单独地编译成一个字节码文件,为了保证文件内的变量与外部变量的值是一致的,从规则上限制了变量的值不可更改。
public class  Demo{
public static void main(String[] args){
Person p =new Person(){
public void say(){
System.out.println("你好"); } }; haha(p); } public static void haha(Person p){
}}
public interface Person{
void say(); }

4、静态内部类

静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。

静态内部类是不需要依赖于外部类对象的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法.

格式

//格式如下:public class Test{
public static void main(String[] args){
Outter.Inner inner = new Outter.Inner();//创建静态内部类的对象 }}class Outter{
public Outter(){
} static class Inner{
public Inner(){
} }}

九、包装类

在这里插入图片描述

以上的八种包装类,可以将基本数据类型按照类的形式进行操作。

八种包装类分为两大类型:

  1. Number:Integer、Short、Long、Double、Float、Byte都是Number的子类表示是一个数字。
  2. Object:Character、Boolean都是Object的直接子类。

1、装箱和拆箱

将一个基本数据类型变为包装类,那么这样的操作称为装箱操作。

将一个包装类变为一个基本数据类型,这样的操作称为拆箱操作,

2、字符串转换

在Integer类中提供了以下的操作方法:	public static int parseInt(String s) :将String变为int型数据在Float类中提供了以下的操作方法:	public static float parseFloat(String s) :将String变为Float在Boolean 类中提供了以下操作方法:	public static boolean parseBoolean(String s) :将String变为boolean	....	....

九、可变参数

一个方法中定义完了参数,则在调用的时候必须传入与其一一对应的参数,但是在JDK 1.5之后提供了新的功能,可以根据需要自动传入任意个数的参数。

public class Demo{
public static void main(String[] args){
System.out.println(sum(1)); System.out.println(sum(1,3,5)); } /** *int... nums :表示可变参数,调用时可以传入n个字符 *在方法的内部,可变参数以数组作为载体体现 */ public static int sum(int... nums){
int n = 0; for(int i = 0;i

可变参数只能出现在参数列表的最后。

转载地址:https://blog.csdn.net/weixin_43454088/article/details/115755025 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:异常处理
下一篇:面向对象2

发表评论

最新留言

路过,博主的博客真漂亮。。
[***.116.15.85]2024年04月02日 01时03分51秒