Java源码解析|String源码与常用方法

String源码与常用方法

1.栗子

代码:

public class JavaStringClass {

    public static void main(String[] args) {
        String s ="hello";
        s = "world"; //内存地址已经修改 原来地址上的值还是不变的
        String s2 = "hello"; //从常量值中找到并引用
        String s4 = new String("hello"); //new产生一个新的对象 不会从常量池中引用
        String s6 = "hel" + "lo";
        String s7 = new String("hel") + new String("lo");
        System.out.println(s == s2); //引用相等
        System.out.println(s2 == s4); //引用不相等
        System.out.println(s2.equals(s4)); //值相等 分析equals方法源码
        System.out.println(s2 == s6); //引用相等 使用相 常量相加 也是从常量池中找到引用
        System.out.println(s2 == s7); //引用不相等

        //开头字母大写
        System.out.println(s2.substring(0,1).toUpperCase() + s2.substring(1));

        System.out.println(s2);
    }
}

输出:

2.String的不变性

2-1:字符串常量池

String s ="hello";
String s2 = "hello";
System.out.println(s == s2); //true  说明引用相等(地址相等)

`s与s2引用相等即地址相等,原因是:Java把字符串常量存入字符串常量池

而 String s4 = new String("hello"); s4和 s2的值不相等,是因为new会产生一个新的对象,不会从字符串常量池中找引用

2-2:String的不变性

`主要是因为 String 和保存数据的 char 数组,都被 final 关键字所修饰,所以是不可变的

如下图所示:被final关键字修饰的变量(这里是字符数组),值不可以改变

所以当 String s ="hello"; s = “world”时,s的内存地址(s的引用对象)已经改变了,
说明产生了新的字符串对象,已经不再指向字符串常量池的“hello”,而是指向了“world”。

3.String重写equal方法,判断相等

1.先判断 引用是否相等 this == object,地址引用相等说明指向同一个对象,那么值肯定也相等了
2.再使用instanceof 判断类型是否与String类型相等
3.最后逐个判断 底层字符数组中的每一个字符是否相等

public boolean equals(Object anObject) {
    // 判断内存地址是否相同
    if (this == anObject) {
        return true;
    }
    // 待比较的对象是否是 String,如果不是 String,直接返回不相等
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = value.length;
        // 两个字符串的长度是否相等,不等则直接返回不相等
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            // 依次比较每个字符是否相等,若有一个不等,直接返回不相等
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}

4.String常用的操作方法

4-1:字符串截取:
方法1:public String substring(int beginIndex, int endIndex) // beginIndex:开始位置,endIndex:结束位置;
方法2:public String substring(int beginIndex) //beginIndex:开始位置一直到到字符串末尾结束位置
截取原理:substring 方法的底层使用的是字符数组范围截取的方法 :Arrays.copyOfRange(字符数组,开始位置,结束位置)

4-2:字符串大小写:
小写方法:String.toLowerCase() //小写
大写方法:String.toUpperCase()//大写

上面两种常用方法的综合应用:将首字母小写
有时候我们会通过 applicationContext.getBean(className); 这种方式得到 SpringBean,这时 className 必须是要满足首字母小写的

name.substring(0, 1).toLowerCase() name.substring(1);

4-3:字符串替换、删除

public void testReplace(){
  String str ="hello word !!";
  log.info("替换之前 :{}",str);
  str = str.replace('l','d');
  log.info("替换所有字符 :{}",str);
  str = str.replaceAll("d","l");
  log.info("替换全部 :{}",str);
  str = str.replaceFirst("l","");
  log.info("替换第一个 l :{}",str);
}
//输出的结果是:
替换之前 :hello word !!
替换所有字符 :heddo word !!
替换全部 :hello worl !!
替换第一个 :helo worl !!

4-4:字符串拆分

String s ="boo:and:foo";
// 我们对 s 进行了各种拆分,演示的代码和结果是:
s.split(":") 结果:["boo","and","foo"]
s.split(":",2) 结果:["boo","and:foo"]
s.split(":",5) 结果:["boo","and","foo"]
s.split(":",-2) 结果:["boo","and","foo"]
s.split("o") 结果:["b","",":and:f"]
s.split("o",2) 结果:["b","o:and:foo"]

但是会拆分出空值

String a =",a,,b,";
a.split(",") 结果:["","a","","b"]

使用google的Guava快速去除空值

String a =",a, ,  b  c ,";
// Splitter 是 Guava 提供的 API
List<String> list = Splitter.on(',')
    .trimResults()// 去掉空格
    .omitEmptyStrings()// 去掉空值
    .splitToList(a);
log.info("Guava 去掉空格的分割方法:{}",JSON.toJSONString(list));
// 打印出的结果为:
["a","b  c"]

4-5:字符串合并

合并我们使用 join 方法,此方法是静态的,我们可以直接使用。
方法有两个入参,参数一是合并的分隔符,参数二是合并的数据源

不足:join不能连续合并(不能链式合并),无法过滤Join对象是List时的null值

解决办法:使用Guava 提供的 API,Joiner快速合并

// 依次 join 多个字符串,Joiner 是 Guava 提供的 API
Joiner joiner = Joiner.on(",").skipNulls();
String result = joiner.join("hello",null,"china");
log.info("依次 join 多个字符串:{}",result);

List<String> list = Lists.newArrayList(new String[]{"hello","china",null});
log.info("自动删除 list 中空值:{}",joiner.join(list));

// 输出的结果为;
依次 join 多个字符串:hello,china
自动删除 list 中空值:hello,china

原文地址:https://www.cnblogs.com/fisherss/p/11626108.html

时间: 2024-08-10 23:20:23

Java源码解析|String源码与常用方法的相关文章

GlusterFS源码解析 —— GlusterFS 源码安装

安装环境: CentOS6.2 glusterfs-3.4.3 GlusterFS 挂载需要 fuse 支持,如果你的内核版本低于 2.6.16 则需要下载fuse的源码包自行编译安装,也可下载 fuse 的rpm包.安装fuse的方法我就不说了,不会源码安装的直接去rpmfind.net上下载rpm即可.高于此版本的内核中已经有了fuse.ko的模块,需要的时候可以执行以下命令进行加载: modprobe -b fuse 1.下载GlusterFS的源码包,目前已经有更新版本 : wget h

JDK 1.8 源码解析 String、StringBuilder和StringBuffer的异同

JDK提供了String.StringBuilder和StringBuffer这三个类来处理字符串,其中StringBuilder类是在JDK 1.5中新增的. 不同点如下: 1 是否有父类 String没有父类. // String类不能被继承 // 实现了Serializable.Comparable和CharSequence(字符序列)接口 public final class String implements java.io.Serializable, Comparable<Strin

Java生成二维码解析二维码

package QrCode; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; import java.util.HashMap; import java.util.Map; import javax.imageio.ImageIO; import com.google.zxing.BarcodeFormat; imp

java生成和解析二维码

前言 现在,二维码的应用已经非常广泛,在线生成器也是诸多,随手生成. 所以就和大家分享一个小案例,用zxing来做一个的二维码生成器,当然这个例子是比较简单,若是写的不好请多多包涵. ZXING项目是谷歌推出的用来识别多种格式条形码的开源项目,项目地址为https://github.com/zxing/zxing. 1.加载zxing依赖 用idea新建个maven项目,pom.xml添加对应的依赖 <dependency> <groupId>com.google.zxing<

Curator源码解析(一)源码结构和测试程序

Curator是Netflix开源的一套ZooKeeper客户端框架. Netflix在使用ZooKeeper的过程中发现ZooKeeper自带的客户端太底层, 应用方在使用的时候需要自己处理很多事情, 于是在它的基础上包装了一下, 提供了一套更好用的客户端框架. Netflix在用ZooKeeper的过程中遇到的问题, 我们也遇到了, 所以开始研究一下, 首先从他在github上的源码, wiki文档以及Netflix的技术blog入手. 看完官方的文档之后, 发现Curator主要解决了三类

spring源码解析——spring源码导入eclipse

一.前言     众所周知,spring的强大之处.几乎所有的企业级开发中,都使用了spring了.在日常的开发中,我们是否只知道spring的配置,以及简单的使用场景.对其实现的代码没有进行深入的了解.开卷有益,在我们空闲的时间里面阅读一下spring的源码,对提升我们的自身能力还是还有很大的帮忙.下面总结一下spring源码导入eclipse的具体的操作. 二.spring的特点 spring的的核心就是IOC(控制反转)和AOP(基于切面的编程) 事务管理方面采用了:声明式事务 为各种主流

MIT 2012分布式课程基础源码解析一-源码概述

课程主页 课程介绍:本课程会在给出的源码的基础上要求完成8个lab lab overviewLab 1 - Lock ServerLab 2 - Basic File ServerLab 3 - MKDIR, UNLINK, and LockingLab 4 - Caching Lock ServerLab 5 - Caching Extent Server + ConsistencyLab 6 - PaxosLab 7 - Replicated lock serverLab 8 - Proje

6、redux源码解析 - 中间件源码剖析

//源码使用方法 import {createStore, applyMiddleWare} from 'redux'; //createStore.js的源码 export default function createStore(reducer, preloadState, enhancer) { //... if(typeof enhancer !== 'undefined'){ if(typeof enhancer !== 'function'){ throw new Error('..

Dialog与FragmentDialog源码解析

<代码里的世界> -UI篇 用文字札记描绘自己 android学习之路 转载请保留出处 by Qiao http://blog.csdn.net/qiaoidea/article/details/46402845 [导航] - 弹出式对话框各种方案 从仿QQ消息提示框来谈弹出式对话框的实现方式 (Dialog,PopupWind,自定义View,Activity,FragmentDialog) - Dialog源码解析 从源码上看Dialog与DialogFragment 1.概述 前一篇写了