Java面向对象

第二章

1、面向对象基础

1.1、面向对象思想:

面向对象(Object Oriented)是软件开发方法。面向对象的概念和应用已超越了程序设计和软件开发,是一种对现 实世界理解和抽象的方法,是计算机编程技术发展到一定阶段后的产物。

面向对象是相对于面向过程来讲的,指的是把 相关的数据和方法组织为一个整体 来看待,从更高的层次来进行系 统建模,更贴近事物的自然运行模式。

面向过程到面向对象思想层面的转变:

面向过程关注的是执行的过程,面向对象关注的是具备功能的对象。

面向过程到面向对象,是程序员思想上 从执行者到指挥者的转变。

此概念如果直接去理解的话可能会比较抽象,因为大家缺少对原始的面向过程的开发语言的了解,下面我们举一个 栗子

1.1.2、例子

问: 把大象装进冰箱 , 需要分几步?

回答:

面向过程回答: 3步:1把冰箱门打开, 2把大象装进去 , 3把冰箱门关闭

面向对象回答: 2步:1招一个能操作冰箱的工人(对象),2指挥工人装大象

思考: 如果问题改成: 把100只大象依次关进冰箱,共分为几步?

面向过程的回答: 此处需要省略N字。。。

面向对象的回答还是2步: 1招一个能操作冰箱的工人(对象) , 2指挥工人把大象依次装进去。

结论: 从上述的栗子中, 我们发现面向过程很死板 ,是很难适应变化的 。 而面向对象更灵活,可复用性更高。

1.1.3、例子2

场景: 当我们独自生活时, 我们经常纠结一日三餐怎么吃。

面向过程: 每天亲力亲为: 买菜 - 做饭 - 吃饭 - 洗碗 的过程。

面向对象: 招聘一个保姆,每天等吃即可。

场景升级:

​ 假设你是一个富豪, 拥有一座占地3000亩地的庄园 ,不再是只关注吃饭问题 , 还有花草树木修剪,泳池维 护清洗,卫生打扫,洗衣做饭。。。。。。

面向过程: 此处省略看着就累的N字。

面向对象: 招聘一个管家, 然后让管家招聘 园丁、泳池维护工、保姆等等。

结论: 从上述的栗子中, 我们发现面向过程,我们需要关注很繁琐的过程 。 而面向对象不用关注具体的细节,更关注的是统筹架构的问题。

其实我们进行大型应用开发时, 就如上述的例子一样, 如果我们写程序只关注过程的话, 代码量达到一定 层次以后, 就很难再编写下去了。

1.1.4、三大思想

面向对象思想从概念上讲分为以下三种:OOA、OOD、OOP

OOA:面向对象分析(Object Oriented Analysis)

OOD:面向对象设计(Object Oriented Design)

OOP:面向对象程序(Object Oriented Programming)

1.1.5、三大特征

封装性:所有的内容对外部不可见

继承性:将其他的功能继承下来继续发展

多态性:方法的重载本身就是一个多态性的体现

1.2、类与对象

1.2.1、两者关系

类表示一个共性的产物,是一个综合的特征,而对象,是一个个性的产物,是一个个体的特征。 (类似生活中的图纸与实物的概念。) 类必须通过对象才可以使用,对象的所有操作都在类中定义。 类由属性和方法组成: · 属性:就相当于人的一个个的特征 · 方法:就相当于人的一个个的行为,例如:说话、吃饭、唱歌、睡觉

1.2.2、类的定义格式
class 类名称{ 

成员属性 

成员方法

 }

code:

/**
 * class 类名{
 * 	成员属性;
 * 	成员方法;
 * }
 * 
 * 类必须编写在。java文件中,。
 * 一个.java文件中, 可以存在N个类, 但是只能存在一个public修饰的类。
 * 。java文件的文件名称 必须 与public修饰的类名 完全一致;
 *
 */
public class Demo {

	public static void main(String[] args) {
		//创建对象的格式:
		//类名 对象名 = new 类名();
		Person p = new Person();
		//对象属性赋值
		//格式:	对象名.属性名=值;
		p.name = "张三";
		p.age = 18;
		p.sex = '男';
		p.say();
		
		
		int s = p.sum(100, 200);
		System.out.println(s);
	}
}
/**
 * 类就像是图纸
 */
class Person{
	//属性 - 特征,品牌等
	String name;
	int age;
	char sex;
	//方法 - 行为
	/**
	 * 定义格式:
	 * 返回值类型 方法名称(形式参数列表){
	 * 	方法体
	 * 	return 返回值;
	 * }
	 * 
	 * 调用格式:
	 * 对象名.方法名称(实际参数列表);
	 */
	
	void say() {
		System.out.println("自我介绍:我是"+name+" , 我的年龄:"+age+",我的性别:"+sex);
	}
	
	int sum(int x,int y) {
		int z = x+y;
		//结束方法
		return z;
	}
	
	/*
	void xxx() {
		if(true) {
			return;
		}
		System.out.println("哈哈");
	}
	*/
	
}
1.2.3、属性与方法
属性定义格式:

 数据类型 属性名;

​	 属性定义并赋值的格式: 

​			数据类型 属性名 = 初始化值;

​			 方法定义格式: 

​				权限修饰符 返回值类型 方法名(形式参数列表){ 

​							//方法体 

​							return 返回值;

​		 }
1.2.4、对象的创建与使用
一个类要想真正的进行操作,则必须依靠对象,对象的定义格式如下:
		类名称 对象名称 = new 类名称() ;
如果要想访问类中的属性或方法(方法的定义),则可以依靠以下的语法形式:
		访问类中的属性: 对象.属性 ;
		调用类中的方法: 对象.方法(实际参数列表) ;

code:

public class Demo4 {

	public static void main(String[] args) {
		//System.out.println("1");
		//创建对象
		Person2 p1 = new Person2("张三");
		//System.out.println("2");
		//p1.name  = "张三";
		p1.age = 18;
		p1.say();
	}

}
class Person2{
	
	Person2(String n){
		name = n;
	}
	
	
	String name;
	int age;
	
	void say() {
		System.out.println("自我介绍。我是:"+name+" , 我的年龄:"+age);
	}
}

1.3、创建对象内存分析

1.3.1、栈
Java栈的区域很小 , 大概2m左右 , 特点是存取的速度特别快
栈存储的特点是, 先进后出

存储速度快的原因:

栈内存, 通过 '栈指针' 来创建空间与释放空间 !

指针向下移动, 会创建新的内存, 向上移动, 会释放这些内存 !

这种方式速度特别快 , 仅次于PC寄存器 !

但是这种移动的方式, 必须要明确移动的大小与范围 ,

明确大小与范围是为了方便指针的移动 , 这是一个对于数据存储的限制, 存储的
数据大小是固定的 , 影响了程序的灵活性 ~

所以我们把更大部分的数据 存储到了堆内存中
存储的是:
基本数据类型的数据 以及 引用数据类型的引用!
例如:
int a =10;
Person p = new Person();
10存储在栈内存中 , 第二句代码创建的对象的引用(p)存在栈内存中
1.3.2、堆
存放的是类的对象 .

Java是一个纯面向对象语言, 限制了对象的创建方式:
所有类的对象都是通过new关键字创建

new关键字, 是指告诉JVM , 需要明确的去创建一个新的对象 , 去开辟一块新的堆内存空间:
堆内存与栈内存不同, 优点在于我们创建对象时,
不必关注堆内存中需要开辟多少存储空间 , 也不需要关注内存占用时长 !

堆内存中内存的释放是由GC(垃圾回收器)完成的
垃圾回收器 回收堆内存的规则:
当栈内存中不存在此对象的引用时,则视其为垃圾 , 等待垃圾回收器回收 !
例如:
Person p0 = new Person();
Person p1 = p0;
Person p2 = new Person();
1.3.3、方法区
存放的是
- 类信息
- 静态的变量
- 常量
- 成员方法
方法区中包含了一个特殊的区域 ( 常量池 )(存储的是使用static修饰的成员)

在这里插入图片描述

1.3.4、PC寄存器
PC寄存器保存的是 当前正在执行的 JVM指令的 地址 !
在Java程序中, 每个线程启动时, 都会创建一个PC寄存器 !
1.3.5、本地方法栈
保存本地(native)方法的地址 

1.4、构造方法(构造器)

1.4.1、回顾对象创建
Person p = new Person();
在右侧Person后面出现的小括号, 其实就是在调用构造方法 
1.4.2、概述
作用:
用于对象初始化。
执行时机:
在创建对象时,自动调用

特点:
所有的Java类中都会至少存在一个构造方法
如果一个类中没有明确的编写构造方法, 则编译器会自动生成一个无参的构造方法, 构造方法中没有任何的代码!

如果自行编写了任意一个构造器, 则编译器不会再自动生成无参的构造方法。

code:

public class Demo4 {

	public static void main(String[] args) {
		//System.out.println("1");
		Person2 p1 = new Person2("张三");
		//System.out.println("2");
		//p1.name  = "张三";
		p1.age = 18;
		p1.say();
	}

}
class Person2{
	//构造方法
	Person2(String n){
		name = n;
	}
	
	
	String name;
	int age;
	
	void say() {
		System.out.println("自我介绍。我是:"+name+" , 我的年龄:"+age);
	}
}
1.4.3、定义格式
定义的格式:
与普通方法基本相同, 区别在于: 方法名称必须与类名相同, 没有返回值类型的声明 !
案例:
public class Demo3{
public static void main(String[] args){
Person p = new Person();
p = new Person();
p = new Person();
p = new Person();
}
}
class Person{
public Person(){
System.out.println("对象创建时,此方法调用");
}
}

1.4.4、构造方法设计
建议自定义无参构造方法,不要对编译器形成依赖,避免错误发生。
当类中有非常量成员变量时,建议提供两个版本的构造方法,一个是无参构造方
法,一个是全属性做参数的构造方法。
当类中所有成员变量都是常量或者没有成员变量时,建议不提供任何版本的构造。

1.5、方法的重载

方法名称相同, 参数类型或参数长度不同, 可以完成方法的重载 ! 方法的重载与返回值无关!
方法的重载 ,可以让我们在不同的需求下, 通过传递不同的参数调用方法来完成具体的功能。

code:

public class Demo5 {

	public static void main(String[] args) {
		Math m = new Math();
		int num = m.sum(100, 500);
		System.out.println(num);
		
		double num2 = m.sum(10.5, 20.6);
		System.out.println(num2);
	}

}
// 命名规范 见名知意
class Math{
	/**
	 * 一个类中定义的方法, 是允许重载 (相同的方法名称)
	 * 
	 * 1、方法名称相同
	 * 2、参数列表长度  或 参数列表类型 或 (参数类型顺序不同)
	 * 
	 * 注意: 与返回值类型无关
	 * 
	 */
	int sum(int x,int y) {
		int z = x+y;
		return z;
	}
	
	double sum(double x,double y) {
		double z = x+y;
		return z;
	}
	
	double sum(int x,double y) {
		double z = x+y;
		return z;
	}
	
	double sum(double y,int x) {
		double z = x+y;
		return z;
	}
	
}

1.6、构造方法的重载

一个类, 可以存在多个构造方法 :
参数列表的长度或类型不同即可完成构造方法的重载 

构造方法的重载 ,可以让我们在不同的创建对象的需求下, 调用不同的方法来完成对象的初始化!

code

public class Demo6 {

	public static void main(String[] args) {
		Person3 p = new Person3("张三",18);
		p.say();
		
		Person3 p2 = new Person3("李四");
		p2.say();
	}

}
class Person3{
	
	Person3(String name2,int age2){
		name = name2;
		age = age2;
	}
	
	Person3(String name2){
		name = name2;
	}
	
	String name;
	int age;
	
	void say() {
		System.out.println("自我介绍: 姓名:"+name+", 年龄:"+age);
	}
	
}

1.7、匿名对象

没有对象名称的对象 就是匿名对象。
匿名对象只能使用一次,因为没有任何的对象引用,所以将称为垃圾,等待被G·C回收。
只使用一次的对象可以通过匿名对象的方式完成,这一点在以后的开发中将经常使用到。
如果一个对象 , 我们准备使用两次 或 以上。 那么一定要给对象 创建对象名。
public class Demo7 {

	/**
	 * 匿名	:	没有名字
	 */
	public static void main(String[] args) {
		int num = new Math2().sum(100, 200);
		System.out.println(num);
	}
}

class Math2{
	
	int sum(int x,int y) {
		return x+y;
	}
}

多属性不能使用匿名对象

2、面向对象进阶

我们观察如下代码:
class Person{
	private String name ; // 表示姓名
	private int age ; // 表示年龄
	void tell(){
		System.out.println("姓名:" + name + ";年龄:" + age) ;
		}
	};
public class Demo{
	public static void main(String args[]){
	Person per = new Person() ;
	per.name = "张三" ;
	per.age = -30 ;
	per.tell() ;
	}
};
以上的操作代码并没有出现了语法错误,但是出现了逻辑错误 (年龄-30岁)
在开发中, 为了避免出现逻辑错误, 我们建议对所有属性进行封装,并为其提供setter及getter方法进行设置和取得
操作。

修改:

/**
 * 封装
 *不合理的程序,修改后的Demo10
 */
public class Demo9 {

	public static void main(String[] args) {
		Person9 p = new Person9();
		//错误:查找不到
		//p.name = "张三";
		//逻辑错误
		//p.age = -10000;
		p.name = "张三";
		p.setAge(-10000);
		p.say();
	}

}

class Person9 {
	//只能当前类使用
	private int age;
	String name;
	
	//先操作不了,在提供可操作的方法
	void setAge(int age2) {
		if(age2 < 0 || age > 150) {
			System.out.println("年龄不合理,设置为1");
			age = 1;
		}else {
			age = age2;
		}
	}
	int getAge() {
		return age;
	}
	
	void say() {
		System.out.println("姓名为:" + name + "年龄为" + age );
	}
}

2.2、this

在Java基础中,this关键字是一个最重要的概念。使用this关键字可以完成以下的操作:
· 调用类中的属性
· 调用类中的方法或构造方法
· 表示当前对象

code:

/**
 * 
 * this关键字
 * 当前属性
 *调用另一个构造方法时,调用的代码,必须编写在构造方法的第一行
 */
public class Demo10 {

	public static void main(String[] args) {
		Person10 p1 = new Person10("张三",18);
		p1.say();
		Person10 p2 = new Person10("李四",18);
		p2.say();
		Person10 p3 = new Person10();
		p3.say();
	}

}
class Person10 {
	private String name;
	private int age;
	
	Person10() {
		this("王五", 8);
	}
	
	Person10(String name,int age){
		this.name = name;
		this.age = age;
	}
	void say() {
		System.out.println(this.name + this.age);
	}
}

2.3、static

static表示“静态”的意思,可以用来修饰成员变量和成员方法(后续还会学习 静态代码块 和 静态内部类)。
static的主要作用在于创建独立于具体对象的域变量或者方法
简单理解:
被static关键字修饰的方法或者变量不需要依赖于对象来进行访问,只要类被加载了,就可以通过类名去进行访
问。
并且不会因为对象的多次创建 而在内存中建立多份数据

重点:

1. 静态成员 在类加载时加载并初始化。
2. 无论一个类存在多少个对象 , 静态的属性, 永远在内存中只有一份( 可以理解为所有对象公用 )
3. 在访问时: 静态不能访问非静态 , 非静态可以访问静态 !

code:

//static关键字
public class Demo1 {

	public static void main(String[] args) {
		//为静态变量赋值
		Student.company = "开课吧";
		Student.subject = "JAVA";
		// 创建对象
		Student s1 = new Student("张三",18,"打篮球");
		Student s2 = new Student("李四",18,"踢足球");
		Student s3 = new Student("王五",18,"跑步");
		Student s4 = new Student("麻子",18,"跳远");
		//调用输出方法
		s1.say();
		s2.say();
		s3.say();
		s4.say();
		

	}

}

class Student {
	String name;
	//封装年龄,防止年龄超出常理值
	private int age;
	String habby;
	static String company;
	static String subject;
	
	//构造方法
	public Student(String name,int age,String habby) {
		this.name = name;
		this.age = age;
		this.habby = habby;
		
	}
	
	//创建输出方法
	void say () {
		System.out.println("学员姓名:" + name + "\t"
				 +"学员年龄:" + age + "\t"
				 + "学员爱好:" + habby + "\t"
				 + "学员公司:" + company + "\t"
				 + "学员学科:" + subject + "\t");
	}
	
}

2.4、代码块

普通代码块
在执行的流程中 出现的 代码块, 我们称其为普通代码块。
构造代码块
在类中的成员代码块, 我们称其为构造代码块, 在每次对象创建时执行, 执行在构造方法之前。
静态代码块
在类中使用static修饰的成员代码块, 我们称其为静态代码块, 在类加载时执行。 每次程序启动到关闭 ,只会
执行一次的代码块。
同步代码块
在后续多线程技术中学习。
面试题:
构造方法 与 构造代码块 以及 静态代码块的执行顺序:
静态代码块 --> 构造代码块 --> 构造方法

2.5、包

2.5.1. 包介绍
1、把功能相似或相关的类或接口组织在同一个包中,方便类的查找和使用。
2、包如同文件夹一样,不同的包中的类的名字是可以相同的,当同时调用两个不同
包中相同类名的类时,应该加上包名加以区别。因此,包可以避免名字冲突。
3、包也限定了访问权限,拥有包访问权限的类才能访问某个包中的类。

2.5.2. 包的使用规则
- 包中java文件的定义:
在.java文件的首部, 必须编写类所属哪个包, 格式:
package 包名;
- 包的定义:
通常由多个单词组成, 所有单词的字母小写, 单词与单词之间使用.隔开 ,一般命名为“com.公司名.项目
名.模块名....”。
规范由来:
由于Java面向对象的特性,每名Java开发人员都可以编写属于自己的Java Package,为了保障每个JavaPackage命名的唯一性,在最新的Java编程规范中,要求开发人员在自己定义的包名前加上唯一的前缀。由于互联网上
的域名称是不会重复的,所以多数开发人员采用自己公司在互联网上的域名称作为自己程序包的唯一前缀。例如:
com.java.xxx
2.5.3 import 关键字

import 包名.类名;

2.6、权限修饰符

在这里插入图片描述

2.7、main方法详解

main()方法一直写到了今天:
public static void main(String args[])
以上的各个参数的含义如下:
· public:表示公共的内容,可以被所有操作所调用
· static:表示方法是静态的,可以由类名称直接调用。java StaticDemo09
· void:表示没有任何的返回值操作
· main:系统规定好的方法名称。如果main写错了或没有,会报错:NoSuchMethodError: main· String[] args:字符串数组,接收参数的
public class StaticDemo10{
public static void main(String args[]){
for(int i=0;i<args.length;i++){
System.out.println(args[i]) ;
}
}
};
】所有的参数在执行类的时候以空格进行分割。
java StaticDemo10 1 2 3 4 5 6 7
但是,如果现在我要输入的是以下几种参数“hello world”、“hello vince”、“hello mjw”。
因为以空格分割,所以以上的三组参数会当做六组参数输入,那么此时如果要想完成有空格的内容输入,则参数需
要使用“"”括起来。
java StaticDemo10 "hello world" "hello vince" "hello mjw

注:可以在dos中尝试

code

public class Demo1 {
	public static void main(String[] args){
		System.out.println();
		for(int i = 0;i < args.lenght;i++){
			System.out.println(args[i]);
		}
	}
}

在txt文件中输入,并用dos启动此java文件

3、面向对象高级

3.1、继承

*继承是java面向对象编程技术的一块基石因为它运行创建分等级层次的类
 * 机场就是子类继承父类的特征和行为,使用子类对象(实例)具有父类的实例域和方法,
 * 或子类和父类继承方法,使得子类具有父类相同的行为
 * 
 * 格式:
 *     class 父类 {
 *     }
 *     
 *     clas 子类 extends 父类{
 *     }
 *
 *继承的限制:
 *java中只有单继承,多重继承,没有多继承(俗称:一个孩子没有两个爸爸)

code:

public class Demo01 {

	public static void main(String[] args) {
		Student s = new Student();
		s.setName("张三");
		s.setAge(18);
		s.say();

	}

}

//人类
class Person {
	private String name;
	private int age;
	String sex;
	
	
	public Person() {
		super();
		// TODO Auto-generated constructor stub
	}
	
	
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}


	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	
	public void say() {
		System.out.println(name + age);
	}
	
}

class Student extends Person {
	
	
}

3.2、super

 *super:
 *		通过super,可以访问父类构造方法
 *		通过super,可以访问父类的数据
 *		通过super,可以访问父类的方法
 *调用super构造方法的代码,必须写在子类构造方法的第一行
 class Student extends Person {
	Student() {
		super("张三",1);
		super.sex = "男";
		super.setAge(10);
	}

this与super同时调用不合乎逻辑,this调用时中间包含super

3.3、重写

* 重写(override)规则:
 * 
 * 1、参数列表必须完全被重写的方法相同;
 * 2、返回类型(void、int等)必须完全与被重写的方法类型相同
 * 3、访问权限不能比父类中北重写的方法的访问权限更低。例如:如果父类的一个方法被声明为publi,子类就不能是protected
 * 4、父类的成员方法只能被它的子类重写
 * 5、声明为static和private的方法不能被重写,但是能够再次声明
 * 
 * 重写(Override)与重载(Overload)的区别
 * 1、发生的位置
 * 		重载:一个类中
 * 		重写:子父类中
 * 2、参数列表限制
 * 		重载:必须不同的
 * 		重写:必须相同的
 * 3、返回值类型
 * 		重载:与返回值类型无关
 * 		重写:与返回值必须一致
 * 4、访问权限
 * 		重载:与访问权限无关
 * 		重写:子的方法权限 必须不能小于必服的方法权限
 * 5、异常处理:
 * 		重载:与异常无关
 * 		重写:异常范围可以更小,但是不能抛出新的异常

code:

public class Demo2 {

	public static void main(String[] args) {
		Student s = new Student();
		s.say();
	}
}

class Person {
	public void say() {
		System.out.println("锄禾日当午,汗滴禾下土");
	}
}

class Student extends Person {
	public void say() {
		System.out.println("床前明月光");
	}
}

3.4、final

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

code:

public class Demo3 {

	public static void main(String[] args) {
		final int a = 10;
		a = 20;//报错

	}

}

3.5、抽象类

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

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

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

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

3.5.2.3、抽象类与普通类区别

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

code

/**
 *抽象类

 */

public class Demo1 {

	public static void main(String[] args) {
		Student a = new Student();
	
		Nurse b = new Nurse();
		

		//抽象类不能创建对象
		//Person p = new Person();
		Person p1 = a;
		Person p2 = b;
		p1.say();
		p2.say();
		
		

	}

}



public abstract class Person {
		
		public abstract void say();
		
		public Person() {
			System.out.println("方法执行");
		}

}


public class Student extends Person{
	
	public void say() {
		System.out.println("我是学生");
	}

}



public class Nurse extends Person {

	@Override//重写
	public void say() {
		
			System.out.println("我是护士");
		

	}

}

3.6、接口

3.6.1、概念
如果一个类中的全部方法都是抽象方法,全部属性都是全局常量,那么此时就可以将
这个类定义成一个接口。
定义格式:
interface 接口名称{
全局常量 ;
抽象方法 ;
}
3.6.2、面向接口编程思想
这种思想是接口是定义(规范,约束)与实现(名实分离的原则)的分离。
优点:
1、 降低程序的耦合性
2、 易于程序的扩展
3、 有利于程序的维护
3.6.3、全局常量和抽象方法的简写
因为接口本身都是由全局常量和抽象方法组成 , 所以接口中的成员定义可以简写:
1、全局常量编写时, 可以省略public static final 关键字,例如:
public static final String INFO = "内容" ;
简写后:
String INFO = "内容" ;
2、抽象方法编写时, 可以省略 public abstract 关键字, 例如:
public abstract void print() ;
简写后:
void print() ;

3.6.4、接口的实现 implements
接口可以多实现:
格式:
class 子类 implements 父接口1,父接口2...{
}

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

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

}
3.6.6、注意
如果一个接口要想使用,必须依靠子类。 子类(如果不是抽象类的话)
要实现接口中的所有抽象方法。
3.6.7、接口和抽象类的区别
1、抽象类要被子类继承,接口要被类实现。
2、接口只能声明抽象方法,抽象类中可以声明抽象方法,也可以写非抽象方法。
3、接口里定义的变量只能是公共的静态的常量,抽象类中的变量是普通变量。
4、抽象类使用继承来使用, 无法多继承。 接口使用实现来使用, 可以多实现
5、抽象类中可以包含static方法 ,但是接口中不允许(静态方法不能被子类重写,
因此接口中不能声明静态方法)
6、接口不能有构造方法,但是抽象类可以有

code:

/**
 * 面向接口类型
 *
 */
public class Demo1 {
	
	//父类引用指向子类对象
	public static void main(String[] args) {
		Student s = new Student();		
		say(s);
		Nurse n = new Nurse();
		say(n);
	}
	
	public static void say(Person p) {
		p.say();
	}
}


public interface Person {
	int a = 10;//默认public static final修饰
	 
	 void say();
}


public class Nurse implements Person{

	@Override
	public void say() {
		System.out.println("后续增加的say");
		
	}	
}


public class Student implements Person{

	@Override
	public void say() {
		System.out.println(1111);
		
	}	
}

4、多态

4.1、概念
多态:就是对象的多种表现形式,(多种体现形态)
4.2、多态的体现
对象的多态性,从概念上非常好理解,在类中有子类和父类之分,子类就是父类的一种形态 ,对象多态性就从此而来。
ps: 方法的重载 和 重写 也是多态的一种, 不过是方法的多态(相同方法名的多种形态)。
重载: 一个类中方法的多态性体现
重写: 子父类中方法的多态性体现。
4.3、多态的使用:对象的类型转换
类似于基本数据类型的转换:
· 向上转型:将子类实例变为父类实例
|- 格式:父类 父类对象 = 子类实例 ;
· 向下转型:将父类实例变为子类实例
|- 格式:子类 子类对象 = (子类)父类实例 ;

code

/**
 * 多态
 *
 */
public class Demo1 {

	public static void main(String[] args) {
		Student a = new Student();
	
		Nurse b = new Nurse();
		
		//抽象类不能创建对象
		//Person p = new Person();
		Person p1 = a;
		Person p2 = b;
		p1.say();
		p2.say();
		
		Student a2 = (Student)p1;
		Student a3 = (Student)p2;
		a2.say();
		a3.say();

	}


public abstract class Person {
		
		public abstract void say();
		
		public Person() {
			System.out.println("方法执行");
		}

}


public class Student extends Person{
	
	public void say() {
		System.out.println("我是学生");
	}

}



public class Nurse extends Person {

	@Override//重写
	public void say() {
		
			System.out.println("我是护士");
		

	}

}

5、instanceof

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

code:

public class Demo2 {

	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("必须传入学生形态,才可以执行");
		}
	}

}

public interface Person {
	int a = 10;//默认public static final修饰
	 
	 void say();
}


public class Nurse implements Person{

	@Override
	public void say() {
		System.out.println("后续增加的say");
		
	}	
}


public class Student implements Person{

	@Override
	public void say() {
		System.out.println(1111);
		
	}	
}

6、Object类

6.1、概念
Object类是所有类的父类(基类),如果一个类没有明确的继承某一个具体的类,
则将默认继承Object类。
例如我们定义一个类:
public class Person{
}
其实它被使用时 是这样的:
public class Person extends Object{
}
6.2、Object的多类
使用Object可以接收任意的引用数据类型
6.3、toString
建议重写Object中的toString方法。 此方法的作用:返回对象的字符串表示形式。
Object的toString方法, 返回对象的内存地址
6.4、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 。

code:

/**
 * object:万物
 *
 */
public class Demo {

	public static void main(String[] args) {
		String text = "123";
		say(text);
		int a = 1;
		say(a);

	}
	
	public static void say(Object o) {
		System.out.println(o);
	}

}


//"=="比较的是内存地址
public class Demo2 {

	public static void main(String[] args) {
		Person p = new Person("张三",18);
		System.out.println(p);
		Person p1 = new Person("张三",19);
		System.out.println(p1);
		//没定义equals方法还是"=="比较
		System.out.println(p1.equals(p));
	}
}



public class Person {
	private String name;
	private int age;
	/**
	 * @return the name
	 */
	public String getName() {
		return name;
	}
	/**
	 * @param name the name to set
	 */
	public void setName(String name) {
		this.name = name;
	}
	/**
	 * @return the age
	 */
	public int getAge() {
		return age;
	}
	/**
	 * @param age the age to set
	 */
	public void setAge(int age) {
		this.age = age;
	}
	public Person() {
		super();
		// TODO Auto-generated constructor stub
	}
	public Person(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}
	
	public String toString() {
		return "这是一个人,这个人的姓名" + name + ",这个人的年龄" + age;
	}
	
	//快捷键生成
	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		Person other = (Person) obj;
		if (age != other.age)
			return false;
		if (name == null) {
			if (other.name != null)
				return false;
		} else if (!name.equals(other.name))
			return false;
		return true;
	}
	
	/*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;
			}else {
				//相同
				return false;
			}
		}else {
			return false;
		}
		
	}*/
	
}

7、内部类

7.1、概念
在Java中,可以将一个类定义在另一个类里面或者一个方法里面,
这样的类称为内部类。
广泛意义上的内部类一般来说包括这四种:
1、成员内部类
2、局部内部类
3、匿名内部类
4、静态内部类
7.2、成员内部类
成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:
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();
7.3、局部内部类
局部内部类是定义在一个方法或者一个作用域里面的类,
它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。
例如:
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修饰符的。
7.4、匿名内部类
匿名内部类由于没有名字,所以它的创建方式有点儿奇怪。创建格式如下:

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

在这里我们看到使用匿名内部类我们必须要继承一个父类或者实现一个接口,
当然也仅能只继承一个父类或者实现一个接口。

同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的
引用。当然这个引用是隐式的。

注意:

在使用匿名内部类的过程中,我们需要注意如下几点:

1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼
得,同时也只能继承一个类或者实现一个接口。

2、匿名内部类中是不能定义构造函数的。

3、匿名内部类中不能存在任何的静态成员变量和静态方法。

4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。

5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

6、只能访问final型的局部变量

code

/*
 * 匿名内部类
 */
public class Demo1 {

	public static void main(String[] args) {
		Person p = new Person() {
			public void say() {
				System.out.println(111);
			}
		};
		haha(p);
	}
	
	public static void haha(Person p) {
		
	}

}

public class Person {

	public Person() {
		System.out.println(111);
	}
	
	void haha() {
		System.out.println(111);
	}

}

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

8、 包装类

8.1、概述
在Java中有一个设计的原则“一切皆对象”,那么这样一来Java中的一些基本的数
据类型,就完全不符合于这种设计思想,因为Java中的八种基本数据类型并不是引
用数据类型,所以Java中为了解决这样的问题,引入了八种基本数据类型的包装类

在这里插入图片描述

以上的八种包装类,可以将基本数据类型按照类的形式进行操作。
但是,以上的八种包装类也是分为两种大的类型的:
· Number:Integer、Short、Long、Double、Float、Byte都是Number的
子类表示是一个数字。
· Object:Character、Boolean都是Object的直接子类。
8.2、装箱和拆箱操作
以下以Integer和Float为例进行操作

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

将一个包装类变为一个基本数据类型,这样的操作称为拆箱操作,因为所有的数值
型的包装类都是Number的子类,Number的类中定义了如下的操作方法,以下的全
部方法都是进行拆箱的操作。

在这里插入图片描述

code:

在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) ; // 直接利用包装类完成
8.3、字符串转换
使用包装类还有一个很优秀的地方在于:可以将一个字符串变为指定的基本数据类型,
此点一般在接收输入数据上使用
较多。
在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
....
....

code:

import java.util.Scanner;

public class Demo2 {

	public static void main(String[] args) {
		Scanner input = new Scanner(System.in);
		System.out.println("请输入内容");
		String text = input.nextLine();
		
		int x = Integer.parseInt(text);
		
		System.out.println(x+1);

	}

}

9、可变参数

一个方法中定义完了参数,则在调用的时候必须传入与其一一对应的参数,但是在
JDK 1.5之后提供了新的功能,可以根据需要自动传入任意个数的参数。
语法:
	返回值类型 方法名称(数据类型…参数名称){
		//参数在方法内部 , 以数组的形式来接收
	}
	
注意:
可变参数只能出现在参数列表的最后。
public class Demo3 {
/**
 * 可变参数
 */
	public static void main(String[] args) {
		System.out.println(sum(1));
		
		/**
		 * int...nums:表示的是可变参数,调用可以传递0-n个数字
		 * 在方法的内部,可变参数以数组作为载体体现
		 */
	}
	public static int su(int...a,String i) {//不能将可变参数放前面
		
	}
		public static int sum(int...nums) {
			int n = 0;
			for(int i = 0;i < nums.length;i++) {
				n += nums[i];
			}
			return n;
		}
	

}

10、递归

递归,在数学与计算机科学中,是指在方法的定义中使用方法自身。也就是说,
递归算法是一种直接或者间接调用自身方法的算法。

递归调用求阶乘code:

public class Demo4 {
/**
 * 阶乘(递归调用
 * @param args
 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(fact(5));

	
}
	private static int fact(int i) {
		if(i == 1) {
			return 1;
		}else {
			return i * fact(i-1);
		}

	}
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值