栈:基础的数据类型变量以及对象的引用
堆:存放new出来的对象
常量池:字符串常量
底层原码中使用final修饰 char[] value来存储字符串的值。字符串常量是存在常量池
中,一旦声明,就不可以改变,同时常量池中不会存储相同内容的字符串,即s1与s2是相等的。
String s1 = "aaa";
s1 = "bbb";
System.out.println(s1);
String s1 = "123";
String s2 = "123";
System.out.println(s1==s2);
输出的s1的值是bbb,不是改变了么?注意这里的s1是引用对象,因此s1存在栈空间中,也就是s1与aaa并不存在同一内存空间中,只是中间连接了一条绳子将aaa指向了s1,现在这条绳子由bbb指向了s1,但是aaa其实还留在了常量池中,所以说字符串常量是不可改变的。
除此之外,String还实现了Serializable接口,表示字符串可以被序列化,还实现了Comparable接口表示字符串可以比较大小
String s1 = null
String s2 = ""
空对象
,并不是字符串,可以赋给任何对象,字符串中表示只是一个引用,还没有内存空间的分配String():构造一个空的字符串
String(byte[] arr): 将字节数组变为一个字符串
String(byte[] arr, int offset, int lengh): 将字节数组部分变为字符串
String(char[] arr): 将char字节数组变为字符串
String(char[] arr, int offset, int length): 将char字节数组部分变为字符串
String(String original): 字符串常量构建字符串
byte[]===>String
byte[] b = {97,98,99,100};
String str = new String(b);
System.out.println(str);//abcd
StringIndexOutOfBoundsException
异常,通常Java中数组索引区间左开右闭的 byte[] b = {97,98,99,100};
String str = new String(b,1,3);
System.out.println(str);//bc
char[]转String等等一些数组转String的方式都和上述大同小异,这些都是对String方法使得构造
理解上面两句:
String s1 = "123";①
String s2 = "123";②
String s3 = new String("123");③
System.out.println(s1==s2);//正确
System.out.println(s1==s3);//错误
只要明白了内存分布,判断不成问题。对于①,②来说,上面已经解释过了都在栈内存中,对于③来说,s3是对象存在Java堆中,s1,s2都是存在Java栈中,所以s1==s3是错误的!!!对于equals()就不一样了三者内容都是相等的。
练习一:
String s1 = new String("hello");
String s2 = new String("hello");
System.out.println(s1==s2);
System.out.println(s1.equals(s2));
String s3 = new String("hello");
String s4 = "hello";
System.out.println(s3==s4);
System.out.println(s3.equals(s4));
String s5 = "hello";
String s6 = "hello";
System.out.println(s5==s6);
System.out.println(s5.equals(s6));
答案:F T F T T T
String s1 = "hello";
String s2 = "world";
String s3 = "helloworld";
System.out.println(s3==(s1+s2));//F 变量的连接存在堆中不相等
System.out.println(s3==(s1+s2).intern());//T 获取的是值相等
System.out.println(s3.equals(s1+s2));//T 获取内容相等
System.out.println(s3=="hello" + "world");//T 常量与常量连接还在常量池中
System.out.println(s3.equals("hello"+"world"));//T 内容相等
字符串的判断:
String s1 = "abcde";
String s2 = "AbCde";
String s3 = "abcde";
//equals
System.out.println(s1.equals(s2));//t
System.out.println(s1.equals(s3));//f
//equalsIgnorecase
System.out.println(s1.equalsIgnoreCase(s2));//t
System.out.println(s1.equalsIgnoreCase(s3));//t
//是否包含指定字符串
System.out.println(s1.contains("bd"));//f
//是否以指定字符串开头
System.out.println(s1.startsWith("ab"));//t
System.out.println(s1.startsWith("cde",2));//t 字符串是否在索引2开始截取
//是否以字符串结尾
System.out.println(s1.endsWith(s3));//t
//是否为空
System.out.println(s1.isEmpty());//false
字符串的获取
注意:如果没有找到都会返回-1
练习:
String str = "abcdeababcdacaeca";
//长度
System.out.println(str.length());//17
//获取第二个元素
System.out.println(str.charAt(1));//b
//获取最后一个元素值
System.out.println(str.charAt(str.length() - 1));//a
//获取'a'第一次的索引值
System.out.println(str.indexOf('a'));//0
System.out.println(str.indexOf(97));//0
//从第二个索引值开始,'a'第一次出现的位置
System.out.println(str.indexOf('a', 2));//5
//获取"bc"第一次出现的索引值
System.out.println(str.indexOf("bc"));//1
//从2索引开始"ad"第一次出现的索引值
System.out.println(str.indexOf("ad", 2));//-1
//获取'a'最后一次存储的索引值
System.out.println(str.lastIndexOf('a'));//16
//获取'bc'最后一次出现的索引值
System.out.println(str.lastIndexOf("bc"));//8
//从7索引结束前获取'a'最后一次索引值
System.out.println(str.lastIndexOf('a', 7));//7
System.out.println(str.lastIndexOf('c', 6));//2
//从索引5截取到末尾
String s1 = str.substring(5);
System.out.println(s1);//ababcdacaeca
//"abcdeabca" ===> "deab"
String s2 = str.substring(3, 7);
System.out.println(s2);
tips:不同编码方式汉字占用的字节不同,GBK编码,一个汉字字符占用2个字节,UTF-8编码中一个字符占用3个字节
String s = "Ab三上悠亚";
//String转为byte[]
byte[] bys =s.getBytes();
//[65, 98, -28, -72, -119, -28, -72, -118, -26, -126, -96, -28, -70, -102]
System.out.println(Arrays.toString(bys));
//String转为char[]
char[] c = s.toCharArray();
//[A, b, 三, 上, 悠, 亚]
System.out.println(Arrays.toString(c));
//其它大部分类型转String
System.out.println(String.valueOf(100));//100
System.out.println(String.valueOf(false));//false
System.out.println(String.valueOf('a'));//a
System.out.println(String.valueOf(3.23f));//3.23
System.out.println(String.valueOf(c,0,3));//Ab三 从0索引开始截取三个字符
char[] data = {'A','B','c'};
System.out.println(String.valueOf(data));//ABc
System.out.println(s.toLowerCase());//ab三上悠亚
System.out.println(s.toUpperCase());//AB三上悠亚
字符串的替换:
String s = "abc三上悠亚abc";
//b替换B 字符替换
String s1 = s.replace('b', 'B');
System.out.println(s1);//aBc三上悠亚aBc
//三上悠亚替换为JAPAN字符串替换
String s2 = s.replace("三上悠亚", "JAPAN");
System.out.println(s2);//abcJAPANabc
//替换指定的字符序列
String s4 = "abc";
String s5 = "abcd";
String s6 = s.replace(s4, s5);
System.out.println(s6);//abcd三上悠亚abcd
字典比较
字符串是英文并且长度不一样
长度长的包含长度短的(这个包含表示从第一个字符开始匹配)直接是长度相减
String str1 = "abcde";
String str2 = "abc";
System.out.println(str1.compareTo(str2));//2
相当于s1.length() - s2.length()
长度不一样,且从第一个字符开始找,没有匹配,则是两个字符ASCALL的差值
String str1 = "abcde";
String str2 = "cabc";
System.out.println(str1.compareTo(str2));//-2
相当于a-c=-2
长度一样且不匹配,单个字符ascall值相减即可,多个字符不一样从第一个字符找,如果第一个字符一样比较第二个字符,依次类推…直到不一样时,ASCALL码值相减即可。
String str1 = "cdea";
String str2 = "cdeb ";
System.out.println(str1.compareTo(str2));//-1
中文与英文类似!!!
编码:将看懂的变为看不懂的
解码:将看不懂的变为看懂的
String s1 = "我爱china";
byte[] c = s1.getBytes();
//默认为UTF-8,占用3个字节
//[-26, -120, -111, -25, -120, -79, 99, 104, 105, 110, 97]
System.out.println(Arrays.toString(c));
byte[] bys = s1.getBytes("GBK");
//指定用GBK进行编码
//[-50, -46, -80, -82, 99, 104, 105, 110, 97]
System.out.println(Arrays.toString(bys));
//对c进行解码
String s = new String(c);
System.out.println(s);
//对bys进行解码
String s2 = new String(bys,"gbk");
System.out.println(s2);
//解码错误出现乱码
String s3 = new String(bys,"utf-8");
System.out.println(s3);
运行图:
本文地址:https://blog.csdn.net/qq_41857955/article/details/107349798
如对本文有疑问, 点击进行留言回复!!
springcloud中feign调用处理mybatis-plus Ipage反序列化问题。
Flume 史上最全面的大数据学习第十篇(一) 别再说不知道flume是什么了
网友评论