java如何用==判断两个值是否相等 java比较语句的基础教程

Java中==对基本类型比较值,对对象比较引用地址;对象内容比较应使用equals(),且重写equals()时必须重写hashCode()以保证哈希集合的正确性。

java如何用==判断两个值是否相等 java比较语句的基础教程

在Java里,

==

这个操作符,说白了,它就是用来比较两个东西是不是“同一个”。对于基本数据类型(比如

int

,

boolean

,

double

),它比较的是它们的值是否相等。但对于对象(比如

String

、你自定义的类),它比较的则是这两个引用变量是否指向内存中的同一个对象实例。这听起来简单,却是很多Java初学者甚至一些老手都会偶尔“翻车”的地方,因为我们常常想当然地认为它应该比较“内容”是否一样。

解决方案

要搞清楚Java里

==

的用法,得把基本类型和对象类型分开看,这两种情况下的行为逻辑完全不同。

1. 基本数据类型(Primitive Types)的比较:当涉及到

int

long

float

double

char

byte

short

boolean

这些基本数据类型时,

==

的行为非常直观:它直接比较这两个变量存储的实际数值是否相等。

int a = 10;int b = 10;System.out.println(a == b); // 输出 truechar c1 = 'A';char c2 = 'A';System.out.println(c1 == c2); // 输出 trueboolean flag1 = true;boolean flag2 = false;System.out.println(flag1 == flag2); // 输出 false

这没什么好说的,符合直觉。

2. 对象(Object Types)的比较:这才是

==

真正让人“头疼”的地方。对于所有非基本数据类型的对象(包括

String

、数组、自定义类的实例,以及包装类如

Integer

double

等),

==

比较的不是它们的内容,而是它们在内存中的地址。也就是说,它检查的是这两个引用变量是否指向了内存中的同一个对象。

String s1 = new String("hello");String s2 = new String("hello");System.out.println(s1 == s2); // 输出 falseString s3 = "world"; // 字符串字面量,会进入常量池String s4 = "world"; // s4会引用常量池中与s3相同的对象System.out.println(s3 == s4); // 输出 trueObject obj1 = new Object();Object obj2 = new Object();System.out.println(obj1 == obj2); // 输出 false// 即使内容相同,只要是两个不同的对象,== 就返回false

你可能会疑惑,

s3 == s4

为什么

true

?这是因为Java对字符串字面量做了优化,它们会被存储在字符串常量池中,如果内容相同,会复用同一个对象。但通过

new String()

方式创建的字符串,每次都会在堆内存中创建一个新的对象。

立即学习“Java免费学习笔记(深入)”;

所以,如果你想比较两个对象的内容是否相等,几乎总是应该使用它们的

equals()

方法,而不是

==

Java中

==

对对象比较的陷阱与原理是什么?

说实话,

==

对于对象来说,它的本意就是判断“身份”是否一致,而不是“内容”是否一致。这其实是个非常底层且高效的判断,因为它只需要比较两个内存地址。但对于我们日常编程,尤其是在处理像

String

这样的“值类型”对象时,往往希望比较的是内容,这就造成了认知上的偏差,进而产生了“陷阱”。

最典型的例子就是字符串。一个初学者写下

if (myString == "某个值")

,往往会得到意想不到的结果。因为

myString

可能是在运行时动态生成的,即使它的字符序列和

"某个值"

一模一样,它们在内存中也可能是两个不同的

String

对象。

String user_input = new Scanner(System.in).nextLine(); // 假设用户输入 "admin"String expected_value = "admin";// 这是一个常见的错误示范!if (user_input == expected_value) {    System.out.println("匹配成功 (可能是巧合或者字符串常量池的魔术)");} else {    System.out.println("匹配失败 (更常见的结果)");}// 正确的做法if (user_input.equals(expected_value)) {    System.out.println("匹配成功 (内容相等)");}

这种设计并非Java的“缺陷”,而是其对引用类型和值类型的区分。

==

提供了最基本的引用比较能力,而

equals()

则提供了更高级、可自定义的“值”比较能力。理解这一点,就能避开很多坑。

Clipfly Clipfly

一站式AI视频生成和编辑平台,提供多种AI视频处理、AI图像处理工具。

Clipfly 129 查看详情 Clipfly

Java中包装类(Wrapper Classes)的

==

比较有何特殊之处?

包装类,比如

Integer

long

boolean

等,它们是基本数据类型的对象表示。当用

==

比较两个包装类对象时,遵循的是对象比较的规则,即比较引用地址。然而,这里又有一个小小的“陷阱”——缓存机制自动装箱/拆箱

Java为了性能优化,对一些常用的包装类值进行了缓存。例如,

Integer

类在

-128

127

之间的值会被缓存。这意味着,如果你创建两个

Integer

对象,它们的值在这个范围内且是通过自动装箱创建的,那么它们很可能引用的是同一个缓存对象。

Integer i1 = 100; // 自动装箱,从缓存获取Integer i2 = 100; // 自动装箱,从缓存获取System.out.println(i1 == i2); // 输出 true (因为100在缓存范围内,引用的是同一个对象)Integer i3 = 200; // 自动装箱,不在缓存范围内,创建新对象Integer i4 = 200; // 自动装箱,不在缓存范围内,创建新对象System.out.println(i3 == i4); // 输出 false (两个不同的对象)// 显式创建新对象,即使值在缓存范围内,也会创建新对象Integer i5 = new Integer(100);Integer i6 = new Integer(100);System.out.println(i5 == i6); // 输出 false

这真的是一个非常隐蔽的坑。我个人就见过不少代码因为不了解这个缓存机制,在比较

Integer

时误用

==

导致偶发性 bug 的。所以,永远建议使用

equals()

方法来比较包装类对象的值,这样可以避免这些由缓存和对象创建方式带来的不确定性。

Integer val1 = 200;Integer val2 = 200;System.out.println(val1.equals(val2)); // 输出 true,这是正确的比较方式

何时应该使用

equals()

方法,以及如何正确重写它?

当你需要比较两个对象的“内容”或“语义”是否相等时,就应该使用

equals()

方法。这是Java中所有对象都继承自

Object

类的一个方法,它的默认实现和

==

一样,也是比较引用地址。因此,对于自定义的类,如果你希望它们在内容相同时被认为是相等的,你就必须重写

equals()

方法。

何时使用

equals()

比较两个

String

对象的内容。比较两个自定义对象(例如

Person

类,你希望姓名和年龄都相同的两个人被认为是相等的)。比较包装类对象的值。比较集合框架中的元素(例如

ArrayList

HashSet

中的元素)。

如何正确重写

equals()

重写

equals()

方法时,必须遵循

Object

类中定义的一系列“约定”(contract),否则可能会导致不可预测的行为,尤其是在使用集合类时。这些约定包括:

自反性 (Reflexive): 对于任何非

null

的引用值

x

x.equals(x)

必须返回

true

对称性 (Symmetric): 对于任何非

null

的引用值

x

y

,当且仅当

y.equals(x)

返回

true

时,

x.equals(y)

才返回

true

传递性 (Transitive): 对于任何非

null

的引用值

x

y

z

,如果

x.equals(y)

返回

true

,并且

y.equals(z)

返回

true

,那么

x.equals(z)

也必须返回

true

一致性 (Consistent): 对于任何非

null

的引用值

x

y

,只要

equals

比较中用到的信息没有被修改,多次调用

x.equals(y)

都会返回相同的结果。非空性 (Nullity): 对于任何非

null

的引用值

x

x.equals(null)

必须返回

false

此外,一个非常重要的最佳实践是:如果重写了

equals()

方法,就必须同时重写

hashCode()

方法。 这是因为

HashSet

HashMap

等基于哈希表的集合类在存储和查找对象时,会先使用

hashCode()

来确定对象的存储位置,再使用

equals()

来确认对象是否相等。如果

equals()

相等的两个对象

hashCode()

不相等,就会导致在哈希集合中找不到本应存在的对象。

一个简单的

Person

类重写

equals()

hashCode()

的例子:

import java.util.Objects; // Java 7+ 引入的 Objects.equals 和 Objects.hash 简化了代码class Person {    private String name;    private int age;    public Person(String name, int age) {        this.name = name;        this.age = age;    }    public String getName() { return name; }    public int getAge() { return age; }    @Override    public boolean equals(Object o) {        // 1. 检查是否是同一个对象引用,这是最快的判断        if (this == o) return true;        // 2. 检查传入对象是否为null,以及类型是否匹配        //    getClass() 比 instanceof 更严格,要求类型完全一致        if (o == null || getClass() != o.getClass()) return false;        // 3. 类型转换        Person person = (Person) o;        // 4. 比较关键字段        //    基本类型直接比较值        //    引用类型使用其 equals 方法比较        return age == person.age && Objects.equals(name, person.name);    }    @Override    public int hashCode() {        // 使用 Objects.hash() 可以方便地为多个字段生成哈希码        return Objects.hash(name, age);    }    @Override    public String toString() {        return "Person{" +               "name='" + name + ''' +               ", age=" + age +               '}';    }}// 使用示例// Person p1 = new Person("张三", 30);// Person p2 = new Person("张三", 30);// System.out.println(p1.equals(p2)); // 输出 true// System.out.println(p1 == p2);     // 输出 false

正确重写

equals()

hashCode()

是构建健壮Java应用的关键一步,尤其是在处理数据模型和集合时。如果忽略了这些细节,后期排查问题可能会非常痛苦。

以上就是java如何用==判断两个值是否相等 java比较语句的基础教程的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/748993.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月25日 19:16:02
下一篇 2025年11月25日 19:16:27

相关推荐

  • CSS mask属性无法获取图片:为什么我的图片不见了?

    CSS mask属性无法获取图片 在使用CSS mask属性时,可能会遇到无法获取指定照片的情况。这个问题通常表现为: 网络面板中没有请求图片:尽管CSS代码中指定了图片地址,但网络面板中却找不到图片的请求记录。 问题原因: 此问题的可能原因是浏览器的兼容性问题。某些较旧版本的浏览器可能不支持CSS…

    2025年12月24日
    900
  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 为什么设置 `overflow: hidden` 会导致 `inline-block` 元素错位?

    overflow 导致 inline-block 元素错位解析 当多个 inline-block 元素并列排列时,可能会出现错位显示的问题。这通常是由于其中一个元素设置了 overflow 属性引起的。 问题现象 在不设置 overflow 属性时,元素按预期显示在同一水平线上: 不设置 overf…

    2025年12月24日 好文分享
    400
  • 网页使用本地字体:为什么 CSS 代码中明明指定了“荆南麦圆体”,页面却仍然显示“微软雅黑”?

    网页中使用本地字体 本文将解答如何将本地安装字体应用到网页中,避免使用 src 属性直接引入字体文件。 问题: 想要在网页上使用已安装的“荆南麦圆体”字体,但 css 代码中将其置于第一位的“font-family”属性,页面仍显示“微软雅黑”字体。 立即学习“前端免费学习笔记(深入)”; 答案: …

    2025年12月24日
    000
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么我的特定 DIV 在 Edge 浏览器中无法显示?

    特定 DIV 无法显示:用户代理样式表的困扰 当你在 Edge 浏览器中打开项目中的某个 div 时,却发现它无法正常显示,仔细检查样式后,发现是由用户代理样式表中的 display none 引起的。但你疑问的是,为什么会出现这样的样式表,而且只针对特定的 div? 背后的原因 用户代理样式表是由…

    2025年12月24日
    200
  • inline-block元素错位了,是为什么?

    inline-block元素错位背后的原因 inline-block元素是一种特殊类型的块级元素,它可以与其他元素行内排列。但是,在某些情况下,inline-block元素可能会出现错位显示的问题。 错位的原因 当inline-block元素设置了overflow:hidden属性时,它会影响元素的…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 为什么使用 inline-block 元素时会错位?

    inline-block 元素错位成因剖析 在使用 inline-block 元素时,可能会遇到它们错位显示的问题。如代码 demo 所示,当设置了 overflow 属性时,a 标签就会错位下沉,而未设置时却不会。 问题根源: overflow:hidden 属性影响了 inline-block …

    2025年12月24日
    000
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 为什么我的 CSS 元素放大效果无法正常生效?

    css 设置元素放大效果的疑问解答 原提问者在尝试给元素添加 10em 字体大小和过渡效果后,未能在进入页面时看到放大效果。探究发现,原提问者将 CSS 代码直接写在页面中,导致放大效果无法触发。 解决办法如下: 将 CSS 样式写在一个单独的文件中,并使用 标签引入该样式文件。这个操作与原提问者观…

    2025年12月24日
    000
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 em 和 transition 设置后元素没有放大?

    元素设置 em 和 transition 后不放大 一个 youtube 视频中展示了设置 em 和 transition 的元素在页面加载后会放大,但同样的代码在提问者电脑上没有达到预期效果。 可能原因: 问题在于 css 代码的位置。在视频中,css 被放置在单独的文件中并通过 link 标签引…

    2025年12月24日
    100
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000

发表回复

登录后才能评论
关注微信