30天学习Java第五天——数组 字符串

news2025/3/17 19:44:38

数组

一维数组

  1. 定义
  • int[] anArray;
  • int anOtherArray[];
  • 初始化int anOtherArray[] = new int[] {1, 2, 3, 4, 5};
  1. 访问
  • anArray[0] = 10;
  • 可变数组:void varargsMethod(String... varargs) {}
    该方法可以接收任意数量的字符串参数,可以是 0 个或者 N 个,本质上,可变参数就是通过数组实现的。
  1. 方法
  • 数组排序Arrays.sort()
    • 基本数据类型按照升序排列
    • 实现了 Comparable 接口的对象按照 compareTo() 的排序
  • 数组复制:System.arraycopy(原数组,从原数组复制开始的索引,新数组,复制到新数组的开始索引,复制长度)

二维数组

  1. 定义
    data_type[][] array_name;
    int[][] oddNumbers = { {1, 3, 5, 7}, {9, 11, 13, 15} };

  2. 访问
    oddNumbers[0][0]

  3. 打印数组
    直接打印数组输出的是其地址,需要使用一些方法打印

  • stream 流打印 Java 数组

    第一种:Arrays.asList(arr).stream().forEach(s -> System.out.println(s));
    第二种:Stream.of(cmowers).forEach(System.out::println);
    第三种:Arrays.stream(cmowers).forEach(System.out::println);

  • for 循环打印 Java 数组
  • Arrays 工具类打印 Java 数组

    String [] arr= {“1”,“1”,“2”};
    System.out.println(Arrays.toString(arr));

  • Arrays工具类打印二维数组
    Arrays.deepToString(arr)

字符串

  1. String类声明
    • 是final的,不能被继承
    • 实现了 Serializable 接口,可以序列化
    • 实现了 Comparable 接口,意味着最好不要用‘==’来比较两个字符串是否相等,而应该用 compareTo() 方法去比较。
    • String 和 StringBuffer、StringBuilder 一样,都实现了 CharSequence 接口,所以它们仨属于近亲。由于 String 是不可变的,所以遇到字符串拼接的时候就可以考虑一下 String 的另外两个好兄弟,StringBuffer 和 StringBuilder,它俩是可变的。
  2. String 类的 hashCode 方法
    每一个字符串都会有一个 hash 值。
  3. String 类的 substring 方法
    用来截取字符串的
  • 提取一段
    String str = "Hello, world!";
    String subStr = str.substring(7, 12);  // 从第7个字符(包括)提取到第12个字符(不包括)
    System.out.println(subStr);  // 输出 "world"
    
  • 提取前缀,后缀
    String str = "Hello, world!";
    String prefix = str.substring(0, 5);  // 提取前5个字符,即 "Hello"
    String suffix = str.substring(7);     // 提取从第7个字符开始的所有字符,即 "world!"
    
  • 处理字符串中的空格,分隔符
    String str = "   Hello,   world!  ";
    String trimmed = str.trim();                  // 去除字符串开头和结尾的空格
    String[] words = trimmed.split("\\s+");       // 将字符串按照空格分隔成单词数组
    String firstWord = words[0].substring(0, 1);  // 提取第一个单词的首字母
    System.out.println(firstWord);                // 输出 "H"
    
  • 处理字符串中的数字,符号
    String str = "1234-5678-9012-3456";
    String[] parts = str.split("-");             // 将字符串按照连字符分隔成四个部分
    String last4Digits = parts[3].substring(1);  // 提取最后一个部分的后三位数字
    System.out.println(last4Digits);             // 输出 "456"
    
  1. String 类的 indexOf 方法
    indexOf 方法用于查找一个子字符串在原字符串中第一次出现的位置,并返回该位置的索引。
  • 查找某个字符的位置
    String str = "Hello, world!";
    int index = str.indexOf(",");     // 查找逗号在 str 中第一次出现的位置
    System.out.println(index);        // 输出 5
    
  • 从指定位置查找子串位置
    String str = "Hello, world!";
    int index = str.indexOf("l", 3);  // 从索引为3的位置开始查找 "l" 子字符串在 str 中第一次出现的位置
    System.out.println(index);        // 输出 3
    
  1. 其他方法
  • length() 用于返回字符串长度。
  • isEmpty() 用于判断字符串是否为空。
  • charAt() 用于返回指定索引处的字符。
  • valueOf() 用于将其他类型的数据转换为字符串。
  • trim() 用于去除字符串两侧的空白字符
  • getBytes() 用于返回字符串的字节数组
  • toCharArray() 用于将字符串转换为字符数组。
  • concat() 方法用于拼接字符串
String 对象一旦被创建后就固定不变了,对 String 对象的任何修改都不会影响到原来的字符串对象,都会生成新的字符串对象。

字符串常量池

String s = new String("王二");
这行代码在创建String时,JVM会先在字符串常量池中找“王二”,如果找到了就直接在堆中创建一个对象。如果没找到,就先在字符串常量池中创建一个对象保存“王二”,然后再在堆中创建一个对象。
如此一来是多此一举吗?
注意,上面我们创建字符串使用的是new对象的方式,但是通常我们都是直接赋值String s = "张三";,这样的方法创建的字符串,JVM会先去字符串常量池里找,如果找到了就不创建对象了,如果没找到就在字符串常量池中创建一个对象,这样一来就能节省堆内存。

  • 使用双引号声明的字符串对象会保存在字符串常量池中。
  • 使用 new 关键字创建的字符串对象会先从字符串常量池中找,如果没找到就创建一个,然后再在堆中创建字符串对象;如果找到了,就直接在堆中创建字符串对象。
  • 调用 intern() 方法可以将new出来的字符串放入字符串常量池中
    Java7之后,由于字符串常量池放在了堆中,执行 String.intern() 方法的时候,如果对象在堆中已经创建了,字符串常量池中就不需要再创建新的对象了,而是直接保存堆中对象的引用,也就节省了一部分的内存空间。

StringBuilder和StringBuffer

由于字符串是不可变的,因此我们对于字符串的操作,都需要创建很多新的字符串,这样一来造成了过多的空间浪费,尤其是字符串拼接时,因此StringBuffer和StringBuilder应运而生。

由于 StringBuffer 操作字符串的方法加了 synchronized 关键字进行了同步,主要是考虑到多线程环境下的安全问题,所以如果在非多线程环境下,执行效率就会比较低,因为加了没必要的锁。

于是只用于单线程的StringBuilder出现了,其功能与StringBuffer功能类似。
实际开发中,StringBuilder 的使用频率也是远高于 StringBuffer,甚至可以这么说,StringBuilder 完全取代了 StringBuffer。

public final class StringBuffer extends AbstractStringBuilder implements Serializable, CharSequence {

    public StringBuffer() {
        super(16);
    }
    
    public synchronized StringBuffer append(String str) {
        super.append(str);
        return this;
    }

    public synchronized String toString() {
        return new String(value, 0, count);
    }

    // 其他方法
}
  • Java对于字符串拼接的操作
    当编译器遇到 + 号这个操作符的时候,会将 new String(“二哥”) + new String(“三妹”) 这行代码解释为以下代码:new StringBuilder().append("二哥").append("三妹").toString();

  • StringBuilder的内部实现 toString()方法

    public String toString() {
    	    return new String(value, 0, count);
    	}
    

    value 是一个 char 类型的数组,在 StringBuilder 对象创建时,会为 value 分配一定的内存空间(初始容量 16),用于存储字符串。随着字符串的拼接,value 数组的长度会不断增加,因此在 StringBuilder 对象的实现中,value 数组的长度是可以动态扩展的,就像ArrayList那样。

    count 是一个 int 类型的变量,表示字符序列的长度。toString() 方法会调用 new String(value, 0, count),使用 value 数组中从 0 开始的前 count 个元素创建一个新的字符串对象,并将其返回。

  • StringBuilder的内部实现 toString()方法
    append(String str) 方法将指定字符串追加到当前字符序列中。如果指定字符串为 null,则追加字符串 “null”;否则会检查指定字符串的长度,然后根据当前字符序列中的字符数和指定字符串的长度来判断是否需要扩容。

    public StringBuilder append(String str) {
        super.append(str);
        return this;
    }
    

    如果需要扩容,则会调用 ensureCapacityInternal(int minimumCapacity) 方法。扩容之后,将指定字符串的字符拷贝到字符序列中。

    private void ensureCapacityInternal(int minimumCapacity) {
        // 不够用了,扩容
        if (minimumCapacity - value.length > 0)
            expandCapacity(minimumCapacity);
    }
    
    void expandCapacity(int minimumCapacity) {
        // 扩容策略:新容量为旧容量的两倍加上 2
        int newCapacity = value.length * 2 + 2;
        // 如果新容量小于指定的最小容量,则新容量为指定的最小容量
        if (newCapacity - minimumCapacity < 0)
            newCapacity = minimumCapacity;
        // 如果新容量小于 0,则新容量为 Integer.MAX_VALUE
        if (newCapacity < 0) {
            if (minimumCapacity < 0) // overflow
                throw new OutOfMemoryError();
            newCapacity = Integer.MAX_VALUE;
        }
        // 将字符序列的容量扩容到新容量的大小
        value = Arrays.copyOf(value, newCapacity);
    }
    
    
  • StringBuilder的 reverse 方法
    StringBuilder 还提供了一个 reverse 方法,用于反转当前字符序列中的字符。

判断字符串是否相等

  1. equals()方法与 ==
    s1.equals(s2)判断的是两个字符串内容是否相等, ==判断的是两个字符串是否是同一个对象
    使用equals()之前要先判空,否则可能会出现空指针异常问题a != null && a.equals(b)
  2. String 类的 .contentEquals()
    .contentEquals() 的优势在于可以将字符串与任何的字符序列(StringBuffer、StringBuilder、String、CharSequence)进行比较。

拼接字符串

  1. String.append拼接字符串
    + 操作符的本质:

    ’ + ’ 号操作符其实被 Java 在编译的时候重新解释了,换一种说法就是,+ 号操作符是一种语法糖,让字符串的拼接变得更简便了

    循环体内,拼接字符串最好使用 StringBuilder 的 append() 方法,而不是 + 号操作符。原因就在于循环体内如果用 + 号操作符的话,就会产生大量的 StringBuilder 对象,不仅占用了更多的内存空间,还会让 Java 虚拟机不停的进行垃圾回收,从而降低了程序的性能。

  2. String.concat 拼接字符串

    String chenmo = "沉默";
    String wanger = "王二";
    System.out.println(chenmo.concat(wanger));
    

    和 + 号操作符相比,concat() 方法在遇到字符串为 null 的时候,会抛出 NullPointerException,而“+”号操作符会把 null 当做是“null”字符串来处理。

    如果拼接的字符串是一个空字符串(“”),那么 concat 的效率要更高一点,毕竟不需要 new StringBuilder 对象。

  3. String.join 拼接字符串
    第一个参数为字符串连接符

    	String chenmo = "沉默";
    	String wanger = "王二";
    	String cmower = String.join("", chenmo, wanger);
    	System.out.println(cmower);
    
  4. StringUtils.join 拼接字符串

    String chenmo = "沉默";
    String wanger = "王二";
    StringUtils.join(chenmo, wanger);
    

    该方法不用担心 NullPointerException,其内部使用的仍然是 StringBuilder。

如何拆分字符串

拆分字符串使用split()方法
首先,需要判断一下拆分符号是否存在字符串中,否则会抛异常
其次,对于位置的字符串,无法确定拆分符号,需要使用正则表达式,下面的文档是正则表达式的学习文档
https://github.com/cdoco/learn-regex-zh
https://github.com/cdoco/common-regex

public class Test {
    public static void main(String[] args) {
        String cmower = "沉默王二,一枚有趣的程序员";
        if (cmower.contains(",")) {
            String [] parts = cmower.split(",");
            System.out.println("第一部分:" + parts[0] +" 第二部分:" + parts[1]);
        } else {
            throw new IllegalArgumentException("当前字符串没有包含逗号");
        }
    }
}
  1. 第一种:使用\\.表示.
    String cmower = "沉默王二.一枚有趣的程序员";
    if (cmower.contains(".")) {
        String [] parts = cmower.split("\\.");
        System.out.println("第一部分:" + parts[0] +" 第二部分:" + parts[1]);
    }
    
  2. 第二种:使用[.]
    cmower.split("[.]");
    
    [] 也是一个正则表达式,用来匹配方括号中包含的任意字符。
  3. 第三种:使用 Pattern 类的 quote() 方法来包裹英文逗点.
    String [] parts = cmower.split(Pattern.quote("."));
    
    当 split() 方法的参数是正则表达式的时候,方法最终会执行下面这行代码:
    return Pattern.compile(regex).split(this, limit);
    也就意味着,拆分字符串有了新的选择,可以不使用 String 类的 split() 方法。
    public class TestPatternSplit {
        private static Pattern twopart = Pattern.compile("\\.");
    
        public static void main(String[] args) {
            String [] parts = twopart.split("沉默王二.一枚有趣的程序员");
            System.out.println("第一部分:" + parts[0] +" 第二部分:" + parts[1]);
        }
    }
    

split() 方法可以传递 2 个参数,第一个为分隔符,第二个为拆分的字符串个数。


注:本文章来源自博主沉默王二的二哥的Java进阶之路学习内容。

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

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

相关文章

【DeepSeek应用】本地部署deepseek模型后,如何在vscode中调用该模型进行代码撰写,检视和优化?

若已成功在本地部署了 DeepSeek 模型(例如通过 vscode-llm、ollama 或私有 API 服务),在 VS Code 中调用本地模型进行代码撰写、检视和优化的完整流程如下: 1. 准备工作:确认本地模型服务状态 模型服务类型: 若使用 HTTP API 服务(如 FastAPI/Flask 封装),假设服务地址…

网络编程、URI和URL的区别、TCP/IP协议、IP和端口、URLConnection

DAY12.1 Java核心基础 网络编程 在互联网时代&#xff0c;网络在生活中处处可见&#xff0c;javaWeb占据了很大一部分 那如何实现javaWeb编程呢&#xff1f; Web编程就是运行在同一个网络下面的终端&#xff0c;使得它们之间可以进行数据传输 计算机网络基本知识 计算机网络…

自探索大语言模型微调(一)

一、数据 1.1、失败案例 Hugging Face&#xff1a; 根据B站上搜索到的资料&#xff0c;datasets这个库可以直接下载丰富的数据集合和与训练模型&#xff0c;调用也非常的简单&#xff0c;唯一的缺点就是&#xff0c;需要外网&#xff08;翻墙&#xff09;&#xff0c;用国内的…

Unity 和 Python 的连接(通过SocketIO)附源码

在游戏或者项目开发中&#xff0c;Unity 通常用于创建前端&#xff0c;而 Python 则因其强大的数据处理能力常被用作后端。通过 Socket.IO&#xff0c;我们可以轻松地实现 Unity 和 Python 的实时通信。本文将介绍如何通过 Socket.IO 连接 Unity 和 Python&#xff0c;并附上完…

89.HarmonyOS NEXT 应用安全与隐私保护指南:构建安全可靠的应用

温馨提示&#xff1a;本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦&#xff01; HarmonyOS NEXT 应用安全与隐私保护指南&#xff1a;构建安全可靠的应用 文章目录 HarmonyOS NEXT 应用安全与隐私保护指南&#xff1a;构建安全可…

spring boot 发送邮件验证码

一、前置需求 1、准备邮箱 2、登录授权码 qq邮箱在–>设置–>账号POP3/IMAP/SMTP/Exchange/CardDAV/CalDAV服务 开启服务 二、发送邮件 1、简单邮件 包含邮件标题、邮件正文 2、引入mail启动器 <dependency><groupId>org.springframework.boot</groupI…

MySQL连接较慢原因分析及解决措施

文章目录 整体说明一、问题现象二、问题分析2.1、DNS反向解析问题2.2、网络问题2.3、SSL/TLS协商问题2.4、自动补全的延迟 三、问题解决 摘要&#xff1a; MySQL连接较慢原因分析及解决措施 关键词&#xff1a; MySQL、连接缓慢、客户端、参数设置 整体说明 在使用MySQL的时候…

【大模型基础_毛玉仁】2.5 基于 Decoder-only 架构的大语言模型 -- GPT和LLaMa模型介绍

更多内容&#xff1a;XiaoJ的知识星球 目录 2.5 基于 Decoder-only 架构的大语言模型2.5.1 Decoder-only 架构2.5.2 GPT 系列语言模型1&#xff09;初出茅庐&#xff1a;GPT-1 模型2&#xff09;小有所成&#xff1a;GPT-2 模型3&#xff09;崭露头角&#xff1a;GPT-3 模型4&a…

如何解决ChatGPTplus/pro o1/o3模型无法识别图片或者文件,限制次数?

你是否遇到ChatGPTplus无法识别图片、或者无法识别文件&#xff0c;甚至回答很简短&#xff0c;o1不思考&#xff0c;GPT-4o不能联网、分析图片和处理文件&#xff01;感觉非常敷衍。本文教你如何确定自己的账号是否被降智&#xff1b;教你如何降智的原因&#xff1b;教你解决降…

go的gmp

参考链接&#xff1a;https://www.bilibili.com/video/BV19r4y1w7Nx Golang的GMP调度模型(协程调度器)是其并发编程的核心。GMP代表Goroutine、Machine和Processor三个关键组成部分。Goroutine是Go语言中的轻量级线程&#xff0c;Machine是操作系统的线程&#xff0c;Processor…

Vue开发者工具(VueDevtools)下载与安装

一、这里采用极简插件 网址&#xff1a;chrome.zzzmh.cn/index#/inde… 输入 vue.js.devtools并访问 点击推荐下载 下载到本地解压缩 然后把这个文件移动到谷歌插件里面,开启开发者模式 点击详情,开启一些权限 写的一个demo&#xff0c;如果有vue代码的话&#xff0c;就会…

【Linux】https 协议

目录 一、https 协议 二、加密和解密 &#xff08;一&#xff09;为什么需要加密与解密 &#xff08;二&#xff09;加密和解密的过程 &#xff08;二&#xff09;常见的加密方式 1、对称加密 2、非对称加密 3、数据摘要 4、数字签名 三、https 的加密方式 &#xff…

新手村:数据预处理-特征缩放

新手村&#xff1a;数据预处理-特征缩放 特征缩放&#xff08;Feature Scaling&#xff09;是数据预处理中的一个重要步骤&#xff0c;特别是在应用某些机器学习算法时。特征缩放可以使不同尺度的特征具有相同的量级&#xff0c;从而提高模型训练的效率和性能。常见的特征缩放方…

Xinference大模型配置介绍并通过git-lfs、hf-mirror安装

文章目录 一、Xinference开机服务systemd二、语言&#xff08;LLM&#xff09;模型2.1 配置介绍2.2 DeepSeek-R1-Distill-Qwen-32B&#xff08;大杯&#xff09;工具下载git-lfs&#xff08;可以绕过Hugging Face&#xff09; 2.3 DeepSeek-R1-Distill-Qwen-32B-Q4_K_M-GGUF&am…

0x04.若依框架微服务开发(含AI模块运行)

微服务本地开发硬件资源有限&#xff0c;所以会将核心微服务组件先部署在服务器上比如&#xff1a;mysql&#xff0c;redis&#xff0c;注册中心Nacos&#xff0c;网关Gateway&#xff0c;认证中心Auth和upms模块以及低代码生成模块。 mysql、redis部署前篇已讲&#xff0c;这…

判断是不是二叉搜索树(C++)

目录 1 问题描述 1.1 示例1 1.2 示例2 2 解题思路 3 代码实现 4 代码解析 4.1 中序遍历函数 inorder 4.2 主函数 isValidBST 初始化及中序遍历调用 4.3 检查数组中元素是否严格递增 4.4 返回验证结果 5 总结 1 问题描述 给定一个二叉树根节点&#xff0c;请你判断…

Linux--gdb/cgdb

ok&#xff0c;我们今天学习gdb的安装和使用 调试器-gdb/cgdb使用 VS、VScode编写的代码一般都是release格式的&#xff0c;gdb 的格式一般是debug 换成debug模式命令 :-g gdb会记录最新的一条命令&#xff0c;直接回车就是默认执行该命令 一个调试周期下&#xff0c;断点…

超精密工件小孔几何尺寸测量:自动化解决方案

下载链接&#xff1a;&#xff08;最新版本&#xff09;超精密工件小孔几何尺寸测量&#xff1a;自动化解决方案python脚本代码&#xff0c;可直接运行&#xff0c;内包含测试数据&#xff0c;亲测好用资源-CSDN文库 在现代制造业中&#xff0c;超精密工件的质量控制至关重要&a…

Blender-MCP服务源码1-项目解读

Blender-MCP服务源码 有个大佬做了一个Blender-MCP源码&#xff0c;第一次提交代码是【2025年3月7号】今天是【2025年月15日】也就是刚过去一周的时间&#xff0c;所以想从0开始学习这个代码&#xff0c;了解一下大佬们的开发思路 1-核心知识点 1&#xff09;第一版&#xff1…

小程序配置

注册小程序账号和安装开发工具 参考文档&#xff1a;注册小程序账号和安装开发工具https://blog.csdn.net/aystl_gss/article/details/127878658 HBuilder新建项目 填写项目名称&#xff0c;选择UNI-APP&#xff0c;修改路径&#xff0c;点击创建 manifest.json 配置 需要分别…