Java学习笔记-面向对象高级

本文详细介绍了Java中的面向对象特性,包括子类实例化内存分析、构造方法调用过程、多层继承构造方法调用顺序。讲解了super关键字的使用、重写与重载的区别、final关键字的应用、抽象类与接口的定义及区别、多态的概念及其表现形式,以及Object类的重要方法。此外,还涵盖了内部类、包装类、可变参数等相关内容。
摘要由CSDN通过智能技术生成

面向对象高级

子类实例化内存分析

子类继承了父类后,拥有了父类的一个地址

继承关系中,构造方法的调取过程

子类创建对象时,默认会调用父类的无参构造
在子类构造方法的第一行都有一条默认语句super(); --调用父类的无参构造
当父类没有无参构造时,可以通过super调用父类的其它构造方法.
构造方法不可以被继承 .
执行子类的构造方法时一定先执行父类的构造方法  

Java中含有多层继承关系的构造方法的调用顺序

(1)当创建一个类的对象,且该类是继承自某个父类时,会先调用父类的构造方法,这个步骤会反复递归下去,首先是构造这种层次结构的根,然后是下一层导出类,直到最底层的导出类。
(2)按照声明顺序初始化成员变量
(3)调用该类的构造方法主体。

super详解

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

重写与重载的区别

1.参数列表必须完全与被重写方法相同
2.返回类型必须完全与被重写方法的返回类型相同;
3.访问权限不能比父类中被重写的访问权限更低。例如:如果父类的一个方法被声明为public,那么在子类中重写该方法的权限就不能声明得比public权限低
4.父类的成员方法只能被它的子类重写
5.声明为static和private的方法不能被重写,但是能够被再次声明
​
面试题:重写与重载的区别
    1.发生位置
        重写发生在子父类之中
        重载发生在一个类中
    2.参数列表限制
        重写的参数列表必须相同
        重载的参数列表必须不相同
    3.返回值类型
        重写的返回值类型必须一致
        重载与返回值类型无关
    4.访问权限
        重写的子类方法权限不能小于父类方法的权限
        重载与访问权限无关
    5.异常处理
        重写的异常范围可以更小,但是不能抛出新的异常
        重载与异常无关
    

final关键字

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

抽象类

概念

抽象类必须使用abstract class 声明
一个抽象类中可以没有抽象方法。抽象方法必须写在抽象类或者接口中
格式:
    abstract class 类名{//抽象类
    }

抽象方法

只声明而未实现的方法称为抽象方法(未实现指的是:没有"{}"方法体),抽象方法必须使用abstract关键字声明
格式:
    abstract class 类名{//抽象类
        public abstract void 方法名();  //抽象方法,只声明而未实现
    }

不能被实例化

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

常见问题

1.抽象类能否使用final声明?
	不能,因为final属性修饰的类不能有子类,而抽象类要有子类才有意义
2.抽象类能否有构造方法?
	能有构造方法,而且子类对象实例化的时候的流程与普通类的继承是一样的,都是要先调用父类中的构造方法(默认是无参构造),之后调用子类自己的构造方法

抽象类和普通类的区别

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

接口

概念

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

面向接口编程思想

这种思想是接口的定义(规范,约束)与实现(名实分离的原则)的分离

优点:
	1.降低程序的耦合性
	2.易于程序的扩展
	3.有利于程序的维护

全局常量和抽象方法的简写

因为接口本身都是由全局常量和抽象方法组成,所以接口中的成员定义可以简写:
	1.全局常量编写时,可以省略 public static final 关键字,例如:
		public static final String INFO = “内容”;
		简写后:
		String INFO = “内容”;
	2.抽象方法编写时,可以省略 public abstract 关键字 	例如:
		public abstract void print();
		简写后:
		void print();

接口的实现implements

接口可以多实现:
格式:
	class 子类 implements 父接口1,父接口2...{
	}
	
以上的代码称为接口的实现。那么如果一个类即要实现接口,又要继承抽象类的话,则按照以下的格式编写即可:
	class 子类 extends 父类 implements 父接口1,父接口2...{
	}

接口的继承

接口因为都是抽象部分,不存在具体的实现,所以允许多继承,例如:
interface C extends A,B{
}

注意

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

接口和抽象类的区别

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

多态

概念

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

多态的体现

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

ps:方法的重载 和 重写 也是多态的一种,不过是方法的多态(方法名相同的多种形态)。
	重载:一个类中方法的多态性体现
	重写:子父类中方法的多态性体现

多态的使用:对象的类型转换

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

instanceOf

作用:
	判断某个对象是否是指定类的实例,则可以用instanceOf关键字
格式:
	实例化对象 instanceOf 类 //此操作返回boolean类型的数据

Object类

概念

object类是所有类的父类(基类),如果一个类没有明确的继承某一个具体的类,则将默认继承object类
例如定义一个类
	public class Person{
	}
其实是这样的
	public class Person extends Object{
	}

object的多态

使用object可以接受任意的引用数据类型

toString

建议重写objecr中的toString方法。此方法的作用:返回对象的字符串表现形式
	object的toString方法,返回对象的内存地址

equals

== 比较的是地址值     默认的equals也是比较地址值

建议重写Object中的equals(Object obj)方法,此方法的作用:指示某个其他对象是否“等于”此对 象。 
	Object的equals方法:实现了对象上最具区别的可能等价关系; 也就是说,对于任何非空引用值x和 y ,当且仅当x和y引用同一对象( x == y具有值true )时,此方法返回true 。 
		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 。

内部类(不常用)

概念

在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。 
	广泛意义上的内部类一般来说包括这四种: 	
		1、成员内部类 
		2、局部内部类 
		3、匿名内部类 
		4、静态内部类

成员内部类

成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式: 
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();

局部内部类

局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限 于方法内或者该作用域内。 
例如:
	class Person{ 
		public Person() {
         } 
     }
     
     class Man{ 
     	public Man(){
        }
        
        public People getPerson(){ 
        	class Student extends People{ //局部内部类 
        		int age =0; 
        		}
        		return new Student(); 
        	} 
        }
 注意:局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及 static修饰符的。

匿名内部类

匿名内部类由于没有名字,所以它的创建方式有点儿奇怪。创建格式如下: 
		new 父类构造器(参数列表)|实现接口() 
		{
		//匿名内部类的类体部分 
		} 
在这里我们看到使用匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类 或者实现一个接口。同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的 引用。当然这个引用是隐式的。

注意:

在使用匿名内部类的过程中,我们需要注意如下几点: 
	1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能 继承一个类或者实现一个接口。 
	2、匿名内部类中是不能定义构造函数的。 
	3、匿名内部类中不能存在任何的静态成员变量和静态方法。 
	4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。 
	5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。 
	6、只能访问final型的局部变量

静态内部类

静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字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() { 
            } 
        } 
    }

包装类

概述

在Java中有一个设计的原则“一切皆对象”,那么这样一来Java中的一些基本的数据类型,就完全不符合于 这种设计思想,因为Java中的八种基本数据类型并不是引用数据类型,所以Java中为了解决这样的问题, 引入了八种基本数据类型的包装类。
序号基本数据类型包装类
1intInteger
2charCharacter
3floatFloat
4doubleDouble
5booleanBoolean
6byteByte
7shortShort
8longLong
以上的八种包装类,可以将基本数据类型按照类的形式进行操作。 
	但是,以上的八种包装类也是分为两种大的类型的: 
		· Number:Integer、Short、Long、Double、Float、Byte都是Number的子类表示是一个 数字。 
		· Object:Character、Boolean都是Object的直接子类。

装箱和拆箱操作

以Integer和Float为例进行操作

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

因为所有的数值型的包装类都是Number的子类,Number中定义了如下的操作方法,以下全部方法都是进行拆箱的操作
序号方法描述
1public byte byteValue()Byte--->byte
2public abstract double doubleValue()Double--->double
3public abstract float floatValue()Float--->float
4public abstract long longValue()Long--->long
5public abstract int intValue()Integer--->int
6public short shortValue()Short--->short
装箱操作:
	在JDK1.4之前 ,如果要想装箱,直接使用各个包装类的构造方法即可,例如: 
		int temp = 10 ; // 基本数据类型 
		Integer x = new Integer(temp) ; // 将基本数据类型变为包装类
	在JDK1.5,Java新增了自动装箱和自动拆箱,而且可以直接通过包装类进行四则运算和自增自建操作。例如: 
		Float f = 10.3f ; // 自动装箱 
		float x = f ; // 自动拆箱 
		System.out.println(f * f) ; // 直接利用包装类完成 
		System.out.println(x * x) ; // 直接利用包装类完成

字符串转换

使用包装类还有一个很优秀的地方在于:可以将一个字符串变为指定的基本数据类型,此点一般在接收输入 数据上使用较多。 
在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之后提供了新的 功能,可以根据需要自动传入任意个数的参数。 
	语法:
		返回值类型 方法名称(数据类型…参数名称){ 
			//参数在方法内部 , 以数组的形式来接收 
		} 
	注意:可变参数只能出现在参数列表的最后。
	 	 原因:放在其他参数前面,调用方法时输入参数是按照参数列表顺序输入的,如果有其他参数在可变参数后,将无法输入(因为可变参数可输入 0--n 个参数)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值