首页 资讯 农业 汽车 房产 科技 养老 教育 展会 自媒体
智能 互联网 摄影 手机 VR

Java工程师培训课(八)

来源:互联网 作者:高晓娜 人气: 发布时间:2018-12-05


?

序言:

爱之于她,不是肌肤之亲,不是一蔬一饭,它是一种不死的欲望,是颓败生活中的英雄梦想。

?



正文:

static关键字
1:如果没有static会怎样?

1:定义Person类

1:姓名、年龄、国籍,说话行为

2:多个构造,重载形式体现

2:中国人的国籍都是确定的

1:国籍可以进行显示初始化


class?Person {

String name;

int?age;

String gender;

String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender, String country) {

this.name?= name;

this.age?= age;

this.gender?= gender;

this.country?= country;

}

?

void?speak() {

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender?+ " 年龄:"?+ age?+ " 哈哈!!!");

}

?

}

?

3:new Person 对象

1:分析内存

2:每个对象都维护实例变量国籍也是。


public?class?PersonDemo {

public?static?void?main(String[] args) {

Person p1 = new?Person("jack", 20, "男");

p1.speak();

?

Person p2 = new?Person("rose", 18, "女");

p2.speak();

}

}

?

?

4:内存分析

1:栈,堆、共享区

2:Demo.class加载近共享区

1:Demo类的main方法进栈

2:Person p1=new Person();

1:Person.class 加载进方法区

2:堆内存开辟空间,实例变量进行默认初始化,显示初始化。

3:内存地址传给变量p1,栈和堆建立连接

3:person p2=new Person();

1:堆内存开辟空间,实例变量进行默认初始化,显示初始化。

2:内存地址传给变量p2,栈和堆建立连接

4:如果建立多个Person对象发现问题

1:每个对象都维护有国籍。

5:解决问题,内存优化

1:为了让所有Person对象都共享一个country ,可以尝试将country放入共享区。

2:country变量如何放入共享区?对象如何访问?

1:使用static

2:static

1:为了实现对象之间重复属性的数据共享

3:static使用

1:主要用于修饰类的成员

1:成员变量

1:非静态成员变量:需要创建对象来访问

2:静态成员变量:使用类名直接调用,也可以通过对象访问


public?static?void?main(String[] args) {

?

//访问静态成员

//直接通过类名来调用

String country=Person.country;

System.out.println(country);

?

//通过对象.成员的形式访问

Person p1 = new?Person("jack", 20, "男");

p1.country="US";

p1.speak();

?

}

class?Person {

String name;

int?age;

String gender;

//static 修饰成员变量

static?String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

?

}

?

void?speak() {

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

}

?

}

?

2:成员方法

可以使用类名直接调用

1:静态函数:

1:静态函数中不能访问非静态成员变量,只能访问静态变量。

2:静态方法不可以定义this,super关键字.

3:因为静态优先于对象存在.静态方法中更不可以出现this

?2:非静态函数:非静态函数中可以访问静态成员变量


class?Person {

String name;

int?age;

String gender;

//static 修饰成员变量

static?String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

?

}

//非静态方法

void?speak() {

//非静态方法可以访问静态成员

System.out.println("国籍:"?+ country?);

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

?

}

//静态方法

static?void?run(){

//静态方法只能访问静态成员变量。

System.out.println("国籍:"+country);

?

//静态方法访问非静态成员变量,编译报错。

System.out.println(" 姓名:"?+ name);

?

//静态方法中不可以出现this,编译报错

this.speak();

}

}

?

2:细节:

1:静态函数中不能使用非静态变量

2:非静态函数可以访问静态变量

3:为什么静态函数中不能访问非静态成员

1:static修饰的成员在共享区中。优先于对象存在

2:验证

1:使用静态代码块验证

1:静态代码块

static{

静态代码块执行语句;

}

?1:静态代码块特点

随着类的加载而加载。只执行一次,优先于主函数。用于给类进行初始化。

?


public?class?PersonDemo {

public?static?void?main(String[] args) {

?

// 访问静态成员

// 直接通过类名来调用

String country = Person.country;

System.out.println(country);

?

// 通过对象.成员的形式访问

Person p1 = new?Person("jack", 20, "男");

p1.country?= "US";

p1.speak();

?

}

}

?

?

class?Person {

String name;

int?age;

String gender;

// static 修饰成员变量

static?String country?= "CN";

static?{

System.out.println("这是静态代码块");

}

?

{

System.out.println("这是构造代码块");

}

?

Person() {

System.out.println("无参数构造");

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

System.out.println(" 有参数构造");

?

}

?

// 非静态方法

void?speak() {

// 非静态方法可以访问静态成员

System.out.println("国籍:"?+ country);

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

// 非静态方法可以调用静态方法。

run();

}

?

// 静态方法

static?void?run() {

// 静态方法只能访问静态成员变量。

System.out.println("国籍:"?+ country);

}

}

?

?

???

4:static特点

1 随着类的加载而加载,静态会随着类的加载而加载,随着类的消失而消失。说明它的生命周期很长。

?????? 2 优先于对象存在。-->静态是先存在,对象是后存在。

????? 3 被所有实例(对象)所共享。

?????? 4 可以直接被类名调用

5:静态变量(类变量)和实例变量的区别:

???? 1存放位置

???????? 1:类变量随着类的加载而加载存在于方法区中.

????????? 2:实例变量随着对象的建立而存在于堆内存中.

???? 2生命周期

????????? 1:类变量生命周期最长,随着类的消失而消失.

????????? 2:实例变量生命周期随着对象的消失而消失.

6:静态优缺点

1: 优点:对对象的共享数据进行单独空间的存储,节省空间 例如Person 都有

国籍。该数据可以共享可以被类名调

2:缺点:生命周期过长

?????????? 访问出现局限性。(静态只能访问静态)

7: 什么时候定义静态变量

1:静态变量(类变量)当对象中出现共享数据

??例如:学生的学校名称。学校名称可以共享

????????对象的数据要定义为非静态的存放在对内存中(学生的姓名,学生的年龄)

8:什么时候定义静态函数

如果功能内部没有访问到非静态数据(对象的特有数据。那么该功能就可以定义为静态)

9:静态的应用

自定义数组工具类


/*

定义数组工具类

1:定义一个遍历数组的函数

2:定义一个求数组和的功能函数 ?1. 遍历 ?2. 两两相加

3:定义一个获取数组最大值的功能函数

4:定义一个获取数组最大值角标的功能函数

5:定义一个返回指定数在指定数组中包含的角标的功能函数

6:定义一个可以用于排序int数组的函数

1:冒泡

2:选择

?

定义自己的工具类

?

?*/

class?Arrays {

?

private?Arrays() {

?

}

?

// 1:定义一个遍历数组的函数

public?static?void?print(int[] arr) {

for?(int?x = 0; x < arr.length; x++) {

if?(x != (arr.length?- 1)) {

System.out.print(arr[x] + ",");

} else?{

System.out.print(arr[x]);

}

?

}

}

?

// 2:定义一个求数组和的功能函数

public?static?int?getSum(int[] arr) {

int?sum = 0;

for?(int?x = 0; x < arr.length; x++) {

sum += arr[x];

}

return?sum;

}

?

// 3:定义一个获取数组最大值的功能函数

public?static?int?getMax(int[] arr) {

int?max = 0;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[max] < arr[x]) {

max = x;

}

}

return?arr[max];

}

?

// 4:定义一个获取数组最大值角标的功能函数

public?static?int?getIndexMax(int[] arr) {

int?max = 0;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[max] < arr[x]) {

max = x;

}

}

return?max;

}

?

// 5:定义一个返回 指定数在指定数组中包含的角标的功能函数

public?static?int?getIndex(int[] arr, int?src) {

int?index = -1;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[x] == src) {

index = x;

}

}

return?index;

}

?

// 冒泡

public?static?void?test(int[] arr) {

for?(int?x = 0; x < arr.length?- 1; x++) {

if?(arr[x] > arr[x + 1]) {

int?temp = arr[x + 1];

arr[x + 1] = arr[x];

arr[x] = temp;

?

}

}

}

?

// 选择排序

public?static?void?selectSort(int[] arr) {

for?(int?x = 0; x < arr.length?- 1; x++) {

for?(int?y = 1 + x; y < arr.length; y++) {

if?(arr[x] > arr[y]) {

int?temp = arr[y];

arr[y] = arr[x];

arr[x] = temp;

}

}

}

}

?

// 7:定义一个可以将整数数组进行反序的功能函数。

public?static?void?reverseSort(int[] arr) {

int?start = 0;

int?end = arr.length?- 1;

for?(int?x = 0; x < arr.length; x++) {

if?(start < end) {

int?tem = arr[start];

arr[start] = arr[end];

arr[end] = tem;

}

start++;

end--;

}

?

}

?

// 折半查找

public?static?int?halfSearch(int?key, int[] arr) {

int?min = 0;

int?max = arr.length?- 1;

int?mid = 0;

?

while?(min < max) {

mid = (min + max) / 2;

if?(key > arr[mid]) {

min = mid + 1;

} else?if?(key < arr[mid]) {

max = mid - 1;

} else?{

return?mid;

}

}

return?-1;

}

?

}

?

class?Demo6 {

?

public?static?void?main(String[] args) {

int[] arr = { 3, 4, 5, 2, 3, 7, 4 };

Arrays.print(arr);

System.out.println();

Arrays.selectSort(arr);

Arrays.print(arr);

?

}

}

?

练习:统计创建对象的人数


class?Person

{

public?String name;

public?int?age;

static?public?long??all_count;

public?Person(){

????all_count++;

}

public?Person( String name , int?age ){

????all_count++;

this.name?= name;

this.age?= age;

}

// 统计人数的函数

public?long?getCount(){

??return?all_count;

}

// 应该具备找同龄人的功能

public?boolean?isSameAge( Person p1 ){

??return?this.age?== p1.age;

}

}

class?Demo9

{

public?static?void?main(String[] args)

{

Person p1 = new?Person( "jame"?, ?34 );

Person p2 = new?Person( "lucy"?, ?34 );

?

Person p3 = new?Person( "lili"?, ?34 );

Person p4?= new?Person();

System.out.println( p1.getCount() + " "?+ p2.getCount() + " ?"?+ p3.getCount() ?);

System.out.println( p1.isSameAge( p2 ) );

System.out.println( p1.isSameAge( p3 ) );

}

}

?

?

?

Genius夜光

main方法详解

主函数是静态的

public static void main(String[] args){

??????

}

主函数是什么:主函数是一个特殊的函数,作为程序的入口,可以被jvm识别。

主函数的定义:

???????public :代表该函数的访问权限是最大的。

???????static :代表主函数随着类的加载,就已经存在了。

???????void: ?主函数没有具体的返回值

???????main : 不是关键字,是一个特殊的单词可以被jvm识别。

???????(String[] args) 函数的参数,参数类型是一个数组,该数组中的元素是字符串。字符串类型的数组。

???????主函数的格式是固定的:jvm能够识别

???????jvm在调用函数是,传入的是new String[0];

可以在dos窗口中执行 java ?Demo5 ??hello ?world ??给类Demo5的main方法传递2个参数,参数与参数之间通过空格隔开。


class?Demo5 {

?

public?static?void?main(String[] args) {

?

// 获取String[] args?数组长度

System.out.println(args.length);

?

// 变量args数组

for?(int?x = 0; x < args.length; x++) {

System.out.println(args[x]);

}

}

?

}

?

class?MainTest {

?

public?static?void?main(String[] args) {

// 字符串数组

String[] arr = { "good", "study", "java"?};

?

// 调用Demo5类的main方法,传递参数。

Demo5.main(arr);

?

}

}

?

单例设计模式
?

一些人总结出来用来解决特定问题的固定的解决方案。

?

解决一个类在内存中只存在一个对象,想要保证对象的唯一。

1 为了避免其他程序过多的建立该类对象。禁止其他程序建立该类对象。

2 为了其他程序可以访问该类对象,在本类中自定义一个对象。

3 方便其他程序对自定义类的对象的访问,对外提供一些访问方式。

?代码:

1将构造函数私有化

2在类中创建一个私有的本类对象

3提供一个用类名调用的公有方法获取该对象。


class?Single {

?

private?static?Single s?= new?Single(); // 恶汉式

?

private?Single() {

?

}

?

public?static?Single getInstance() {

return?s;

}

}

?

class?Single2 {

private?static?Single2 s?= null; // 懒汉

?

private?Single2() {

?

}

?

public?static?Single2 getInstance() {

if?(s?== null) {

s?= new?Single2();

}

return?s;

}

}

继承
类和类之间的常见关系。

1:既然继承是描述类和类之间的关系,就需要先来了解类和类之间的常见关系



现实生活的整体与部分


举例说明

1:现实生活

1:学生 ??是人??

2:狗 ????是动物

3:球队 包含 球员 整体与部分的关系,部分可以删除和增加

4:笔记本包含 cpu 整体与部分的关系,部分不可以删除和增加

5:航母编队 包含(航母 护卫舰 驱逐舰 舰载机 核潜艇)



java中的类与类关系


java中的类关系

1:is a 关系 (学生是人)

2:has a 整体与部分


class?Person{

String name;

int?age;

Address add;

?

Person(){

?

}

Person(String name,int?age,Address add){

?

this.name=name;

this.age=age;

this.add=add;

?

}

?

void?speak(){

System.out.println("姓名:"+name+" 年龄:"+age+" "+add.print());

}

}

class?Address{

String country;

String city;

String street;

?

Address(){

?

}

Address(String country,String city,String street){

this.country=country;

this.city=city;

this.street=street;

}

?

String print(){

return?"地址:"+country+" "+"城市:"+city+" ?街道;"+street;

}

}

class?Demo3{

?

public?static?void?main(String[] args){

?

Address add=new?Address("中国","广州","棠东东路");

Person p=new?Person("jack",27,add);

p.speak();

?

?

System.out.println();

????? }

}

?

?

继承

1:描述一个学生类

1:姓名年龄学号属性,学习的方法

2:描述一个工人类

1:姓名年龄工号属性,工作的方法

3:描述一个人类

1:姓名年龄属性,说话的方法。

4:发现学生类和人类天生有着联系,学生和工人都是人。所以人有的属性和行为学生和工人都会有。出现类代码重复


class?Person {

String name;

int?age;

?

// 静态变量(类变量)对象和对象之间的代码重复使用静态变量

static?String country?= "CN";

?

Person() {

?

}

?

void?speak() {

System.out.println(name?+ ":哈哈,我是人!!!");

}

?

}

?

// 让学生类和人类产生关系,发现学生is a 人,就可以使用继承

class?Student {

?

String name;

int?age;

?

Student() {

?

}

?

void?study() {

System.out.println("姓名:"?+ name?+ "年纪:"?+ age?+ ":好好学习");

}

}

?

class?Worker {

String name;

int?age;

?

void?work() {

System.out.println(name?+ ":好好工作,好好挣钱。");

}

?

}

?

class?Demo1 {

?

public?static?void?main(String[] args) {

Student s = new?Student();

s.name?= "jack";

s.age?= 20;

s.study();

?

Worker w = new?Worker();

w.name?= "rose";

?

w.work();

}

}

?

5:问题:

1:如果没有继承,出现类和类的关系无法描述

2:如果没有继承,类和类之间有关系会出现类和类的描述代码的重复。

继承特点

1:描述类和类之间的关系

2:降低类和类之间的重复代码

1:降低对象和对象之间的代码重复使用静态变量

2:降低类和类之间的代码重复使用就继承

?

extends关键字

继承使用extends关键字实现

1:发现学生是人,工人是人。显然属于is a 的关系,is a就是继承。

2:谁继承谁?

学生继承人,发现学生里的成员变量,姓名和年龄,人里边也都进行了定义。有重 复代码将学生类的重复代码注释掉,创建学生类对象,仍然可以获取到注释的成员。这就是因为继承的关系,学生类(子类)继承了人类(父类)的部分


class?Person {

String name;

int?age;

?

// 静态变量(类变量)对象和对象之间的代码重复使用静态变量

static?String country?= "CN";

?

Person() {

?

}

?

void?speak() {

System.out.println(name?+ ":哈哈,我是人!!!");

}

?

}

?

// 让学生类和人类产生关系,发现学生is a 人,就可以使用继承

class?Student extends?Person {

?

Student() {

?

}

?

void?study() {

System.out.println("姓名:"?+ name?+ "年纪:"?+ age?+ ":好好学习");

}

}

?

class?Worker extends?Person {

?

void?work() {

System.out.println(name?+ ":好好工作,好好挣钱。");

}

?

}

?

class?Demo1 {

?

public?static?void?main(String[] args) {

Student stu = new?Student();

stu.name?= "jack";

stu.age?= 20;

stu.study();

stu.speak();

System.out.println(stu.country);

System.out.println(Student.country);

?

Worker worker = new?Worker();

worker.name?= "rose";

System.out.println(worker.country);

worker.work();

worker.speak();

?

System.out.println();

}

}

?

继承细节;

1:类名的设定,被继承的类称之为父类(基类),继承的类称之为子类

2:子类并不能继承父类中所有的成员

1:父类定义完整的成员 静态成员,非静态,构造方法。静态变量和静态方

法都可以通过子类名.父类静态成员的形式调用成功。

2:所有的私有成员不能继承,private修饰的成员。

3:构造函数不能被继承

?

3:如何使用继承

1:不要为了使用继承而继承。工人和学生都有共性的成员,不要为了节省代

码,让工人继承学生。


/*

如何使用继承:验证是否有 is ?a 的关系

?例如:学生是人, 小狗是动物

?注意:不要为了使用某些功能而继承,java只支持单继承

?*/

class?DK {

?

void?Ip4S() {

System.out.println("好玩");

}

}

?

class?BGir extends?DK {

?

}

?

class?Demo {

?

public?static?void?main(String[] args) {

?

new?BGir().Ip4S();

?

}

}

?

super关键字

1:定义Father(父类)类

1:成员变量int x=1;

2:构造方法无参的和有参的,有输出语句

2:定义Son类extends Father类

1:成员变量int y=1;

2:构造方法无参和有参的。有输出语句

1:this.y=y+x;

3:创建Son类对象

Son son=new Son(3);

System.out.println(son.y); //4


class?Father {

int?x?= 1;

?

Father() {

System.out.println("这是父类无参构造");

}

?

Father(int?x) {

?

this.x?= x;

System.out.println("这是父类有参构造");

}

?

void?speak() {

System.out.println("我是父亲");

}??????????????????

}

?

class?Son extends?Father {

int?y?= 1;

?

Son() {

System.out.println("这是子类的无参构造");

}

?

Son(int?y) {

?

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

void?run() {

super.speak(); // 访问父类的函数

System.out.println("我是儿子");

}

}

?

class?Demo6 {

?

public?static?void?main(String[] args) {

Son s = new?Son(3);

System.out.println(s.y);// 4

}

}

?

4:子类对象为什么可以访问父类的成员。

1:this.y=y+x;有一个隐式的super super.x

?

5:super关键字作用

1:主要存在于子类方法中,用于指向子类对象中父类对象。

2:访问父类的属性

3:访问父类的函数

4:访问父类的构造函数

6:super注意

this和super很像,this指向的是当前对象的调用,super指向的是当前调用对象的父类。Demo类被加载,执行main方法,Son.class加载,发现有父类Father类,于是Father类也加载进内存。类加载完毕,创建对象,父类的构造方法会被调用(默认自动无参),然后执行子类相应构造创建了一个子类对象,该子类对象还包含了一个父类对象。该父类对象在子类对象内部。this super只能在有对象的前提下使用,不能在静态上下文使用。

2:子类的构造函数默认第一行会默认调用父类无参的构造函数,隐式语句

super();

1:父类无参构造函数不存在,编译报错。

???


Son(int?y) {

//super();隐式语句

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

3:子类显式调用父类构造函数

在子类构造函数第一行通过super关键字调用父类任何构造函数。如果显式调用父类构造函数,编译器自动添加的调用父类无参数的构造就消失。构造函数间的调用只能放在第一行,只能调用一次。super() 和this()不能同时存在构造函数第一行。


Son(int?y) {

super(y);//?子类显式调用父类构造函数

this.y?= y?+ x;

System.out.println("这是子类的有参构造");

}

?


Son(int?y) {

this();??//不能同时存在构造函数第一行??

super(y);

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

4:super思考

如果开发者自定义了一个类,没有显示的进行类的继承,那么该类中成员函数是否可以使用super关健健字?可以使用,继承了Object类,Object类是所有类的父类。


class?Demo7 {

public??void?print(){

System.out.println(super.toString());

}

public?static?void?main(String[] args){

new?Demo7().print();

System.out.println();

????? }

}

?

5:继承练习

?

7:重写(Override)

1:定义Father类

1:姓名,吃饭方法,吃窝窝头。

2:定义Son类,继承Father

1:Son类中不定义任何成员,子类创建对象,仍然可以调用吃饭的方法。

2:父类的吃饭的方法,Son不愿吃。Son自己定义了吃饭的方法。

1:此时父类中有一个吃饭的方法,子类中有2个吃饭的方法,一模一样,只是方法体不一样。

2:一个类中两个函数一模一样,是不允许的。

1:编译运行,执行了子类的方法。

2:使用父类的方法,在子类方法中,使用super.父类方法名。


class?Father {

String name;

?

void?eat() {

System.out.println("吃窝窝");

}

}

?

class?Son extends?Father {

?

public?void?eat() { // 继承可以使得子类增强父类的方法

System.out.println("来俩小菜");

System.out.println("来两杯");

System.out.println("吃香喝辣");

System.out.println("来一根");

}

}

?

class?Demo8 {

?

public?static?void?main(String[] args) {

Son s = new?Son();

????????//执行子类的方法

s.eat();

?

}

}

?

3:该现象就叫做重写(覆盖 override)

1: 在继承中,子类可以定义和父类相同的名称且参数列表一致的函数,将这种函数

称之为函数的重写.

4:前提

1:必须要有继承关系

5:特点

1:当子类重写了父类的函数,那么子类的对象如果调用该函数,一定调用的是重写过后的函数。

???可以通过super关键字进行父类的重写函数的调用。

2: 继承可以使得子类增强父类的方法

6:细节

1: 函数名必须相同

2:参数列表必须相同

3: 子类重写父类的函数的时候,函数的访问权限必须大于等于父类的函数的访

问权限否则编译报错

4:子类重写父类的函数的时候,返回值类型必须是父类函数的返回值类型或该返回值类型的子类。不能返回比父类更大的数据类型: 如子类函数返回值类型是Object

?

1:定义 A B ?C 类 B extends A ?

2:Father类中定义A getA();

3:Son 类中重写getA(); 方法,尝试将返回值修改为B,C ,Object

1:B编译通过

2:C 编译失败?,没有继承关系

3:Object编译失败,比父类的返回值类型更大


class?A {

?

}

?

class?B extends?A {

?

}

?

class?C {

?

}

class?Father {

String name;

?

void?eat() {

System.out.println("吃窝窝");

}

?

// 定义一个函数,获取A类的对象,

A getA() {

return?new?A();

}

?

}

?

class?Son extends?Father {

?

public?void?eat() { // 继承可以使得子类增强父类的方法

System.out.println("来两杯");

System.out.println("来俩小菜");

super.eat();

System.out.println("来一根");

}

?

// B类是A类的子类

B getA() {

return?new?B();

}

}

?

class?Demo8 {

?

public?static?void?main(String[] args) {

Son s = new?Son();

s.eat();

?

}

}

?

7:子类对象查找属性或方法时的顺序:

1:原则:就近原则。

如果子类的对象调用方法,默认先使用this进行查找,如果当前对象没有找到属性或方法,找当前对象中维护的super关键字指向的对象,如果还没有找到编译报错,找到直接调用。

8:重载和重写的不同

1:重载(overload): ?

1:前提: 所有的重载函数必须在同一个类中

????????2:特点:

???????函数名相同,参数列表不同,与其他的无关(访问控制符、返回值类型)

3:不同:

个数不同 、 顺序不同、 类型不同

2:重写(override):

?????1:前提: 继承

2:特点:

??????函数名必须相同、参数列表必须相同。

??子类的返回值类型要等于或者小于父类的返回值

9:重写练习

描述不同的动物不同的叫法

1:定义动物类

有名字,有吃和叫的方法

2:定义狗继承动物重写父类吃和叫的方法

3:定义猫继承动物重写父类吃和叫的方法


class?Animal{

int?x=1;

String name;

?

void?eat(){

System.out.println("吃东西");

}

void?shout(){

System.out.println("我是动物");

}

}

?

class?Dog extends?Animal{

?

?

void?eat(){

System.out.println("啃骨头");

}

void?shout(){

System.out.println("旺旺");

}

void?eat(String food){

System.out.println("吃:"+food);

}

}

class?Cat extends?Animal{

?

void?eat(){

System.out.println("吃老鼠");

}

void?shout(){

System.out.println("喵喵");

}

}

?

class?Demo9{

?

public?static?void?main(String[] args){

Dog d=new?Dog();

d.shout();

d.eat();

?

Cat c=new?Cat();

c.shout();

c.eat();

System.out.println();

????? }

}

?

instanceof 关键字

1:快速演示instanceof


Person?p=new?Person();

System.out.println( p instanceof?Person);

?

2:instanceof是什么?

1:属于比较运算符:

? 2:instanceof关键字:该关键字用来判断一个对象是否是指定类的对象。

? 3:用法:

?????? 对象 ?instanceof 类; ??

?? 该表达式是一个比较运算符,返回的结果是boolea类型 ?true|false

注意:使用instanceof关键字做判断时,两个类之间必须有关系。

3:案例

定义一个功能表函数,根据传递进来的对象的做不同的事情,如果是狗让其看家,如果是猫让其抓老鼠

1:定义动物类

2:定义狗类继承动物类

3:定义猫类继承动物类

4:定义功能根据传入的动物,执行具体的功能

5:instanceof好处

1:可以判断对象是否是某一个类的实例


package?oop01;

?

/*

?instanceof

?比较运算符

?检查是否是类的对象

? 1:可以判断对象是否是某一个类的实例

? 用法

? 对象 ?instanceof?类;

?

?案例

定义一个功能函数,根据传递进来的对象的做不同的事情

如果是狗让其看家,如果是猫让其抓老鼠

1:定义动物类

2:定义狗类继承动物类

3:定义猫类继承动物类

4:定义功能根据传入的动物,执行具体的功能

?*/

?

class?Animal {

?

String name;

?

void?eat() {

System.out.println("吃东西");

}

?

void?shout() {

System.out.println("我是动物");

}

}

?

class?Dog extends?Animal {

?

void?eat() {

System.out.println("啃骨头");

}

?

void?shout() {

System.out.println("旺旺");

}

?

}

?

class?Cat extends?Animal {

?

void?eat() {

System.out.println("吃老鼠");

}

?

void?shout() {

System.out.println("喵喵");

}

}

?

class?Demo11 {

?

public?static?void?main(String[] args) {

?

Demo11 d = new?Demo11();

?

// 对象 instanceof?类;

System.out.println(d instanceof?Demo11);

?

?d.doSomething(new Dog());

d.doSomething(new?Cat());

}

?

// 定义一个功能函数,根据传递进来的对象的做不同的事情

// 如果是狗让其看家,如果是猫让其抓老鼠

// 对象 instanceof?类;

void?doSomething(Animal a) {

if?(a instanceof?Dog) {

a.eat();

a.shout();

System.out.println("小狗看家");

} else?if?(a instanceof?Cat) {

a.eat();

a.shout();

System.out.println("抓老鼠");

}

}

}

练习:


byte[] bs = new?byte[] { 1, 2, 3 };

int[] is = new?int[] { 1, 2, 3 };

String[] ss = new?String[] { "jack", "lucy", "lili"?};

System.out.println(bs instanceof?byte[]); // true

System.out.println(is instanceof?int[]); // true

System.out.println(ss instanceof?String[]); // true

// System.out.println(bs?instanceof?int[]); // 不可转换的类型

?

static关键字
1:如果没有static会怎样?

1:定义Person类

1:姓名、年龄、国籍,说话行为

2:多个构造,重载形式体现

2:中国人的国籍都是确定的

1:国籍可以进行显示初始化


class?Person {

String name;

int?age;

String gender;

String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender, String country) {

this.name?= name;

this.age?= age;

this.gender?= gender;

this.country?= country;

}

?

void?speak() {

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender?+ " 年龄:"?+ age?+ " 哈哈!!!");

}

?

}

?

3:new Person 对象

1:分析内存

2:每个对象都维护实例变量国籍也是。


public?class?PersonDemo {

public?static?void?main(String[] args) {

Person p1 = new?Person("jack", 20, "男");

p1.speak();

?

Person p2 = new?Person("rose", 18, "女");

p2.speak();

}

}

?

?

4:内存分析

1:栈,堆、共享区

2:Demo.class加载近共享区

1:Demo类的main方法进栈

2:Person p1=new Person();

1:Person.class 加载进方法区

2:堆内存开辟空间,实例变量进行默认初始化,显示初始化。

3:内存地址传给变量p1,栈和堆建立连接

3:person p2=new Person();

1:堆内存开辟空间,实例变量进行默认初始化,显示初始化。

2:内存地址传给变量p2,栈和堆建立连接

4:如果建立多个Person对象发现问题

1:每个对象都维护有国籍。

5:解决问题,内存优化

1:为了让所有Person对象都共享一个country ,可以尝试将country放入共享区。

2:country变量如何放入共享区?对象如何访问?

1:使用static

2:static

1:为了实现对象之间重复属性的数据共享

3:static使用

1:主要用于修饰类的成员

1:成员变量

1:非静态成员变量:需要创建对象来访问

2:静态成员变量:使用类名直接调用,也可以通过对象访问


public?static?void?main(String[] args) {

?

//访问静态成员

//直接通过类名来调用

String country=Person.country;

System.out.println(country);

?

//通过对象.成员的形式访问

Person p1 = new?Person("jack", 20, "男");

p1.country="US";

p1.speak();

?

}

class?Person {

String name;

int?age;

String gender;

//static 修饰成员变量

static?String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

?

}

?

void?speak() {

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

}

?

}

?

2:成员方法

可以使用类名直接调用

1:静态函数:

1:静态函数中不能访问非静态成员变量,只能访问静态变量。

2:静态方法不可以定义this,super关键字.

3:因为静态优先于对象存在.静态方法中更不可以出现this

?2:非静态函数:非静态函数中可以访问静态成员变量


class?Person {

String name;

int?age;

String gender;

//static 修饰成员变量

static?String country?= "CN";

?

Person() {

?

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

?

}

//非静态方法

void?speak() {

//非静态方法可以访问静态成员

System.out.println("国籍:"?+ country?);

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

?

}

//静态方法

static?void?run(){

//静态方法只能访问静态成员变量。

System.out.println("国籍:"+country);

?

//静态方法访问非静态成员变量,编译报错。

System.out.println(" 姓名:"?+ name);

?

//静态方法中不可以出现this,编译报错

this.speak();

}

}

?

2:细节:

1:静态函数中不能使用非静态变量

2:非静态函数可以访问静态变量

3:为什么静态函数中不能访问非静态成员

1:static修饰的成员在共享区中。优先于对象存在

2:验证

1:使用静态代码块验证

1:静态代码块

static{

静态代码块执行语句;

}

?1:静态代码块特点

随着类的加载而加载。只执行一次,优先于主函数。用于给类进行初始化。

?


public?class?PersonDemo {

public?static?void?main(String[] args) {

?

// 访问静态成员

// 直接通过类名来调用

String country = Person.country;

System.out.println(country);

?

// 通过对象.成员的形式访问

Person p1 = new?Person("jack", 20, "男");

p1.country?= "US";

p1.speak();

?

}

}

?

?

class?Person {

String name;

int?age;

String gender;

// static 修饰成员变量

static?String country?= "CN";

static?{

System.out.println("这是静态代码块");

}

?

{

System.out.println("这是构造代码块");

}

?

Person() {

System.out.println("无参数构造");

}

?

Person(String name, int?age, String gender) {

this.name?= name;

this.age?= age;

this.gender?= gender;

System.out.println(" 有参数构造");

?

}

?

// 非静态方法

void?speak() {

// 非静态方法可以访问静态成员

System.out.println("国籍:"?+ country);

?

System.out.println("国籍:"?+ country?+ " 姓名:"?+ name?+ " 性别:"?+ gender

+ " 年龄:"?+ age?+ " 哈哈!!!");

// 非静态方法可以调用静态方法。

run();

}

?

// 静态方法

static?void?run() {

// 静态方法只能访问静态成员变量。

System.out.println("国籍:"?+ country);

}

}

?

?

???

4:static特点

1 随着类的加载而加载,静态会随着类的加载而加载,随着类的消失而消失。说明它的生命周期很长。

?????? 2 优先于对象存在。-->静态是先存在,对象是后存在。

????? 3 被所有实例(对象)所共享。

?????? 4 可以直接被类名调用

5:静态变量(类变量)和实例变量的区别:

???? 1存放位置

???????? 1:类变量随着类的加载而加载存在于方法区中.

????????? 2:实例变量随着对象的建立而存在于堆内存中.

???? 2生命周期

????????? 1:类变量生命周期最长,随着类的消失而消失.

????????? 2:实例变量生命周期随着对象的消失而消失.

6:静态优缺点

1: 优点:对对象的共享数据进行单独空间的存储,节省空间 例如Person 都有

国籍。该数据可以共享可以被类名调

2:缺点:生命周期过长

?????????? 访问出现局限性。(静态只能访问静态)

7: 什么时候定义静态变量

1:静态变量(类变量)当对象中出现共享数据

??例如:学生的学校名称。学校名称可以共享

????????对象的数据要定义为非静态的存放在对内存中(学生的姓名,学生的年龄)

8:什么时候定义静态函数

如果功能内部没有访问到非静态数据(对象的特有数据。那么该功能就可以定义为静态)

9:静态的应用

自定义数组工具类


/*

定义数组工具类

1:定义一个遍历数组的函数

2:定义一个求数组和的功能函数 ?1. 遍历 ?2. 两两相加

3:定义一个获取数组最大值的功能函数

4:定义一个获取数组最大值角标的功能函数

5:定义一个返回指定数在指定数组中包含的角标的功能函数

6:定义一个可以用于排序int数组的函数

1:冒泡

2:选择

?

定义自己的工具类

?

?*/

class?Arrays {

?

private?Arrays() {

?

}

?

// 1:定义一个遍历数组的函数

public?static?void?print(int[] arr) {

for?(int?x = 0; x < arr.length; x++) {

if?(x != (arr.length?- 1)) {

System.out.print(arr[x] + ",");

} else?{

System.out.print(arr[x]);

}

?

}

}

?

// 2:定义一个求数组和的功能函数

public?static?int?getSum(int[] arr) {

int?sum = 0;

for?(int?x = 0; x < arr.length; x++) {

sum += arr[x];

}

return?sum;

}

?

// 3:定义一个获取数组最大值的功能函数

public?static?int?getMax(int[] arr) {

int?max = 0;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[max] < arr[x]) {

max = x;

}

}

return?arr[max];

}

?

// 4:定义一个获取数组最大值角标的功能函数

public?static?int?getIndexMax(int[] arr) {

int?max = 0;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[max] < arr[x]) {

max = x;

}

}

return?max;

}

?

// 5:定义一个返回 指定数在指定数组中包含的角标的功能函数

public?static?int?getIndex(int[] arr, int?src) {

int?index = -1;

for?(int?x = 0; x < arr.length; x++) {

if?(arr[x] == src) {

index = x;

}

}

return?index;

}

?

// 冒泡

public?static?void?test(int[] arr) {

for?(int?x = 0; x < arr.length?- 1; x++) {

if?(arr[x] > arr[x + 1]) {

int?temp = arr[x + 1];

arr[x + 1] = arr[x];

arr[x] = temp;

?

}

}

}

?

// 选择排序

public?static?void?selectSort(int[] arr) {

for?(int?x = 0; x < arr.length?- 1; x++) {

for?(int?y = 1 + x; y < arr.length; y++) {

if?(arr[x] > arr[y]) {

int?temp = arr[y];

arr[y] = arr[x];

arr[x] = temp;

}

}

}

}

?

// 7:定义一个可以将整数数组进行反序的功能函数。

public?static?void?reverseSort(int[] arr) {

int?start = 0;

int?end = arr.length?- 1;

for?(int?x = 0; x < arr.length; x++) {

if?(start < end) {

int?tem = arr[start];

arr[start] = arr[end];

arr[end] = tem;

}

start++;

end--;

}

?

}

?

// 折半查找

public?static?int?halfSearch(int?key, int[] arr) {

int?min = 0;

int?max = arr.length?- 1;

int?mid = 0;

?

while?(min < max) {

mid = (min + max) / 2;

if?(key > arr[mid]) {

min = mid + 1;

} else?if?(key < arr[mid]) {

max = mid - 1;

} else?{

return?mid;

}

}

return?-1;

}

?

}

?

class?Demo6 {

?

public?static?void?main(String[] args) {

int[] arr = { 3, 4, 5, 2, 3, 7, 4 };

Arrays.print(arr);

System.out.println();

Arrays.selectSort(arr);

Arrays.print(arr);

?

}

}

?

练习:统计创建对象的人数


class?Person

{

public?String name;

public?int?age;

static?public?long??all_count;

public?Person(){

????all_count++;

}

public?Person( String name , int?age ){

????all_count++;

this.name?= name;

this.age?= age;

}

// 统计人数的函数

public?long?getCount(){

??return?all_count;

}

// 应该具备找同龄人的功能

public?boolean?isSameAge( Person p1 ){

??return?this.age?== p1.age;

}

}

class?Demo9

{

public?static?void?main(String[] args)

{

Person p1 = new?Person( "jame"?, ?34 );

Person p2 = new?Person( "lucy"?, ?34 );

?

Person p3 = new?Person( "lili"?, ?34 );

Person p4?= new?Person();

System.out.println( p1.getCount() + " "?+ p2.getCount() + " ?"?+ p3.getCount() ?);

System.out.println( p1.isSameAge( p2 ) );

System.out.println( p1.isSameAge( p3 ) );

}

}

?

?

?

?

main方法详解

主函数是静态的

public static void main(String[] args){

??????

}

主函数是什么:主函数是一个特殊的函数,作为程序的入口,可以被jvm识别。

主函数的定义:

???????public :代表该函数的访问权限是最大的。

???????static :代表主函数随着类的加载,就已经存在了。

???????void: ?主函数没有具体的返回值

???????main : 不是关键字,是一个特殊的单词可以被jvm识别。

???????(String[] args) 函数的参数,参数类型是一个数组,该数组中的元素是字符串。字符串类型的数组。

???????主函数的格式是固定的:jvm能够识别

???????jvm在调用函数是,传入的是new String[0];

可以在dos窗口中执行 java ?Demo5 ??hello ?world ??给类Demo5的main方法传递2个参数,参数与参数之间通过空格隔开。


class?Demo5 {

?

public?static?void?main(String[] args) {

?

// 获取String[] args?数组长度

System.out.println(args.length);

?

// 变量args数组

for?(int?x = 0; x < args.length; x++) {

System.out.println(args[x]);

}

}

?

}

?

class?MainTest {

?

public?static?void?main(String[] args) {

// 字符串数组

String[] arr = { "good", "study", "java"?};

?

// 调用Demo5类的main方法,传递参数。

Demo5.main(arr);

?

}

}

?

单例设计模式
?

一些人总结出来用来解决特定问题的固定的解决方案。

?

解决一个类在内存中只存在一个对象,想要保证对象的唯一。

1 为了避免其他程序过多的建立该类对象。禁止其他程序建立该类对象。

2 为了其他程序可以访问该类对象,在本类中自定义一个对象。

3 方便其他程序对自定义类的对象的访问,对外提供一些访问方式。

?代码:

1将构造函数私有化

2在类中创建一个私有的本类对象

3提供一个用类名调用的公有方法获取该对象。


class?Single {

?

private?static?Single s?= new?Single(); // 恶汉式

?

private?Single() {

?

}

?

public?static?Single getInstance() {

return?s;

}

}

?

class?Single2 {

private?static?Single2 s?= null; // 懒汉

?

private?Single2() {

?

}

?

public?static?Single2 getInstance() {

if?(s?== null) {

s?= new?Single2();

}

return?s;

}

}

继承
类和类之间的常见关系。

1:既然继承是描述类和类之间的关系,就需要先来了解类和类之间的常见关系



现实生活的整体与部分


举例说明

1:现实生活

1:学生 ??是人??

2:狗 ????是动物

3:球队 包含 球员 整体与部分的关系,部分可以删除和增加

4:笔记本包含 cpu 整体与部分的关系,部分不可以删除和增加

5:航母编队 包含(航母 护卫舰 驱逐舰 舰载机 核潜艇)



java中的类与类关系


java中的类关系

1:is a 关系 (学生是人)

2:has a 整体与部分


class?Person{

String name;

int?age;

Address add;

?

Person(){

?

}

Person(String name,int?age,Address add){

?

this.name=name;

this.age=age;

this.add=add;

?

}

?

void?speak(){

System.out.println("姓名:"+name+" 年龄:"+age+" "+add.print());

}

}

class?Address{

String country;

String city;

String street;

?

Address(){

?

}

Address(String country,String city,String street){

this.country=country;

this.city=city;

this.street=street;

}

?

String print(){

return?"地址:"+country+" "+"城市:"+city+" ?街道;"+street;

}

}

class?Demo3{

?

public?static?void?main(String[] args){

?

Address add=new?Address("中国","广州","棠东东路");

Person p=new?Person("jack",27,add);

p.speak();

?

?

System.out.println();

????? }

}

?

?

继承

1:描述一个学生类

1:姓名年龄学号属性,学习的方法

2:描述一个工人类

1:姓名年龄工号属性,工作的方法

3:描述一个人类

1:姓名年龄属性,说话的方法。

4:发现学生类和人类天生有着联系,学生和工人都是人。所以人有的属性和行为学生和工人都会有。出现类代码重复


class?Person {

String name;

int?age;

?

// 静态变量(类变量)对象和对象之间的代码重复使用静态变量

static?String country?= "CN";

?

Person() {

?

}

?

void?speak() {

System.out.println(name?+ ":哈哈,我是人!!!");

}

?

}

?

// 让学生类和人类产生关系,发现学生is a 人,就可以使用继承

class?Student {

?

String name;

int?age;

?

Student() {

?

}

?

void?study() {

System.out.println("姓名:"?+ name?+ "年纪:"?+ age?+ ":好好学习");

}

}

?

class?Worker {

String name;

int?age;

?

void?work() {

System.out.println(name?+ ":好好工作,好好挣钱。");

}

?

}

?

class?Demo1 {

?

public?static?void?main(String[] args) {

Student s = new?Student();

s.name?= "jack";

s.age?= 20;

s.study();

?

Worker w = new?Worker();

w.name?= "rose";

?

w.work();

}

}

?

5:问题:

1:如果没有继承,出现类和类的关系无法描述

2:如果没有继承,类和类之间有关系会出现类和类的描述代码的重复。

继承特点

1:描述类和类之间的关系

2:降低类和类之间的重复代码

1:降低对象和对象之间的代码重复使用静态变量

2:降低类和类之间的代码重复使用就继承

?

extends关键字

继承使用extends关键字实现

1:发现学生是人,工人是人。显然属于is a 的关系,is a就是继承。

2:谁继承谁?

学生继承人,发现学生里的成员变量,姓名和年龄,人里边也都进行了定义。有重 复代码将学生类的重复代码注释掉,创建学生类对象,仍然可以获取到注释的成员。这就是因为继承的关系,学生类(子类)继承了人类(父类)的部分


class?Person {

String name;

int?age;

?

// 静态变量(类变量)对象和对象之间的代码重复使用静态变量

static?String country?= "CN";

?

Person() {

?

}

?

void?speak() {

System.out.println(name?+ ":哈哈,我是人!!!");

}

?

}

?

// 让学生类和人类产生关系,发现学生is a 人,就可以使用继承

class?Student extends?Person {

?

Student() {

?

}

?

void?study() {

System.out.println("姓名:"?+ name?+ "年纪:"?+ age?+ ":好好学习");

}

}

?

class?Worker extends?Person {

?

void?work() {

System.out.println(name?+ ":好好工作,好好挣钱。");

}

?

}

?

class?Demo1 {

?

public?static?void?main(String[] args) {

Student stu = new?Student();

stu.name?= "jack";

stu.age?= 20;

stu.study();

stu.speak();

System.out.println(stu.country);

System.out.println(Student.country);

?

Worker worker = new?Worker();

worker.name?= "rose";

System.out.println(worker.country);

worker.work();

worker.speak();

?

System.out.println();

}

}

?

继承细节;

1:类名的设定,被继承的类称之为父类(基类),继承的类称之为子类

2:子类并不能继承父类中所有的成员

1:父类定义完整的成员 静态成员,非静态,构造方法。静态变量和静态方

法都可以通过子类名.父类静态成员的形式调用成功。

2:所有的私有成员不能继承,private修饰的成员。

3:构造函数不能被继承

?

3:如何使用继承

1:不要为了使用继承而继承。工人和学生都有共性的成员,不要为了节省代

码,让工人继承学生。


/*

如何使用继承:验证是否有 is ?a 的关系

?例如:学生是人, 小狗是动物

?注意:不要为了使用某些功能而继承,java只支持单继承

?*/

class?DK {

?

void?Ip4S() {

System.out.println("好玩");

}

}

?

class?BGir extends?DK {

?

}

?

class?Demo {

?

public?static?void?main(String[] args) {

?

new?BGir().Ip4S();

?

}

}

?

super关键字

1:定义Father(父类)类

1:成员变量int x=1;

2:构造方法无参的和有参的,有输出语句

2:定义Son类extends Father类

1:成员变量int y=1;

2:构造方法无参和有参的。有输出语句

1:this.y=y+x;

3:创建Son类对象

Son son=new Son(3);

System.out.println(son.y); //4


class?Father {

int?x?= 1;

?

Father() {

System.out.println("这是父类无参构造");

}

?

Father(int?x) {

?

this.x?= x;

System.out.println("这是父类有参构造");

}

?

void?speak() {

System.out.println("我是父亲");

}??????????????????

}

?

class?Son extends?Father {

int?y?= 1;

?

Son() {

System.out.println("这是子类的无参构造");

}

?

Son(int?y) {

?

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

void?run() {

super.speak(); // 访问父类的函数

System.out.println("我是儿子");

}

}

?

class?Demo6 {

?

public?static?void?main(String[] args) {

Son s = new?Son(3);

System.out.println(s.y);// 4

}

}

?

4:子类对象为什么可以访问父类的成员。

1:this.y=y+x;有一个隐式的super super.x

?

5:super关键字作用

1:主要存在于子类方法中,用于指向子类对象中父类对象。

2:访问父类的属性

3:访问父类的函数

4:访问父类的构造函数

6:super注意

this和super很像,this指向的是当前对象的调用,super指向的是当前调用对象的父类。Demo类被加载,执行main方法,Son.class加载,发现有父类Father类,于是Father类也加载进内存。类加载完毕,创建对象,父类的构造方法会被调用(默认自动无参),然后执行子类相应构造创建了一个子类对象,该子类对象还包含了一个父类对象。该父类对象在子类对象内部。this super只能在有对象的前提下使用,不能在静态上下文使用。

2:子类的构造函数默认第一行会默认调用父类无参的构造函数,隐式语句

super();

1:父类无参构造函数不存在,编译报错。

???


Son(int?y) {

//super();隐式语句

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

3:子类显式调用父类构造函数

在子类构造函数第一行通过super关键字调用父类任何构造函数。如果显式调用父类构造函数,编译器自动添加的调用父类无参数的构造就消失。构造函数间的调用只能放在第一行,只能调用一次。super() 和this()不能同时存在构造函数第一行。


Son(int?y) {

super(y);//?子类显式调用父类构造函数

this.y?= y?+ x;

System.out.println("这是子类的有参构造");

}

?


Son(int?y) {

this();??//不能同时存在构造函数第一行??

super(y);

this.y?= y + x;

System.out.println("这是子类的有参构造");

}

?

4:super思考

如果开发者自定义了一个类,没有显示的进行类的继承,那么该类中成员函数是否可以使用super关健健字?可以使用,继承了Object类,Object类是所有类的父类。


class?Demo7 {

public??void?print(){

System.out.println(super.toString());

}

public?static?void?main(String[] args){

new?Demo7().print();

System.out.println();

????? }

}

?

5:继承练习

?

7:重写(Override)

1:定义Father类

1:姓名,吃饭方法,吃窝窝头。

2:定义Son类,继承Father

1:Son类中不定义任何成员,子类创建对象,仍然可以调用吃饭的方法。

2:父类的吃饭的方法,Son不愿吃。Son自己定义了吃饭的方法。

1:此时父类中有一个吃饭的方法,子类中有2个吃饭的方法,一模一样,只是方法体不一样。

2:一个类中两个函数一模一样,是不允许的。

1:编译运行,执行了子类的方法。

2:使用父类的方法,在子类方法中,使用super.父类方法名。


class?Father {

String name;

?

void?eat() {

System.out.println("吃窝窝");

}

}

?

class?Son extends?Father {

?

public?void?eat() { // 继承可以使得子类增强父类的方法

System.out.println("来俩小菜");

System.out.println("来两杯");

System.out.println("吃香喝辣");

System.out.println("来一根");

}

}

?

class?Demo8 {

?

public?static?void?main(String[] args) {

Son s = new?Son();

????????//执行子类的方法

s.eat();

?

}

}

?

3:该现象就叫做重写(覆盖 override)

1: 在继承中,子类可以定义和父类相同的名称且参数列表一致的函数,将这种函数

称之为函数的重写.

4:前提

1:必须要有继承关系

5:特点

1:当子类重写了父类的函数,那么子类的对象如果调用该函数,一定调用的是重写过后的函数。

???可以通过super关键字进行父类的重写函数的调用。

2: 继承可以使得子类增强父类的方法

6:细节

1: 函数名必须相同

2:参数列表必须相同

3: 子类重写父类的函数的时候,函数的访问权限必须大于等于父类的函数的访

问权限否则编译报错

4:子类重写父类的函数的时候,返回值类型必须是父类函数的返回值类型或该返回值类型的子类。不能返回比父类更大的数据类型: 如子类函数返回值类型是Object

?

1:定义 A B ?C 类 B extends A ?

2:Father类中定义A getA();

3:Son 类中重写getA(); 方法,尝试将返回值修改为B,C ,Object

1:B编译通过

2:C 编译失败?,没有继承关系

3:Object编译失败,比父类的返回值类型更大


class?A {

?

}

?

class?B extends?A {

?

}

?

class?C {

?

}

class?Father {

String name;

?

void?eat() {

System.out.println("吃窝窝");

}

?

// 定义一个函数,获取A类的对象,

A getA() {

return?new?A();

}

?

}

?

class?Son extends?Father {

?

public?void?eat() { // 继承可以使得子类增强父类的方法

System.out.println("来两杯");

System.out.println("来俩小菜");

super.eat();

System.out.println("来一根");

}

?

// B类是A类的子类

B getA() {

return?new?B();

}

}

?

class?Demo8 {

?

public?static?void?main(String[] args) {

Son s = new?Son();

s.eat();

?

}

}

?

7:子类对象查找属性或方法时的顺序:

1:原则:就近原则。

如果子类的对象调用方法,默认先使用this进行查找,如果当前对象没有找到属性或方法,找当前对象中维护的super关键字指向的对象,如果还没有找到编译报错,找到直接调用。

8:重载和重写的不同

1:重载(overload): ?

1:前提: 所有的重载函数必须在同一个类中

????????2:特点:

???????函数名相同,参数列表不同,与其他的无关(访问控制符、返回值类型)

3:不同:

个数不同 、 顺序不同、 类型不同

2:重写(override):

?????1:前提: 继承

2:特点:

??????函数名必须相同、参数列表必须相同。

??子类的返回值类型要等于或者小于父类的返回值

9:重写练习

描述不同的动物不同的叫法

1:定义动物类

有名字,有吃和叫的方法

2:定义狗继承动物重写父类吃和叫的方法

3:定义猫继承动物重写父类吃和叫的方法


class?Animal{

int?x=1;

String name;

?

void?eat(){

System.out.println("吃东西");

}

void?shout(){

System.out.println("我是动物");

}

}

?

class?Dog extends?Animal{

?

?

void?eat(){

System.out.println("啃骨头");

}

void?shout(){

System.out.println("旺旺");

}

void?eat(String food){

System.out.println("吃:"+food);

}

}

class?Cat extends?Animal{

?

void?eat(){

System.out.println("吃老鼠");

}

void?shout(){

System.out.println("喵喵");

}

}

?

class?Demo9{

?

public?static?void?main(String[] args){

Dog d=new?Dog();

d.shout();

d.eat();

?

Cat c=new?Cat();

c.shout();

c.eat();

System.out.println();

????? }

}

?

instanceof 关键字

1:快速演示instanceof


Person?p=new?Person();

System.out.println( p instanceof?Person);

?

2:instanceof是什么?

1:属于比较运算符:

? 2:instanceof关键字:该关键字用来判断一个对象是否是指定类的对象。

? 3:用法:

?????? 对象 ?instanceof 类; ??

?? 该表达式是一个比较运算符,返回的结果是boolea类型 ?true|false

注意:使用instanceof关键字做判断时,两个类之间必须有关系。

3:案例

定义一个功能表函数,根据传递进来的对象的做不同的事情,如果是狗让其看家,如果是猫让其抓老鼠

1:定义动物类

2:定义狗类继承动物类

3:定义猫类继承动物类

4:定义功能根据传入的动物,执行具体的功能

5:instanceof好处

1:可以判断对象是否是某一个类的实例


package?oop01;

?

/*

?instanceof

?比较运算符

?检查是否是类的对象

? 1:可以判断对象是否是某一个类的实例

? 用法

? 对象 ?instanceof?类;

?

?案例

定义一个功能函数,根据传递进来的对象的做不同的事情

如果是狗让其看家,如果是猫让其抓老鼠

1:定义动物类

2:定义狗类继承动物类

3:定义猫类继承动物类

4:定义功能根据传入的动物,执行具体的功能

?*/

?

class?Animal {

?

String name;

?

void?eat() {

System.out.println("吃东西");

}

?

void?shout() {

System.out.println("我是动物");

}

}

?

class?Dog extends?Animal {

?

void?eat() {

System.out.println("啃骨头");

}

?

void?shout() {

System.out.println("旺旺");

}

?

}

?

class?Cat extends?Animal {

?

void?eat() {

System.out.println("吃老鼠");

}

?

void?shout() {

System.out.println("喵喵");

}

}

?

class?Demo11 {

?

public?static?void?main(String[] args) {

?

Demo11 d = new?Demo11();

?

// 对象 instanceof?类;

System.out.println(d instanceof?Demo11);

?

?d.doSomething(new Dog());

d.doSomething(new?Cat());

}

?

// 定义一个功能函数,根据传递进来的对象的做不同的事情

// 如果是狗让其看家,如果是猫让其抓老鼠

// 对象 instanceof?类;

void?doSomething(Animal a) {

if?(a instanceof?Dog) {

a.eat();

a.shout();

System.out.println("小狗看家");

} else?if?(a instanceof?Cat) {

a.eat();

a.shout();

System.out.println("抓老鼠");

}

}

}

练习:


byte[] bs = new?byte[] { 1, 2, 3 };

int[] is = new?int[] { 1, 2, 3 };

String[] ss = new?String[] { "jack", "lucy", "lili"?};

System.out.println(bs instanceof?byte[]); // true

System.out.println(is instanceof?int[]); // true

System.out.println(ss instanceof?String[]); // true

// System.out.println(bs?instanceof?int[]); // 不可转换的类型

final关键字

?

1:定义静态方法求圆的面积

2:定义静态方法求圆的周长

3:发现方法中有重复的代码,就是PI,圆周率。

1:如果需要提高计算精度,就需要修改每个方法中圆周率。

4:描述一个变量

1:方法都是静态的,静态只能访问静态,所以变量也定义为静态的。

public static double PI=3.14;

1:如果定义为public后,新的问题,类名.PI=300; 改变了PI的值。

2:修改为private,修改为private后进行了封装,需要getset公共访问方法。

3:现有的知识不能解决这样的问题了。可以使用final


class?Demo12 {

?

public?static?final?double?PI?= 3.14; // 静态常量

?

public?static?double?getArea(double?r) {

return?PI?* r * r;

}

?

public?static?double?getLength(double?r) {

return?PI?* r * 2;

}

?

public?static?void?main(String[] args) {

?

// Demo12.PI=300; 无法为最终变量 PI 指定值

System.out.println(Demo12.PI);

?

}

?

}

?

5:使用final

1:final关键字主要用于修饰类、类成员、方法、以及方法的形参。

2:final修饰成员属性:

?

1:说明该成员属性是常量,不能被修改。

public static final double PI=3.14;

1:public :访问权限最大

2:static :内存中只有一份

3:final ?:是一个常量

4:常量名大写

5:必须初赋值。

?

2:使用类名.成员。修改该成员的值,报错。--常量不能被修改

1:基本数据类型,final使值不变

2:对象引用,final使其引用恒定不变,无法让其指向一个新的对象,但是对象自身却可以被修改。

3:该关键字一般和static关键字结合使用

1:常量可以优先加载,不必等到创建对象的时候再初始化。

4:final和static可以互换位置

5:常量一般被修饰为final

3:fianl修饰类:

1:该类是最终类,不能被继承。

1:将父类加final修饰,子类继承,就会报错。

2:查看api文档发现String类是final的。Integer类也是final的

1:为了防止代码功能被重写

2:该类没有必要进行扩展

4:final修饰方法:

1:该方法是最终方法,不能被重写

2:当一个类被继承,那么所有的非私有函数都将被继承,如果函数不想被子类继承并重写可以将该函数final修饰

3:当一个类中的函数都被修饰为final时,可以将类定义为final的。


class?Father2{

final?void?eat(){

System.out.println("eating....");

}

}

?

class?Son2 extends?Father2{

//该方法是最终方法,不能被重写

void?eat(){

System.out.println("eating....");

}

}

class?Demo12 {

?

?

public?static?void?main(String[] args) {

?

// Demo12.PI=300; 无法为最终变量 PI 指定值

System.out.println(Demo12.PI);

Son2 s=new?Son2();

s.eat();

?

}

?

5:final关键字修饰形参

1:当形参被修饰为final,那么该形参所属的方法中不能被篡改。

2: 项目中主要用于一些只用来遍历未知数据的函数。将未知变量声明为final的。增强数据的安全性。


class?Demo14 {

?

public?static?void?main(String[] args) {

?

System.out.println();

String[] arr = { "think in java", "java就业教程", "java核心技术"?};

?

print(arr);

?

}

?

// 该方法,打印书名。

public?static?void?print(final?String[] arr) {

//arr?= null; ,无法重新赋值

?

for?(int?x = 0; x < arr.length; x++) {

System.out.println(arr[x]);

}

}

?

}

?

10:思考

为什么子类一定要访问父类的构造函数呢

1:子类继承了父类的属性,如果要使用父类的属性必须初始化,创建子类对象,必须先初始化父类属性

必须调用父类的构造方法。

2:为什么调用父类无参的构造函数

设计java语言之时,只知道编译器会默认添加无参的构造函数,有参的无法确定。

但是可以通过super关键字显式调用父类指定构造函数。

3:为什么super()this()语句要放在构造函数的第一行

子类可能会用到父类的属性,所以必须先初始化父类。

?

夜光:作业

静态和非静态的区别。说一下内存。
成员变量和静态变量的区别?
静态的特点以及注意事项?
什么时候使用静态?
继承的好处?
java改良多继承的原因?
当使用一个已存在的继承体系时,该如何更快应用
什么时候用继承?
super和this的特点?
覆盖的特点,何时应用,注意事项?
子类的实例化过程?为什么是这样的实例化过程?
super语句,和this语句为什么不能同时存在,super为什么要定义在第一行?
?




免责声明:本文仅代表作者个人观点,与华纳网无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
责任编辑:高晓娜
地址: 辽宁省大连市中山区港湾街20号名仕财富中心B座1517室 联系电话: 0411-84950851
© 2017 大连华纳文化传媒有限公司 All rights reserved
经营许可证编号:辽B2-20170212 备案号:辽ICP备17007383号-2
辽公网安备 21021102000241