当前位置: 首页 > news >正文

Java之字符串实践

功能概述

  • 字符串是Java编程中常用的数据类型,本文对String部分常见功能做了对应实践以及分析。

功能实践

场景1:字符串比较

用例代码

@Test
public void test_string_compare() {String s1 = "abc";String s2 = s1;String s5 = "abc";String s3 = new String("abc");String s4 = new String("abc");System.out.println("s1 == s5: " + (s1 == s5));System.out.println("s1 == s2: " + (s1 == s2));System.out.println("s1.equals(s2): " + s1.equals(s2));System.out.println("s3 == s4: " + (s3 == s4));System.out.println("s1.equals(s4): " + s1.equals(s4));System.out.println("s3.equals(s4): " + s3.equals(s4));
}

运行结果

s1 == s5: true
s1 == s2: true
s1.equals(s2): true
s3 == s4: false
s1.equals(s4): true
s3.equals(s4): true

结果分析

  • “==” 比较基本类型时,判断值是否相等,比较对象类型时,判断是否指向同一内存地址。
  • equals是比较两个对象是否一样(即所有成员的值是否相同)。
  • 字符串,如"abc"是放在常量池中的,在内存中只存在一份副本,所以s1 == s5,同一个字符串,只有一个内存地址(会在常量池中检查是否存在已有的字符串,存在时,返回原有字符串的内存地址,就不会新建新的内存地址了)。
  • s3、s4指向的对象都是通过new创建的新对象,内存地址不一样,所以比较为false。

场景2:字符串intern方法使用

用例代码

@Test
public void test_intern_v1() {// 未使用intern时String s1 = new String("aaa777");String s2 = "aaa777";System.out.println(s1 == s2);// 使用intern时String s3 = new String("aaa777");s3 = s3.intern();String s4 = "aaa777";System.out.println(s3 == s4);
}

运行结果

false
true

结果分析

  • s1是指向堆中对象的引用,s2是指向常量池中字符串的引用,所以两者的内存地址不一样。
  • intern()会查找常量池中是否存在"aaa777"的引用,若存在返回,否则把String对象添加到池中,再返回在池中的引用。

场景3:String、StringBuffer、StringBuilder性能比较

用例代码

@Test
public void test_String_StringBuffer_StringBuilder_efficiency() {testString();testStringBuffer();testStringBuilder();
}private void testString() {String s = "Hello";String s1 = "World";long start = System.currentTimeMillis();for (int i = 0; i < 10000; i++) {s = s + s1;}long end = System.currentTimeMillis();long runtime = (end - start);System.out.println("testString runtime:" + runtime);
}private void testStringBuilder() {StringBuilder s = new StringBuilder("Hello");String s1 = "World";long start = System.currentTimeMillis();for (int i = 0; i < 10000; i++) {s.append(s1);}long end = System.currentTimeMillis();long runtime = (end - start);System.out.println("testStringBuilder runtime:" + runtime);
}private void testStringBuffer() {StringBuffer s = new StringBuffer("Hello");String s1 = "World";long start = System.currentTimeMillis();for (int i = 0; i < 10000; i++) {s.append(s1);}long end = System.currentTimeMillis();long runtime = (end - start);System.out.println("testStringBuffer runtime:" + runtime);
}

运行结果

testString runtime:646
testStringBuffer runtime:1
testStringBuilder runtime:0

结果分析

  • 在执行方面来看:StringBuilder最高、StringBuffer次之、String最低。

场景4:StringTokenizer字符串分割

用例代码

@Test
public void test_StringTokenizer() {StringTokenizer st2 = new StringTokenizer("Hello&World&!","&"); //指定自定义分隔符while (st2.hasMoreTokens()) {System.out.println(st2.nextToken());}
}

运行结果

Hello
World
!

结果分析

  • StringTokenizer可按指定分隔符,第字符串进行分隔。

场景5:字符串与字符数组比较

用例代码

@Test
public void test_string_with_char_array_compare() {String s = "hello";String t = "hello";char c[] = {'h','e','l','l','o'};System.out.println(s.equals(t));System.out.println(t.equals(c));System.out.println(s == t);System.out.println(t.equals(new String("hello")));String s1 = "he" + "llo";System.out.println(s == s1);
}

运行结果

true
false
true
true
true

结果分析

  • s.equals(t)值为true:因为"hello"会存在堆中的字符串常量池中,并且按照享元模式,判断字符串是否存在,不存在才创建,所以s、t是同一个字符串引用
  • t.equals©值为false:按照String重写的equals方法,会判断入参是否为String类型,若不是则返回false
  • s == t值为true:==比较的是引用,因为s、t是同一个引用,所以相等
  • t.equals(new String(“hello”))值为true:因为String的比较逻辑,会去字符串中的字符数组来比较,字符都相等,则返回true
  • s == s1值为true:因为"he" + “llo"在编译器就会转化"hello”,存在常量区,因为s已经放在常量池中,与s1不会创建新的字符串,s与s1是同一个字符串引用

功能总结

  • String常量池采用了享元模式(Flyweight)即会共享字符串,判断字符串是否存在,若存在则使用,否则新创建。常量池:它是一个由数组组成的表,用来存储程序中使用的各种常量,包括Class、String、Integer等各种基本的Java数据类型

  • intern()方法分析:(intern:助理、实习生)

    • intern方法主要把字符串放入字符串常量池中。有两种情况,会将字符串放在常量池中

      • 直接使用双引号声明的String对象会直接存储在常量池中,如:String s1 = “aa” 。
      • 通过调用String提供的intern方法把字符串放在常量池中,会检查字符串在常量池中是否存在,存在则不处理,若不存在则放入字符串常量池中。
    • Returns a canonical representation for the string object. (jdk中的描述)

      • 根据jdk中描述:intern()会返回字符串对象的规范标识,即会从常量池中查找指定字符串,若能找到则返回对应引用,否则把String对象添加到池中,再返回在池中的引用。
    • 注明

      • a)字符串放入字符串常量池,需要满足上面所说的两种情况。
      • b)并不是调用了intern就会将字符串放入字符串常量池中,会先检查字符串是否存在,若存在则不处理。若不存在,要看字符串在堆中是否已存在,已存在只存对象引用,减少对象的创建 c)字符串常量池,在jdk1.7后就从Perm区迁移到堆中,这样对于字符串常量池中可以存对象引用,减少了对象的创建,就大大减少了字符串所占的空间了。
  • String、StringBuilder、StringBuffer使用总结:

    • 如果要操作的数据量比较小,优先使用String类(量多的话,String本质是使用StringBuilder处理的,会产生许多临时对象,触发垃圾回收,影响性能)
    • 如果是单线程下处理大量数据,优先使用StringBuilder类(StringBuilder是线程不安全的,单线程下可以使用,减少线程同步的开销)
    • 如果是多线程下处理大量数据,优先使用StringBuffer类(因为是线程安全的)

相关文章:

Java之字符串实践

功能概述 字符串是Java编程中常用的数据类型&#xff0c;本文对String部分常见功能做了对应实践以及分析。 功能实践 场景1&#xff1a;字符串比较 用例代码 Test public void test_string_compare() {String s1 "abc";String s2 s1;String s5 "abc&quo…...

BM20 数组中的逆序对

描述 解题思路&#xff1a;归并排序 分治&#xff1a;分治即“分而治之”&#xff0c;“分”指的是将一个大而复杂的问题划分成多个性质相同但是规模更小的子问题&#xff0c;子问题继续按照这样划分&#xff0c;直到问题可以被轻易解决&#xff1b;“治”指的是将子问题单独进…...

高德猎鹰轨迹查询相关接口

高德猎鹰轨迹官网&#xff1a;服务管理-API文档-开发指南-猎鹰轨迹服务 | 高德地图API 轨迹查询 httpclient的post // post方法请求 创建轨迹 private static void createTrace() {String key "高德注册的key";String sid "服务id"; // 服务idString…...

整理总结新手开始抖音小店经营:常见问题及解决办法

抖音小店作为一种新兴的电商模式&#xff0c;在短时间内获得了广泛的关注和使用。然而&#xff0c;对于新手来说&#xff0c;抖音小店经营可能会遇到一些问题。下面是四川不若与众总结的一些常见的问题以及相应的解决办法。 问题一&#xff1a;产品选择困难 对于新手来说&#…...

4-1-netty

非阻塞io 服务端就一个线程&#xff0c;可以处理无数个连接 收到所有的连接都放到集合channelList里面 selector是有事件集合的 对server来说优先关注连接事件 遍历连接事件...

hive 动态分区-动态分区数量太多也会导致效率下降只设置非严格模式也能执行动态分区

hive 动态分区-动态分区数量太多也会导致效率下降&只设置非严格模式也能执行动态分区 结论 在非严格模式下不开启动态分区的功能的参数&#xff08;配置如下&#xff09;&#xff0c;同样也能进行动态分区数据写入&#xff0c;目测原因是不严格检查SQL中是否指定分区或者…...

java八股文面试[JVM]——JVM调优

知识来源&#xff1a; 【2023年面试】JVM性能调优实战_哔哩哔哩_bilibili...

FairyGUI-Unity 异形屏适配

本文中会修改到FairyGUI源代码&#xff0c;涉及两个文件Stage和StageCamera&#xff0c;需要对Unity的屏幕类了解。 在网上查找有很多的异形屏适配操作&#xff0c;但对于FairyGUI相关的描述操作很少&#xff0c;这里我贴出一下自己在实际应用中的异形屏UI适配操作。 原理 获…...

Oracle监听器启动出错:本地计算机上的OracleOraDb11g_home1TNSListener服务启动后又停止了解决方案

在启动oracle的服务OracleOraDb11g_home1TNSListener时&#xff0c;提示服务启动后又停止了。 解决方法&#xff1a; 修改oracle安装目录下的两个配置文件&#xff1a; 以上两个文件&#xff0c;对应的HOST的值&#xff0c;都改为127.0.0.1 然后再启动服务&#xff0c;启动成…...

Spring复习:(58)<context:annotation-config/>的作用

引入如下的BeanPostProcessor • ConfigurationClassPostProcessor • AutowiredAnnotationBeanPostProcessor • CommonAnnotationBeanPostProcessor • PersistenceAnnotationBeanPostProcessor • EventListenerMethodProcessor如果xml文件配置了bean中使用了Autowired注解…...

“东方杯”英特尔oneAPI黑客松大赛—参赛经验分享

目录 前言1、大赛要求2、oneMKL介绍3、准备 oneMKL基本使用1、下载&#xff1a;2、安装&#xff1a;3、初始化oneMKL环境&#xff1a;4、编译代码5、运行 所需的头文件使用oneMKL工具生成随机数使用fftw3计算FFT调用oneMKL API加速计算FFT对比两种方法的准确性输出结果结束语 前…...

win10家庭版远程桌面补丁_rdp wrapper

RDP Wrapper Library 就是可以帮你在 Windows 7、Windows 8、Windows 10 家庭版中打开远程桌面的工具。 1、把电脑上打开的安全软件与杀毒软件都关掉&#xff0c;因为这个远程桌面补丁会修改系统文件&#xff0c;所以安全软件可能会拦截。 2、下载RDP Wrapper Library补丁压缩…...

【C++设计模式】开放-封闭原则

2023年8月27日&#xff0c;周日下午 我觉得我的这篇博客还是写得很不错的&#xff0c;哈哈哈。 目录 概述举例说明用开放-封闭原则重构 概述 开放-封闭原则&#xff08;Open-Closed Principle&#xff0c;OCP&#xff09;是面向对象设计中的一个重要原则&#xff0c;也是许多…...

vue+file-saver+xlsx+htmlToPdf+jspdf实现本地导出PDF和Excel

页面效果如下&#xff08;echarts图表按需添加&#xff0c;以下代码中没有&#xff09; 1、安装插件 npm install xlsx --save npm install file-saver --save npm install html2canvas --save npm install jspdf --save2、main.js引入html2canvas import htmlToPdf from …...

axios 进阶

axios 进阶 接口传参方式 使用 xhr 原生技术或者是 axios 时&#xff0c;它的 post 传参方式是键值对的形式 keyvalue。但是在实际开发中一般是使用对象的形式定义数据&#xff0c;方便读取和赋值。所以当我们需要发起请求时可以通过 qs 这一款插件将对象转成键值对形式&…...

Redis限流实践:实现用户消息推送每天最多通知2次的功能

&#x1f3c6;作者简介&#xff0c;黑夜开发者&#xff0c;CSDN领军人物&#xff0c;全栈领域优质创作者✌&#xff0c;CSDN博客专家&#xff0c;阿里云社区专家博主&#xff0c;2023年6月CSDN上海赛道top4。 &#x1f3c6;数年电商行业从业经验&#xff0c;历任核心研发工程师…...

uniapp 存储base64资源为http链接图片

1. 新建一个base64.js 文件 const fsm wx.getFileSystemManager(); // base64data base64资源 // name 文件名 function base64src(base64data, name, cb) {const time new Date().getTime();const filePath ${wx.env.USER_DATA_PATH}/${name}.${time}.png;const buffer …...

列表类控件虚拟化

WPF列表控件提供的最重要的功能是UI虚拟化&#xff08;WPF编程宝典说的&#xff09;。所有的WPF列表控件&#xff08;所有继承自ItemsControl的控件&#xff0c;包括ListBox、CombBox、ListView、TreeView、DataGrid&#xff09;都支持UI虚拟化。 UI虚拟化的支持实际上没有被构…...

c# 多线程Task.Run 取消正在执行的多线程

c# 异步处理&#xff0c;上次处理没有完成&#xff0c;下次有紧接着处理多线程出错 在 C# 中进行异步处理时&#xff0c;确保处理上一个任务完成后再处理下一个任务是很重要的&#xff0c;特别是在涉及多线程的情况下。如果上一个任务尚未完成&#xff0c;而下一个任务又开始执…...

sql server 如何设置主键

开始之前 限制和局限 一个表只能包含一个 PRIMARY KEY 约束。 在 PRIMARY KEY 约束中定义的所有列都必须定义为 NOT NULL。 如果没有指定为 Null 性&#xff0c;则加入 PRIMARY KEY 约束的所有列的为 Null 性都将设置为 NOT NULL。 创建主键会自动创建相应的唯一群集索引、…...

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练

前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1)&#xff1a;从基础到实战的深度解析-CSDN博客&#xff0c;但实际面试中&#xff0c;企业更关注候选人对复杂场景的应对能力&#xff08;如多设备并发扫描、低功耗与高发现率的平衡&#xff09;和前沿技术的…...

Spring数据访问模块设计

前面我们已经完成了IoC和web模块的设计&#xff0c;聪明的码友立马就知道了&#xff0c;该到数据访问模块了&#xff0c;要不就这俩玩个6啊&#xff0c;查库势在必行&#xff0c;至此&#xff0c;它来了。 一、核心设计理念 1、痛点在哪 应用离不开数据&#xff08;数据库、No…...

laravel8+vue3.0+element-plus搭建方法

创建 laravel8 项目 composer create-project --prefer-dist laravel/laravel laravel8 8.* 安装 laravel/ui composer require laravel/ui 修改 package.json 文件 "devDependencies": {"vue/compiler-sfc": "^3.0.7","axios": …...

Python 包管理器 uv 介绍

Python 包管理器 uv 全面介绍 uv 是由 Astral&#xff08;热门工具 Ruff 的开发者&#xff09;推出的下一代高性能 Python 包管理器和构建工具&#xff0c;用 Rust 编写。它旨在解决传统工具&#xff08;如 pip、virtualenv、pip-tools&#xff09;的性能瓶颈&#xff0c;同时…...

Spring是如何解决Bean的循环依赖:三级缓存机制

1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间‌互相持有对方引用‌,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...

Python Ovito统计金刚石结构数量

大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...

深入理解Optional:处理空指针异常

1. 使用Optional处理可能为空的集合 在Java开发中&#xff0c;集合判空是一个常见但容易出错的场景。传统方式虽然可行&#xff0c;但存在一些潜在问题&#xff1a; // 传统判空方式 if (!CollectionUtils.isEmpty(userInfoList)) {for (UserInfo userInfo : userInfoList) {…...