零基础学Java第十八讲---抽象类和接口(3)

         续接上一讲

目录

一、内部类

1、内部类的分类

2、静态内部类

3、实例内部类---未被static修饰的成员内部类

4、局部内部类

5、匿名内部类

二、Object类

1、获取对象信息

2、equals方法

3、hashcode方法


一、内部类

        当⼀个事物的内部,还有⼀个部分需要⼀个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么这个内部的完整结构最好使用内部类。在Java中,可以将⼀个类定义在另⼀个类或者⼀个方法的内部,前者称为内部类,后者称为外部类。内部类也是封装的⼀种体现。

 public class OutClass {class InnerClass{}}// OutClass是外部类 // InnerClass是内部类 

#注:

(1)定义在class类名{ }花括号外部的,即使是在⼀个文件里,都不能称为内部类

public class A{}class B{}// A 和 B是两个独⽴的类,彼此之前没有关系 

(2)内部类和外部类共用同⼀个java源文件,但是经过编译之后,内部类会形成单独的字节码文件

1、内部类的分类

public class OutClass {// 成员位置定义:未被static修饰 --->实例内部类 public class InnerClass1{}// 成员位置定义:被static修饰 ---> 静态内部类 static class InnerClass2{}public void method(){// ⽅法中也可以定义内部类 ---> 局部内部类:⼏乎不⽤ class InnerClass5{}}
}

根据内部类定义的位置不同,⼀般可以分为以下几种形式:

(1)成员内部类(普通内部类:未被static修饰的成员内部类和静态内部类:被static修饰的成员内部类)

(2)局部内部类(不谈修饰符)、匿名内部类

#注:内部类其实日常开发中使用并不是非常多,大家在看⼀些库中的代码时候可能会遇到的比较 多,日常开始中使用最多的是匿名内部类。

2、静态内部类

public class OutClass {public int a;public static int b;// 静态内部类:被static修饰的成员内部类 static class InnerClass{public void methodInner(){// 在内部类中只能访问外部类的静态成员 // a = 100;      // 编译失败,因为a不是类成员变量 b =200;}}public static void main(String[] args) {// 静态内部类对象创建 & 成员访问 OutClass.InnerClass innerClass = new OutClass.InnerClass();innerClass.methodInner();}}

#注:

(1)在静态内部类中只能访问外部类中的静态成员

          如果确实想访问,我们该如何做?

方法一:通过外部类实例传递

class OuterClass {private int nonStaticField = 10;private static int staticField = 20;static class StaticInnerClass {private OuterClass outer;  // 持有外部类实例的引用public StaticInnerClass(OuterClass outer) {this.outer = outer;  // 通过构造函数传入外部类实例}public void accessNonStaticField() {System.out.println(outer.nonStaticField);  // 通过实例访问非静态成员System.out.println(staticField);  // 可以直接访问静态成员}}
}// 使用方式
OuterClass outer = new OuterClass();
OuterClass.StaticInnerClass inner = new OuterClass.StaticInnerClass(outer);
inner.accessNonStaticField();

方法二:通过方法传参传递

class OuterClass {private int nonStaticField = 10;static class StaticInnerClass {public void printField(OuterClass outer) {System.out.println(outer.nonStaticField);  // 通过参数访问}}
}// 使用方式
OuterClass outer = new OuterClass();
OuterClass.StaticInnerClass inner = new OuterClass.StaticInnerClass();
inner.printField(outer);

(2)创建静态内部类对象时,不需要先创建外部类对象

3、实例内部类---未被static修饰的成员内部类

public class OutClass {public int a;public static int b;public  int c;//  实例内部类:未被static修饰 class InnerClass{int c;public void methodInner(){// 在实例内部类中可以直接访问外部类中:任意访问限定符修饰的成员 a = 100;b =200;// 如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类⾃⼰的c = 300;System.out.println(c);// 如果要访问外部类同名成员时候,必须:外部类名称.this.同名成员名字 OutClass.this.c = 400;System.out.println(OutClass.this.c);}}public static void main(String[] args) {// 外部类:对象创建 以及 成员访问 OutClass outClass = new OutClass();System.out.println(outClass.a);System.out.println(OutClass.b);System.out.println(outClass.c);System.out.println("=============实例内部类的访问=============");// 要访问实例内部类中成员,必须要创建实例内部类的对象 // ⽽普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类// 创建实例内部类对象 OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();// 上述语法⽐较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象 OutClass.InnerClass innerClass2 = outClass.new InnerClass();innerClass2.methodInner();}
}

#注:

(1)外部类中的任何成员都可以在实例内部类方法中直接访问

(2)实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束

(3)在实例内部类方法中访问同名的成员时,优先访问自己的,如果要访问外部类同名的成员,必须:外部类名称.this.同名成员来访问

(4)实例内部类对象必须在先有外部类对象前提下才能创建

(5)实例内部类的非静态方法中包含了⼀个指向外部类对象的引用

(6)外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。

4、局部内部类

        定义在外部类的方法体或者{ }中,该种内部类只能在其定义的位置使用,⼀般使⽤的非常少,此处简单了解下语法格式。

 public class OutClass {int a = 10;public void method(){int b = 10;// 局部内部类:定义在⽅法体内部// 不能被public、static等访问限定符修饰 class InnerClass{public void methodInnerClass(){System.out.println(a);System.out.println(b);}}// 只能在该⽅法体内部使⽤,其他位置都不能⽤ InnerClass innerClass = new InnerClass();innerClass.methodInnerClass();}public static void main(String[] args) {// OutClass.InnerClass innerClass = null; 编译失败 }}

#注:

(1)局部内部类只能在所定义的方法体内部使用

(2)不能被public、static等修饰符修饰

(3)编译器也有自己独立的字节码文件,命名格式:外部类名字$数字内部类名字.class

(4)几乎不会使用

5、匿名内部类

        匿名内部类是Java中⼀种特殊的嵌套类,它没有类名,在声明的同时完成实例化。匿名内部类通常用于创建只需使用⼀次的类

语法形式:

 new SuperType(constructor-arguments) {   //类体 };

SuperType 可以是接口、抽象类或具体类。

示例代码:

 interface Greeting {void greet();}public class Test  {public static void main(String[] args) {Greeting greeting = new Greeting() {@Overridepublic void greet() {System.out.println("Hello!");}};greeting.greet();}}

        匿名内部类当中可以定义和正常类⼀样的成员变量,但是和正常类⼀样都不能直接包含执行语句

interface Greeting {void greet();
}public class Derived  extends Base   {public static void main(String[] args) {Greeting greeting = new Greeting() {public static int a = 10;a = 100;//error@Overridepublic void greet() {System.out.println("Hello!"+a);}};greeting.greet();}}

二、Object类

        Object是Java默认提供的⼀个类。Java里面除了Object类,所有的类都是存在继承关系的。默认会继承Object父类。即所有类的对象都可以使用Object的引用进行接收。

        使用Object可以接收所有类的对象,所以在开发之中,Object类是参数的最高统⼀类型。但是Object类也存在有定义好的⼀些方法。

        对于整个Object类中的方法需要实现全部掌握。 我们主要来熟悉这几个方法:toString()方法,equals()方法,hashcode()方法

1、获取对象信息

        如果要打印对象中的内容,可以直接重写Object类中的toString()方法,之前已经讲过了,此处不再赘述啦。

// Object类中的toString()⽅法实现: 
public String toString() {return getClass().getName() + "@" + Integer.toHexString(hashCode());
}

2、equals方法

在Java中,==进行比较时:

a.如果 == 左右两侧是基本类型变量,比较的是变量中值是否相同

b.如果 == 左右两侧是引用类型变量,比较的是引用变量地址是否相同

c.如果要比较对象中内容,必须重写Object中的equals方法,因为equals方法默认也是按照地址比较的:

// Object类中的equals⽅法 
public boolean equals(Object obj) {return (this == obj);   // 使⽤引⽤中的地址直接来进⾏⽐较  
}
class Person{private String name ; private int age ; public Person(String name, int age) {this.age = age ; this.name = name ;}
}
public class Test {public static void main(String[] args) {Person p1 = new Person("gaobo", 20) ; Person p2 = new Person("gaobo", 20) ; int a = 10;int b = 10;System.out.println(a == b);             // 输出trueSystem.out.println(p1 == p2);           // 输出falseSystem.out.println(p1.equals(p2));      // 输出false}
}

        Person类重写equals方法后,然后比较:

class Person{...@Overridepublic boolean equals(Object obj) {if (obj == null) {return false ; }if(this == obj) {return true ; }// 不是Person类对象 if (!(obj instanceof Person)) {return false ; }Person person = (Person) obj ; // 向下转型,⽐较属性值 return this.name.equals(person.name) && this.age==person.age ; }
}

        因此,比较对象中内容是否相同的时候,⼀定要重写equals方法!!!

3、hashcode方法

回忆刚刚的toString方法的源码:

 public String toString() {return getClass().getName() + "@" + Integer.toHexString(hashCode());}

        我们看到了hashCode()这个方法,他帮我算了⼀个具体的对象位置,这里面涉及数据结构,但是我们还没讲到数据结构,没法讲述,所以我们只能说它是个内存地址。然后调用Integer.toHexString()方法, 将这个地址以16进制输出。

hashcode方法源码:

 public native int hashCode();

该方法是⼀个native方法,底层是由C/C++代码写的。我们看不到。

        我们认为两个名字相同,年龄相同的对象,将存储在同⼀个位置,如果不重写hashcode()方法,我们可以来看示例代码:

 class Person {public String name;public int age;public Person(String name, int age) {this.name = name;this.age = age;}}public class TestDemo {public static void main(String[] args) {Person per1 = new Person("gaobo", 20) ;Person per2 = new Person("gaobo", 20) ;System.out.println(per1.hashCode());System.out.println(per2.hashCode());}}//执⾏结果 4601419581163157884

#注:两个对象的hash值不⼀样。

        像重写equals方法⼀样,我们也可以重写hashcode()方法。此时我们再来看看。

 class Person {public String name;public int age;public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic int hashCode() {return Objects.hash(name, age);}}public class TestDemo4 {public static void main(String[] args) {Person per1 = new Person("gaobo", 20) ;Person per2 = new Person("gaobo", 20) ;System.out.println(per1.hashCode());System.out.println(per2.hashCode());}}//执⾏结果 460141958460141958

#注:哈希值⼀样。

总结:

(1)hashcode方法用来确定对象在内存中存储的位置是否相同

(2)事实上hashCode()在散列表中才有用,在其它情况下没用。在散列表中hashCode()的作用是获取对象的散列码,进而确定该对象在散列表中的位置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/93566.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/93566.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

字节数据流

记录 干货|8000字长文,深度介绍Flink在字节跳动数据流的实践 字节跳动基于Flink的MQ-Hive实时数据集成

Vision Master的C#脚本与opencv联合编程

需要在VM的C#脚本设置string类型Out变量和float类型OutF变量,python的输出信息会在Out变量显示 using System; using System.IO; using Script.Methods; using System.Diagnostics; using System.Net.Sockets; using System.Text; using System.Threading;public pa…

运维工作架构流程搭建

前言 在解决了运维是干什么的,运维的工作的意义后,这一章我们系统性的讲讲运维工作流程搭建,希望大家能通过我的分享有所收获,这一章干货满满。 一、网络服务器架构 按照一般的访问方式,按照我自己的理解画了一个网络服…

安全存储之 SAES+HUK 使用技巧和常见问题 LAT1543

关键字:AES,SAES, HUK, DHUK, 安全存储 引言 近年来,嵌入式设备信息安全被越来越多地提及,从智能穿戴产品、智能工业设备到物联网产品都对设备信息安全提出了要求,比如基础的安全启动,安全升级&#xff0…

ubuntu 20.04 搭建多用户远程桌面共享及同时在线使用

使用效果: 物理机: 远程桌面用户: 实现过程: 1.使用脚本安装xrdp并修改配置及启动服务 setup_xrdp.sh sudo apt update sudo apt install xrdp -ysudo systemctl enable xrdp sudo systemctl start xrdp#sudo vim /etc/xrdp/startwm.sh #unset DBUS_SESSION_BUS_ADDRES…

[激光原理与应用-287]:理论 - 波动光学 - 电磁波既能承载能量,又能承载信息?

电磁波既能承载能量,又能承载信息,这一特性源于其物理本质和调制技术的结合。以下从能量承载和信息承载两方面进行详细解析:一、电磁波如何承载能量?电磁波的能量承载源于其电场和磁场的周期性振荡,具体机制如下&#…

哪里找最新AI工具官网?如何快速对比ChatGPT替代品?AI工具导航指南 - AIbase

你是否曾有这样的经历: 听闻某款新AI工具爆火,翻遍网络却找不到可靠官网或真实评测? 面对功能相似的ChatGPT替代品,参数对比表格散落各处,决策耗时耗力? 想紧跟AI领域突破,却淹没在海量资讯碎…

第一阶段C#基础-15:面向对象梳理

面向对象对象三(四)大特征:封装,继承,多态,(抽象)1_封装(1)封装是指将数据(属性)和行为(方法)组合在一个类中&…

中国星网发展情况全面分析

中国星网作为我国卫星互联网领域的"国家队"先锋,自2021年4月成立以来已取得显著进展。截至2025年8月,中国星网主导的GW星座已累计发射73颗卫星,形成"四天两发"的高频发射节奏,标志着我国低轨卫星互联网建设进入加速期。在战略定位上,中国星网不仅承担…

C++ Qt 成员对象初始化与 TCP 长连接问题深度解析

文章目录C Qt 成员对象初始化与 TCP 长连接问题深度解析1. 栈对象、堆对象与类成员对象的区别1.1 栈对象(局部变量)1.2 堆对象(动态分配)1.3 类成员对象1.4 栈对象 vs 成员对象 vs 堆对象对比表2. 为什么初始化列表必须用2.1 构造…

深度学习周报(8.11~8.17)

目录 摘要 Abstract 1 CNN--卷积神经网络简介 2 CNN核心操作 2.1 卷积 2.2 池化 3 总结 摘要 本周主要学习了卷积神经网络(CNN)的相关知识,包括概念、基本架构与应用领域等知识,了解了CNN利用其结构高效地从图像等网格化数…

oracle dg duplicate限速

一些客户在搭建dg的时候需要进行限速&#xff0c;不然对生产库的影响比较大&#xff0c;例如将速度限制到200M每秒&#xff0c;语法如下&#xff1a;rman target sys/XXXX auxiliary sys/XXXXdg <<EOF run{ allocate channel d1 type disk rate 200M; allocate auxiliar…

飞算JavaAI智慧校园场景实践:从校园管理到师生服务的全链路技术革新

目录一、智慧校园核心场景的技术突破1.1 智能校园综合管理系统1.2 智慧教学资源共享系统1.3 校园生活服务集成系统二、智慧校园系统效能升级实践结语&#xff1a;重新定义智慧校园技术边界在校园管理领域&#xff0c;“规模化运营”与“个性化服务”的矛盾、“管理效率”与“服…

PTPX分析中,如何处理fsdb文件过大的问题?

PTPX分析中&#xff0c;如何处理fsdb文件过大的问题&#xff1f;摘要&#xff1a;下面将基于Synopsys工具链&#xff08;PrimeTime PX&#xff0c;即PTPX&#xff0c;用于功耗分析&#xff1b;Verdi&#xff0c;用于波形查看&#xff09;逐一解答每个部分。这些工具在SoC功耗验…

004.Redis 数据持久化概述及实战

文章目录Redis持久化说明Redis持久化RDB持久化AOF持久化混合持久化save与bgsaveRedis RDB持久化Redis 安装Redis RDB配置手动触发RDB持久化模拟写入测试数据模拟进程异常RDB的优缺点优势劣势Redis AOF持久化Redis 安装Redis AOF配置AOF持久化模拟写入测试数据模拟进程异常AOF的…

Kubernetes(K8s)常用命令全解析:从基础到进阶

Kubernetes&#xff08;K8s&#xff09;常用命令全解析&#xff1a;从基础到进阶 引言&#xff1a;为什么掌握K8s命令是云原生时代的必备技能&#xff1f; Kubernetes&#xff08;简称K8s&#xff09;作为容器编排的事实标准&#xff0c;已成为云原生应用部署、扩展和管理的核…

深入解析StatefulSet与K8s服务管理

目录 一、Statefulset控制器&#xff1a;概念、原理解读 有状态服务 无状态服务 StatefulSet部分组成 Headless service 二、Statefulset资源清单文件编写技巧 三、Statefulset使用案例&#xff1a;部署web站点 四、Statefulset管理pod&#xff1a;扩容、缩容、更新 St…

GaussDB常用术语缩写及释义

序号 术语全称释义1ACLAccess Control List访问控制列表&#xff0c;被授权访问某一资源的实体及其访问权限的列表。2APIApplication Programming Interface应用程序接口&#xff0c;一些预先定义的接口&#xff08;如函数&#xff0c;http接口&#xff09;或指软件系统不同组…

工业一体机5G通讯IC/ID刷卡让MES系统管理更智能

在智能制造升级过程中&#xff0c;MES 系统作为生产执行的核心枢纽&#xff0c;其数据采集的实时性、准确性与管理安全性直接决定管理效能。传统 MES 系统常面临数据传输滞后、设备对接繁琐、权限管理粗放等问题&#xff0c;而工业一体机凭借 5G 通讯与 IC/ID 刷卡功能&#xf…

电脑开机几秒后就停止然后再循环是怎么回事

硬件问题电源问题&#xff1a;电源内部的滤波电容若出现鼓包或漏液现象&#xff0c;会导致电源供电不足&#xff0c;从而使电脑开机后自动停止并重新启动。另外&#xff0c;电源线连接不良、插座无电或电源供应器故障&#xff0c;也可能引发此问题。CPU 散热问题&#xff1a;如…