《Java小子怒闯数据结构九重天》第二重天-字符串

《Java小子怒闯数据结构九重天》第二重天-字符串

前言

自古以来数据结构界就分为九重天,据说冲破这九重天之后就可以去进攻算法界最终修炼最后成大佬,受万人敬仰。

但是这谈何容易,因为每一重天都有神兽把守,想要冲破每一重天都必须收服守护的神兽才行。

守护九重天的神兽分别是:数组字符串、栈、队列、链表、树、散列表、堆、图。可见他们的战斗力也是逐层增强的。想只凭靠自身的能力拿下他们谈何容易。

不过大家不必惊慌,我这里有一本上古秘籍《Java小子怒闯数据结构九重天》,里面有每一重天神兽的攻略。只要修炼者仔细钻研里面的每一篇,对九重天了如指掌之后,冲破这九重天也是易如反掌的。

今天为大家带来的是第二重天的攻略!

注意:所有资料都是免费提供!大家领取之后不要做收藏党哦!,获得更多自学java咨询!

目录

  • 前言
  • 1.字符串基础知识
  • 2.StringBuilder类
  • 3.初始化String类
  • 4.String类常用API
  • 5.字符串进阶练习
  • 结语

1.字符串基础知识

串(string)是由零个或多个字符组成的有限序列,又名字符串。

我们可以从这段段基本概念中知道:

  • 零个或多个字符组成:说明字符串的内部元素类型为字符。
  • 有限:说明字符串的内容长度是有一定限制的,小于一个最大范围,但是在该范围内,实际的长度是不确定的。
  • 序列:说明字符串中的相邻字符存在前驱和后继的关系。

在Java中没有内置的字符串类型,每个用双引号括起来的字符串都是Java中String类的一个实例。

也就是说Java中String并不是数据类型,在Java中所有的字符串都是String的实例化对象

Java中的String

Java中的String类代表字符串,Java 程序中的所有字符串文字(例如“abc”)都为此类的实例。

也就是说,Java程序中所有的双引号字符串,都是 String 类的对象。String 类在 java.lang 包下,所以使用的时候不需要导包!

在Java中String最重要的特点就是:

String 类是不可改变的,所以你一旦创建了 String 对象,那它的值就无法改变了。我们将这种特性称为String的不可变性。

字符串的不可变性

不可变性:当你给一个字符串重新赋值之后,老值并没有在内存中销毁,而是重新开辟一块空间存储新值。

也就是说一旦一个String对象在内存中创建,它将是不可改变的,所有的String类中方法并不是改变String对象自己,而是重新创建一个新的String对象。

例如:

String s=”dcm”;String s=”ddccmm”

当s的值发生改变之后,ddccmm这个值并没有覆盖dcm,只是重新开发了一个新的空间去储存ddccmm然后将s指向它。

如果我们在实际开发中对含有大量字符的字符串进行遍历赋值修改,会对内存中产生很多无法释放的字符串对象,造成内存垃圾。

正因为String对象的不可变性,如果需要对字符串进行大量的修改、添加字符、删除字符等操作尽量不要使用String对象,因为这样会频繁的创建新的对象导致程序的执行效率下降。

这时我们可以使用Java中另外一个字符串类StringBuilder。

我们在做题的时候关于字符串一般用的都是String类,但是考虑到我们有时候也会用到StringBuilder类这里我就对StringBuilder类进行稍微细致一点的讲解。

2.StringBuilder类

StringBuilder 是一个可变的字符串类,我们可以把它看成是一个容器,这里的可变指StringBuilder对象中的内容是可变的。

2.1 StringBuilder类常用的方法

可以看出来,构建一个StringBuilder的对象只能使用它的构造方法来构造,不像String一样可以直接String s= “123”来创建

因为StringBuilder类对象是可变的,所以当我们对一个字符串需要进行改变比较多的时候一般定义为StringBuilder类。

2.2 String和StringBuilder的区别

String 对象是不可改变的。每次使用String 类中的方法之一时,都要在内存中创建一个新的字符串对象,这就需要为该新对象分配新的空间。

StringBuilder对象是动态对象,允许扩充它所封装的字符串中字符的数量,但是您可以为它可容纳的最大字符数指定一个值,当修改 StringBuilder 时,在达到容量之前,它不会为其自己重新分配空间。当达到容量时,将自动分配新的空间且容量翻倍。也就是说当对字符串进行改变时,都是对当前对象的状态进行更新的。

可以使用重载的构造函数之一来指定 StringBuilder 类的容量。

2.3 String类与StringBuilder类的相互转换

String类转换为StringBuilder类

public class String{ public static void main(String[] args){ String s = “baibai”; StringBuilder s1 = new StringBuilder(s); System.out.println(s1); }}

StringBuilder类转换为String类

public class String { public static void main(String[] args){ StringBuilder s1 = new StringBuilder(); //连续连接 s1.append(“abc”).append(“efg”); String s = s1.toString(); System.out.println(s); }}

3.初始化String类

3.1初始化String对象的两种方法:

//方法一:直接创建String s1= “大聪明 超牛的”;//方法二:对象创建String s2 = new String(“大聪明 超牛的”); String s3 = new String();//也可以创建一个空串

虽然两种方法看起来是一样的但是本质上是不一样的。String 创建的字符串存储在公共池中,而 new 创建的字符串对象在堆上。那存放在公共池(常量池)与堆中有什么不一样吗?

我们来举个例子:

String s1 = “大聪明 超牛的”; // String 直接创建String s2 = “大聪明 超牛的”; // String 直接创建String s3 = s1; // 相同引用String s4 = new String(“大聪明 超牛的”); // String 对象创建String s5 = new String(“大聪明 超牛的”); // String 对象创建System.out.println(System.identityHashCode(s1));System.out.println(System.identityHashCode(s2));System.out.println(System.identityHashCode(s3));System.out.println(System.identityHashCode(s4));System.out.println(System.identityHashCode(s5));

输出:

可见前三个字符串的地址相同,后两个各不相同!

这是因为直接创建的字符串时,会先在公共池中找有没有这样的字符串,如果有那就将引用直接指向它,而不去开发新的空间。在这里s1,s2,s3这三个引用指向了公共池中的同一块内存。

对象创建时,每次都会在堆上开新的空间来存放字符串,也就是说s4,s5分别指向在堆上的两块不同的内存,只不过这两块内存里面都储存着相同的东西。

4.String类常用API

这里再次强调一下,我们在做题的时候遇到关于字符串相关题目我们几乎都是使用String类来解决问题,除了在字符串进行大量更改时我们可能会暂时用到StringBuilder类。

这里的暂时就是我们在对字符串更改等操作之后一般还是要把字符串转换为String类的。

所以我们要学习的API主要还是String类的API。对应刷题我StringBuilder的API我们只需要学习上面提到的两个就够了。

String 类在 java.lang 包下,所以使用的时候不需要导包!

4.1 基本数据类型转换成字符串

有三种办法:

(1)基本类型数据的值+“” (最常用,最简单);(2)使用包装类中的静态方法 static String toString(int i)返回一个表示指定整数的String 对象。如:在Integer中:Integer.toString(6);(3)使用String类中的静态方法 static String valueOf(int i) 返回int 参数的字符串表示形式。如:String.valueOf(6);

String 类别中已经提供了将基本数据型态转换成 String 的 static 方法也就是 String.valueOf() 这个参数多载的方法 :

String.valueOf(boolean b) //将 boolean 变量 b 转换成字符串 String.valueOf(char c) //将 char 变量 c 转换成字符串 String.valueOf(char[] data)//将 char 数组 data 转换成字符串 String.valueOf(char[] data, int offset, int count)//将char数组data中由data[offset]开始取 count个元素转换成字符串 String.valueOf(double d)//将 double 变量 d 转换成字符串 String.valueOf(float f)//将 float 变量 f 转换成字符串 String.valueOf(int i)//将 int 变量 i 转换成字符串 String.valueOf(long l)//将 long 变量 l 转换成字符串 String.valueOf(Object obj)//将 obj 对象转换成 字符串, 等于 obj.toString()

因为是静态方法所以不需要实例化。

4.2 字符串转换为基本数据类型

一般使用包装类的静态方法parseXX(“字符串”)

要将 String 转换成基本数据类型大多需要使用基本数据型态的包装类别,如:String 转换成 byte可以使用 Byte.parseByte(String s)

Byte.parseByte(String s)//将 s 转换成 byte Byte.parseByte(String s, int radix)//以 radix 为基底 将 s 转换为 byteDouble.parseDouble(String s)//将 s 转换成 double Float.parseFloat(String s)//将 s 转换成 float Integer.parseInt(String s)//将 s 转换成 int Long.parseLong(String s)//将 s 转换成 long

注意这里也是静态方法,只不过都是对应包装类的静态方法

4.3 使用length()得到一个字符串的字符个数

int len = String.length();

4.4 使用toCharArray() 将一个字符串转换成字符数组

Char[] arr = String.toCharArray();

4.5 判断两个字符串的内容是否相等返回true/false

String1.equals(String2);//区分大小写String1.equalsIgnoreCase(String2);//不区分大小写

4.6 与位置相关的字符串

charAt(int)//得到指定下标位置对应的字符indexOf(String)//得到指定内容第一次出现的下标lastIndexOf(String)//得到指定内容最后一次出现的下标

4.7 将一个字符串按照指定内容劈开split(String) ,返回字符串数组。

String s = “wa,dcm,nb!”;String[] str = s.split(“,”);//返回结果中str[1]=dcm

4.8 contains(String) 判断一个字符串里面是否包含指定的内容,返回true/false

Boolean a = String1.contains(String2)

4.9 使用substring()截取字符串,返回子串

String.substring(int)//从指定下标开始一直截取到字符串的最后String.substring(int,int)//从下标x截取到下标y-1对应的元素

4.10 字符串大小写转换

String.toUpperCase()//将一个字符串全部转换成大写String.toLowerCase()//将一个字符串全部转换成小写

4.11 使用replace()进行字符串内容替换

String.replace(String,String)//将某个内容全部替换成指定内容String.replaceAll(String,String)//将某个内容全部替换成指定内容,支持正则String.repalceFirst(String,String)//将第一次出现的某个内容替换成指定的内容

5.字符串进阶练习

387. 字符串中的第一个唯一字符

题解:

把字符串的单个字符转化为对应数组下标,遍历一遍字符串获得26个字母分别出现几次。然后在遍历一遍字符串看哪个字符先出现次数为1,就输出对应下标。

class Solution { public int firstUniqChar(String s) { int len = s.length(); int[] vis = new int[26]; int temp = -1; for(int i = 0; i < len; i ++) { vis[s.charAt(i) – 'a'] ++; } for(int i = 0; i < len; i ++) { if(vis[s.charAt(i) – 'a'] == 1) { return i; } } return -1; }}

或者我们也可以把字符串先转换为字符数组来解题,原理都是一样的!

class Solution { public int firstUniqChar(String s) { int[] arr = new int[26]; char[] chars = s.toCharArray(); for (int i = 0; i < chars.length; i++) { arr[chars[i] – 'a']++; } for (int i = 0; i < chars.length; i++) { if (arr[chars[i] – 'a'] == 1) { return i; } } return -1; }}

结语

恭喜你修炼到这里,你已经基本有了收服神兽字符串的能力。神兽字符串是我们到进攻算法界最基础的能力之一。大家不可懈怠。

感兴趣的修炼者可以关注下面公众号,会持续推送最新更新的!

持续更新中…

原文链接:https://mp.weixin.qq.com/s/5-uS4SGYLzqilLX8HkXBqw

郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
(0)
用户投稿
上一篇 2022年6月20日
下一篇 2022年6月20日

相关推荐

  • 一个人走出低谷的9个好方法

    你有过低谷期的经历吗? 感情破裂,家人重病,被公司裁员,创业失败…… 在低谷期,你或许会质疑自己的能力,埋怨命运的不公,觉得前路渺茫,看不到希望。 但,诚如曼德拉所说: “生命最伟…

    2022年6月1日
  • 内存数据库

    内存数据库从范型上可以分为关系型内存数据库和键值型内存数据库。 在实际应用中内存数据库主要是配合oracle或mysql等大型关系数据库使用,关注性能。 作用类似于缓存,并不注重数…

    2022年6月19日
  • Spring项目中的@Transactional事务管理

    项目中,很多时候数据是相对的, 例如:用户A付钱给用户B,那么用户A的账户需被扣钱。所扣的钱需被加到用户B的账户上; 但是在项目中,我们的代码走向是,先扣除了用户A的账户钱,这个时…

    2022年6月16日
  • 苹果手机上实现微信双开,完美解决iPhone ios系统微信分身

    #谣零零计划# 微信是手机上必不可少的社交软件,安卓手机有自带的分身软件可以多开微信1-2个,但是苹果就不一样了,由于安全性的保障,苹果还没有出官方的渠道,这就让工作号和私人号的用…

    2022年6月27日
  • 中国首个国际标识代码“MA”正式纳入工业互联网标识体系

    中新网6月28日电 题:中国首个国际标识代码“MA”正式纳入工业互联网标识体系 中新财经记者 王恩博 记者28日自中关村工信二维码技术研究院(下称“中码院”)获悉,工业和信息化部已…

    2022年7月10日
  • Mongodb和mysql的区别

    Mongodb和mysql的区别 1.Mongodb简介及优缺点分析 Mongodb是非关系型数据库(nosql ),属于文档型数据库。文档是mongoDB中数据的基本单元,类似关…

    2022年6月22日
  • 现代编程语言都具备的Lambda到底是个啥?详聊Lambda与函数式接口

    1. Lambda 咱们首先来说说 Lambda 这个名字,Lambda 并不是一个什么的缩写,它是希腊第十一个字母 λ 的读音,同时它也是微积分函数中的一个概念,所表达的意思是一…

    2022年7月24日
  • 重磅!2022-2023跨境出口电商白皮书(营销、渠道、布局、趋势)

    来源:Morketing研究院 此份报告可以帮助卖家在不确定的环境下寻找当前行业中的确定性,找到长期的增长点 亮点内容: 亮点一、共28家受调研方,囊括跨境电商卖家、服务机构、业内…

    2022年8月29日
  • 使用Windbg静态分析dump文件(实战经验总结)

    在日常分析C++软件异常的日常工作中,大多数情况下我们都是使用Windbg去静态分析dump文件去排查软件异常的,今天我们就来详细地讲一下如何使用Windbg去静态分析dump文件…

    2022年7月12日
  • 原来身边就有人被割韭菜的

    最近见了一位好朋友,他正想从传统行业转型,但又不知道做什么好,比较迷茫。 平常他也有关注我的公众号文章,所以就想找我了解一下跨境电商的情况。 从下午三点见面开始聊,聊到晚上十二点,…

    2022年7月26日

联系我们

联系邮箱:admin#wlmqw.com
工作时间:周一至周五,10:30-18:30,节假日休息