前言
String 是我们实际开发中使用频率非常高的类,Java 可以通过 String 类来创建和操作字符串,使用频率越高的类,我们就越容易忽视它,因为见的多所以熟悉,因为熟悉所以认为它很简单,其实只是了解到皮毛,并没有真正掌握,而 String 又是面试的高频考点,所以我们有必要将 String 这个类深入研究,彻底搞定,本节课就为大家详细讲解 String 的核心机制以及实际使用。
1、不变性:String 是只读字符串,是一个典型的 immutable 对象,对它进行任何操作,其实都是创建一个新的对象,再把引用指向该对象。不变模式的主要作用在于当一个对象需要被多线程共享并频繁访问时,可以保证数据的一致性。
2、常量池优化:String 对象创建之后,会在字符串常量池中进行缓存,如果下次创建同样的对象时,会直接返回缓存的引用。
3、final:使用 final 来定义 String 类,表示 String 类不能被继承,提高了系统的安全性。
String 不是基本数据类型
这是很基础的东西,但是很多初学者却容易忽视,Java 的 8 种基本数据类型中不包括 String,基本数据类型中用来描述文本数据的是 char,但是它只能表示单个字符,比如 'a','好' 之类的,如果要描述一段文本,就需要用多个 char 类型的变量,也就是一个 char 类型数组,比如“你好” 就是长度为2的数组 char[] chars = {'你','好'};
但是使用数组过于麻烦,所以就有了 String,String 底层就是一个 char 类型的数组,只是使用的时候开发者不需要直接操作底层数组,用更加简便的方式即可完成对字符串的使用。
高频面试题
== 可以理解为是比较栈内存中的值,如果变量是基本数据类型,则栈内存中存放的就是具体数值,如果是引用类型,则栈中存放的是引用的内存地址。
所以对于基本数据类型,== 是比较值是否相等,对于引用数据类型,比较的是引用的内存地址是否相等。
equals 是 Object 类提供的一个方法,其本质就是在用 == 进行判断。
public boolean equals(Object obj) { return (this == obj); }
同时 Java 中任意一个类都可以对其进行重写,根据具体需求重新定义其判断逻辑,比如我们自定义一个 Student 类,如下所示。
public class Student { private Integer id; private String name; public Student(Integer id, String name) { this.id = id; this.name = name; } }
创建两个成员变量值完全相等的实例化对象,并用 equals 方法判断是否相等。
Student student1 = new Student(1,"张三"); Student student2 = new Student(1,"张三"); System.out.println(student1.equals(student2));
结果为 false,因为有两个实例化对象,就必然会在堆内存中开辟两块空间来存储,引用一定是不相同的。而在现实的逻辑中,如果两个学生的 id 和 name 都一样,我们就认为他们是同一个学生,用程序如何来实现呢?通过重写 equals 方法即可,如下所示。
public class Student { private Integer id; private String name; public Student(Integer id, String name) { this.id = id; this.name = name; } @Override public boolean equals(Object obj) { Student student = (Student) obj; if(id.equals(student.id) && name.equals(student.name)){ return true; } return false; } }
String str1 = "Hello World"; String str2 = "Hello"+" World"; System.out.println(str1 == str2);
true,"Hello" 和 " World" 都是字符串字面值,字符串字面值 + 字符串字面值的结果仍然保存在字符串常量池中,所以 str1 和 str2 相同。
String str1 = "Hello World"; String str2 = "Hello"; str2 += " World"; System.out.println(str1 == str2);
false,这题看似与第 2 题一样,为什么结果完全不同呢?因为 str2 = "Hello"+" World" 是直接创建,str2 = "Hello"; str2 = "Hello"; 是先创建再修改,同时修改完成之后的字符串是放在堆内存中的,为什么呢?因为 str2 是一个字符串变量," World" 是字符串字面值,当字符串字面值与 String 类型变量拼接时,得到的新字符串不再保存在常量池中,而是在堆中开辟一块新的空间来存储,所以 str1 引用指向字符串常量池,str2 引用指向堆内存,肯定不相同。
String str1 = "Hello World"; String str2 = " World"; String str3 = "Hello"+str2; System.out.println(str1 == str3);
false,str2 是变量,"Hello" 是字符串字面值,字符串字面值 + 变量会在堆内存中开辟新的空间来存储,所以 str1 和 str3 不同。
String str1 = "Hello World"; final String str2 = " World"; String str3 = "Hello"+str2; System.out.println(str1 == str3);
true,"Hello" 是字符串字面值,str2 是常量,字符串字面值+常量的结果仍然保存在字符串常量池中,所以 str1 和 str3 相同。
String str1 = "Hello World"; final String str2 = new String(" World"); String str3 = "Hello"+str2; System.out.println(str1 == str3);
false,str2 是常量,但是 new String(" World") 保存在堆内存中,所以即使使用 final 进行了修饰,str2 仍然保存在堆中,则 str3 也就保存在堆中,所以 str1 和 str3 不同。
String str1 = "Hello World"; String str2 = "Hello"; String str3 = " World"; String str4 = str2 + str3; System.out.println(str4.intern() == str1);
true,当调用 str4 的 intern 方法时,如果字符串常量池已经包含一个等于 str4 的字符串,则返回该字符串,否则将 str4 添加到字符串常量池中,并返回其引用,所以 str4.intern() 与 str1 相同。
字符串常量池位于堆内存中,专门用来存储字符串常量,可以提高内存的使用率,避免开辟多块空间存储相同的字符串,在创建字符串时 JVM 会首先检查字符串常量池,如果该字符串已经存在池中,则返回它的引用,如果不存在,则实例化一个字符串放到池中,并返回其引用。
String 是不可变类,一旦创建了String对象,我们就无法改变它的值。因此它是线程安全的,同一个字符串实例可以被多个线程共享,保证了多线程的安全性。
10、在使用 HashMap 的时候,用 String 做 key 有什么好处?
HashMap 内部实现是通过 key 的 hashcode 来确定 value 的存储位置,因为字符串是不可变的,所以当创建字符串时,它的 hashcode 被缓存下来,不需要再次计算,所以相比于其他对象更快。
最后 欢迎大家一起交流,喜欢文章记得点个赞哟,感谢支持!
- 备战金九银十,Java研发面试题(Spring、MySQL、JVM、Mybatis、Redis、Tomcat)[带答案],刷起来!
八月在即,马上就是"金九银十",又是跳槽招聘季.咱们这行公认涨薪不如跳槽加的快.但不建议频繁跳槽,还是要学会融合团队,抓住每个机会提升技能. 苏先生在这里给大家整理了一套各大互联网 ...
- Java程序员备战“金九银十”必备的面试技巧(附携程Java岗面试题)
一.面试前的准备 1.1 如何准备一场面试1.1.1 如何获取大厂面试机会1.1.2 面试必知 ①. 准备介绍自己 ②. 关于着装 ③ .随身带上自己的成绩单和简历 ④. 如果笔试就提前刷一些笔试题 ...
- 金九银十跳槽高峰,面试必备之 Redis + MongoDB 常问80道面试题
前言 有着“金九银十”之称的招聘旺季已经开启,跳槽高峰期也如约而至. 本文为主要是 Redis + MongoDB 知识点的攻略,希望能帮助到大家. 内容较多,大家准备好耐心和瓜子矿泉水. Redis ...
- 金九银十想去跳槽面试?那这份Java面经你真得看看了,写的非常详细!
前言 前两天在和朋友吃饭的时候聊到时间这个东西是真的过的好坏啊,金三银四仿佛还在昨天.一眨眼金九银十又快到了,对程序员来说这两个是一年最合适的跳槽涨薪环节了,今年的你已经做好准备了吗?不妨看看这篇文章 ...
- 不等"金九银十",金风八月,我早已拿下字节跳动的offer
字节跳动,我是在网上投的简历,之前也投过一次,简历都没通过删选,后来让师姐帮我改了一下简历,重新投另一个部门,获得了面试机会.7月23日,中午HR打电话过来预约了下午4点半面试,说会在线写代码,让我准 ...
- “金九银十”已过,总结我的天猫、蚂蚁、头条面试经历(Java岗)
跳槽时时刻刻都在发生,但是我建议大家跳槽之前,先想清楚为什么要跳槽.切不可跟风,看到同事一个个都走了,自己也盲目的开始面试起来(期间也没有准备充分),到底是因为技术原因(影响自己的发展,偏移自己规划的 ...
- 金九银十,史上最强 Java 面试题整理。
以下会重新整理所有 Java 系列面试题答案.及各大互联网公司的面试经验,会从以下几个方面汇总,本文会长期更新. Java 面试篇 史上最全 Java 面试题,带全部答案 史上最全 69 道 Spri ...
- 金九银十中,看看这31道Android面试题
阅读目录 1.如何对 Android 应用进行性能分析 2.什么情况下会导致内存泄露 3.如何避免 OOM 异常 4.Android 中如何捕获未捕获的异常 5.ANR 是什么?怎样避免和解决 ANR ...
- 金九银十,收下这份 Java String 面试题
请点赞关注,你的支持对我意义重大. Hi,我是小彭.本文已收录到 GitHub · Android-NoteBook 中.这里有 Android 进阶成长知识体系,有志同道合的朋友,关注公众号 [彭旭 ...
随机推荐
- hostnamectl命令 主机名 host相关命令
hostnamectl set-hostname CentOS7设置主机名为CentOS7 hostnamectl status查看主机系统信息 注:host+TAB查阅host相关的所有命令 hos ...
- 关于手机微信端ios的input不能选中问题解决方案
最近在做一个微信端的商城,以前做web端的比较多,手机端做的相对来说要少点,老板说让我用俗称”靠谱的移动前端框架”—-AUI来搭建项目. 当时觉得用不用框架无所谓啦.结果后来写到一半把项目发布到手机上 ...
- ReentreantLock:重入锁
ReentreantLock:重入锁 参考:https://www.cnblogs.com/nullzx/p/4968674.html 一). ReentrantLock与synchronized的区 ...
- Windows 10上源码编译glog和gflags 编写glog-config.cmake和gflags-config.cmake | compile glog and glags on windows from source
本文首发于个人博客https://kezunlin.me/post/bb64e398/,欢迎阅读! compile glog v0.3.5 and glags on windows from sour ...
- mysql锁简谈
1.mysql锁, 作用:解决因资源共享而造成的并发问题. 实例:买最好一件衣服X A: X 买: X加锁----->试衣服……下单……付款……打包….------>X解锁 B: X 买: ...
- beta 2/2 阶段中间产物提交
此作业要求参见:https://edu.cnblogs.com/campus/nenu/2019fall/homework/9961 一.小组情况 队名:扛把子 组长:孙晓宇 组员:宋晓丽 梁梦瑶 韩 ...
- python爬取中国知网部分论文信息
爬取指定主题的论文,并以相关度排序. #!/usr/bin/python3 # -*- coding: utf-8 -*- import requests import linecache impor ...
- 使用 buildx 构建多平台 Docker 镜像
原文链接:使用 buildx 构建多平台 Docker 镜像 在工作和生活中,我们可能经常需要将某个程序跑在不同的 CPU 架构上,比如让某些不可描述的软件运行在树莓派或嵌入式路由器设备上.特别是 D ...
- day20191104笔记
MyBatis笔记: 一.MyBatis半自动ORM映射框架, 将数据库中的数据和程序中的数据进行自动映射的前提条件 1. 数据库中的字段必须和程序中的属性保持一致 2. 程序中属性的数据类型必须是基 ...
- MySQL统计各个表中的记录数
通过下面的SQL语句可以统计出数据库的各个表中的记录数: select table_schema, table_name,table_rows from information_schema.tabl ...