java中的String类(转)

本文详细介绍Java中字符串的各种操作方法,包括格式化、获取字符、判断、转换、替换、切割等功能,以及String类的使用技巧和内存分配机制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载:http://blog.youkuaiyun.com/koches/article/details/7608139


JAVA字符串格式化-String.format()的使用

String类的format()方法用于创建格式化的字符串以及连接多个字符串对象。熟悉C语言的同学应该记得C语言的sprintf()方法,两者有类似之处。format()方法有两种重载形式。

format(String format, Object... args) 新字符串使用本地语言环境,制定字符串格式和参数生成格式化的新字符串。

format(Locale locale, String format, Object... args) 使用指定的语言环境,制定字符串格式和参数生成格式化的字符串。

显示不同转换符实现不同数据类型到字符串的转换,如图所示。

转 换 符

说 明

示 例

%s

字符串类型

"mingrisoft"

%c

字符类型

'm'

%b

布尔类型

true

%d

整数类型(十进制)

99

%x

整数类型(十六进制)

FF

%o

整数类型(八进制)

77

%f

浮点类型

99.99

%a

十六进制浮点类型

FF.35AE

%e

指数类型

9.38e+5

%g

通用浮点类型(f和e类型中较短的)

%h

散列码

%%

百分比类型

%n

换行符

%tx

日期与时间类型(x代表不同的日期与时间转换符

测试用例

复制代码
    public static void main(String[] args) {
        String str=null;
        str=String.format("Hi,%s", "王力");
        System.out.println(str);
        str=String.format("Hi,%s:%s.%s", "王南","王力","王张");          
        System.out.println(str);                         
        System.out.printf("字母a的大写是:%c %n", 'A');
        System.out.printf("3>7的结果是:%b %n", 3>7);
        System.out.printf("100的一半是:%d %n", 100/2);
        System.out.printf("100的16进制数是:%x %n", 100);
        System.out.printf("100的8进制数是:%o %n", 100);
        System.out.printf("50元的书打8.5折扣是:%f 元%n", 50*0.85);
        System.out.printf("上面价格的16进制数是:%a %n", 50*0.85);
        System.out.printf("上面价格的指数表示:%e %n", 50*0.85);
        System.out.printf("上面价格的指数和浮点数结果的长度较短的是:%g %n", 50*0.85);
        System.out.printf("上面的折扣是%d%% %n", 85);
        System.out.printf("字母A的散列码是:%h %n", 'A');
    }
复制代码

以下是string的七种用法,注意哦,记得要时常去查看java的API文档,那个里面也有很详细的介绍

1>获取
1.1:字符串中包含的字符数,也就是字符串的长度。
int length():获取长度
1.2:根据位置获取位置上某个字符。
char charAt(int index)
1.3:根据字符获取该字符在字符串中的位置。
int indexOf(int ch):返回的是ch在字符串中第一次出现的位置。
int indexOf(int ch,int fromIndex):从fromIndex指定位置开始,获取ch在字符串中出现的位置。

int indexOf(String str):返回的是str在字符串中第一次出现的位置。
int indexOf(String str,int fromIndex):从fromIndex指定位置开始,获取str在字符串中出现的位置。
1.4:int lastIndexOf(String str):反向索引。


2>判断
2.1:字符串中是否包含某一个子串。
boolean contains(str);
特殊之处:indexOf(str):可以索引str第一次出现为止,如果返回-1,表示该str不在字符串中存在。
所以,也可以用于对指定判断是否包含。
if(str.indexOf("a")!=1)

而且该方法既可以判断,也可以获取出现的位置。

2.2:字符串中是否有内容。
boolean isEmpty():原理就是判断长度是否为0。
2.3:字符串是否以指定内容开头。
boolean startsWith(str);
2.4:字符串是否以指定内容结尾。
boolean endsWith(str);
2.5:判断字符内容是否相同,复写了object类中的equals方法。
boolean equals(str);
2.6:判断内容是否相同,并忽略大小写。
boolean.equalsIgnorecase();

3>转换
3.1:将字符数组转成字符串。
构造函数:String(char[])
String(char[],offset,count):将字符数组中的一部分转成字符串
静态方法:
static String copyValueOf(char[]);
static String copyValueOf(char[] data,int offset,int count);

static String valueOf(char[]);


整型和字符串相互转换的方法

1)将字符串转化为整型

int i = Integer.parseIn(String str);

int i = Integer.valueOf().intValue();

注:Integer.parseIn 和 Integer.valueOf 不同,前者生成的是整型,而后者是一个对象,所以要通过intValue()来获得对象的值;

字串转成 Double, Float, Long 的方法大同小异.


2)整型转化为字符串

String str = String.valueOf(int i);

String str = Integer.toString(int i);

String str = “” + i ;

注: Double, Float, Long 的方法大同小异.


3.2:将字符串转成字符组
char[] tocharArray();

3.3:将字节数组转成字符串。
String(byte[])
String(byte[],offset,count):将字节数组中的一部分转成字符串
3.4:将字符串转成字节数组。
byte[] getBytes()

3.5:将基本数据类型转成字符串,
static String valueOf(int)
static String valueOf(double)

// 3+"" 与 String.valueOf(3)的值是一样的
特殊:字符串和字节数组在转换过程中,是可以指定编码的。

4>替换
String replace(oldchar,newchar);

5>切割
String[] split(regex);

6>子串。获取字符串中的而一部分
String subString(begin);
String subString(begin,end);

7>转换,去除空格,比较。
7.1:将字符串转成大写或小写
String toUpperCsae() 大转小
String toLowerCsae() 小转大

7.2:将字符串两端的多个空格去除
String trim();

7.3:对两个字符串进行自然顺序的比较
int compareTo(string);

请看如下代码,下面的代码都是针对上面string七种用法而进行一一举例说明:

[java] view plain copy
  1. classStringMethodDemo
  2. {
  3. publicstaticvoidmethod_Zhuanhuan_Qukong_Bijiao()
  4. {
  5. Strings="helloJava";
  6. //打印结果是:(hello和java前后门都有空格)hellojava
  7. sop(s.toUpperCase());
  8. //打印结果是:(HELLO和JAVA前后门都有空格)HELLOJAVA
  9. sop(s.toLowerCase());
  10. //打印及结果是:不带空格的“hellojava”
  11. sop(s.trim());
  12. //比较数的大写,打印结果是:1,因为b对应ascii值是98,
  13. //a对应是97,所以b-a=1
  14. Strings1="abc";
  15. Strings2="aaa";
  16. sop(s1.compareTo(s2));
  17. }
  18. publicstaticvoidmethod_sub()
  19. {
  20. Strings="abcdef";
  21. //打印结果是:cdef,从指定位置开始到结尾。如果角标不存在,会出现字符串角标越界。
  22. sop(s.substring(2));
  23. //打印结果是:cd,包含头,不包含尾。
  24. sop(s.substring(2,4));
  25. }
  26. publicstaticvoidmethod_split()
  27. {
  28. Strings="zhangsan,lisi,wangwu";
  29. String[]arr=s.split(",");
  30. for(intx=0;x<arr.length;x++)
  31. {
  32. sop(arr[x]);
  33. }
  34. }
  35. publicstaticvoidmethod_replace()
  36. {
  37. Strings="hellojava";
  38. //Strings1=s.replace('a','n');
  39. //Strings1=s.replace('w','n');如果要替换的字符不存在,返回的还是原串
  40. Strings1=s.replace("java","world");//打印结果是:helloworld
  41. sop("s="+s);//打印结果是:hellojava因为字符串一旦被初始化,值就不可被改变
  42. sop("s1="+s1);//打印结果是:hellojnvn
  43. }
  44. publicstaticvoidmethod_trans()
  45. {
  46. char[]arr={'a','b','c','d','e','f'};
  47. Strings=newString(arr,1,3);
  48. sop("s="+s);//打印结果是:bcd
  49. Strings1="zxcvbnm";
  50. char[]chs=s1.toCharArray();
  51. for(intx=0;x<chs.length;x++)
  52. {
  53. sop("ch="+chs[x]);//打印结果是:ch=z,x,c,v,b,n,m
  54. }
  55. }
  56. publicstaticvoidmethod_is()
  57. {
  58. Stringstr="ArrayDemo.java";
  59. //判断文件名称是否是Array单词开头
  60. sop(str.startsWith("Array"));
  61. //判断文件名称是否是.java的文件
  62. sop(str.endsWith(".java"));
  63. //判断文件中是否包含Demo
  64. sop(str.contains("Demo"));
  65. }
  66. publicstaticvoidmethod_get()
  67. {
  68. Stringstr="abcdeakpf";
  69. //长度
  70. sop(str.length());
  71. //根据索引获取字符
  72. sop(str.charAt(4));
  73. //sop(str.charAt(40));当访问到字符串中不存在的角标时会发生StringIndexOutOfBoundsException(字符串角标越界异常)
  74. //根据字符获取索引
  75. //sop(str.indexOf('a'));
  76. sop(str.indexOf('a',3));//打印的是5,因为角标3是d,
  77. //所以从d后面开始找a,第5个角标是a
  78. //sop(str.indexOf('t',3))打印:-1,如果没有找到角标,返回-1
  79. //反向索引一个字符出现的位置(从右往左查找,但是角标还是从左开始)
  80. sop(str.lastIndexOf("a"));
  81. }
  82. publicstaticvoidmain(String[]args)
  83. {
  84. method_Zhuanhuan_Qukong_Bijiao();
  85. //method_sub();
  86. //method_split();
  87. //method_replace();
  88. //method_trans();
  89. //method_is();
  90. //method_get();
  91. /*
  92. Strings1="abc";
  93. Strings2=newString("abc");
  94. Strings3="abc";
  95. System.out.println(s1==s2);
  96. System.out.println(s1==s3);
  97. */
  98. }
  99. publicstaticvoidsop(Objectobj)
  100. {
  101. System.out.println(obj);
  102. }
  103. }

最近在论坛上看到关于String s = new String("XYZ") + new String("XYZ");到底创建几个对象的讨论,觉得比较有意思,在此总结一下。

在JAVA中除了8种基本类型之外,其他的都是类对象及其引用。所以 "XYZ"在JAVA中是一个String对象,对于String类对象来说它的对象值是不能修改的,也就是具有不变性。

但是在下面一段程序中:

  1. publicclassTestString{
  2. publicstaticvoidmain(Stringargs[]){
  3. Strings="Hello";
  4. s="Java";
  5. Strings1="Java";
  6. Strings2=newString("Java");
  7. System.out.println(s);
  8. System.out.println(s==s1);
  9. System.out.println(s==s2);
  10. }
  11. }
  12. /*output:
  13. *Java
  14. *true
  15. *false
  16. */

打印出s的结果是"Java",看起来s所引用的String变量好像是被修改了,但是如果你了解JVM(Java虚拟机)处理String变量时的机制,你就会知道事实并非如此。

在JVM的工作过程中,会创建一片的内存空间专门存入String对象,我们把这片内存空间叫做String池。

对于语句String s= "Hello";,当JVM看到"Hello"时,会在String池创建String对象存储它,并将它的引用返回给String变量s。

语句s = "Java";,当JVM看到"Java"时,会在String池创建新的String对象存储它,再把新建的String对象的引用返回给String变量s。而原先的String对象"Hello"仍然在String池内,并没有消失,它是不能被修改的。

所以我们仅仅是改变了变量s的引用,而没有改变它所引用的对象,因为String对象的值是不能被修改的。

String s1 = "Java";,JVM首先在String池里面看能否找到字符串"Java",如果找到则返回它的引用给s1,否则创建新的String对象,放到String池里。这里由于有s = "Java",所以对象已经被引用,所以依据规则s和s1都是引用同一个对象。所以s==s1返回true。(注: 比较运算符==,对于非基本类型,是比较两引用是否引用内存中的同一个对象)。

String s2 = new String( "Java");,JVM首先还是在String池里面看能否找到字符串 "Java",如果找到,不做任何事情;否则创建新的String对象,放到String池里面。由于遇到了new关键字,还会在内存上(不是String池里面)创建String对象存储 "Java",并将内存上的(不是String池里面的)String对象返回给s2。所以s==s2将返回false,因为它们引用的不是同一个对象。

所以对于语句String s = new String("XYZ") + new String("XYZ");

JVM先在String池中创建一个String对象存储"XYZ",然后由于遇到new关键字,再在内存上创建一个String对象存储"XYZ";

接着由于String池中已经有了"XYZ"的对象,所以第二个new语句不会在String池中创建对象,而只会在内存上创建一个String对象;

最后两个字符串相加会在String池中创建一个String对象"XYZXYZ",并将其引用传给s。

所以总共会创建4个String对象。





java堆与栈 java String分配内存空间(详解)

栈内存

堆内存

基础类型,对象引用(堆内存地址)

由new创建的对象和数组,

存取速度快

相对于栈内存较慢

数据大小声明周期必须确定

分配的内存由java虚拟机自动垃圾回收器管理。动态分配内存大小

共享特性

栈中如果有字符串,则直接引用

如果没有,开辟新的空间存入值

每new一次在堆内存中生成一个新的对象。

创建之后值可以改变

String类声明后则不可改变

一、栈内存

基础类型int, short, long, byte, float, double, boolean, char和对象引用

栈的共享特性

String str1 = "abc";
String str2 = "abc";
System.out.println(str1==str2); //true

1、编译器先处理String str1 = "abc";它会在栈中创建一个变量为str1的引用,然后查找栈中是否有abc这个值,如果没找到,就将abc存放进来,然后将str1指向abc。

2、接着处理String str2 = "abc";在创建完b的引用变量后,因为在栈中已经有abc这个值,便将str2直接指向abc。这样,就出现了str1与str2同时均指向abc的情况。

二、堆内存

new、newarray、anewarray和multianewarray等指令建立

要注意:我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的 对象。只有通过new()方法才能保证每次都创建一个新的对象。 由于String类的immutable性质,当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率。

三、 ==内存地址比对

String str1 = "abc";
String str2 = "abc";
System.out.println(str1==str2); //true str1和str2同时指向 栈内存 中同一个内存空间

String str3 = "abc";
String str4 = new String("abc") ;

System.out.println(str3 == str4); //flase str3值在栈内存中,str4值在堆内存中

String hello = "hello" ;

String hel = "hel" ;

String lo = "lo" ;

System.out.println(hello == "hel" + "lo") ; //true

//两个常量相加,先检测栈内存中是否有hello如有有,指向已有的栈中的hello空间

System.out.println(hello == "hel" + lo) ; //flase

System.out.println(hello == hel + lo) ; //flase

//lo是在常量池中,不检查栈内存,在堆中产生一个新的hello


ava堆与栈 java String分配内存空间(详解)

栈内存

堆内存

基础类型,对象引用(堆内存地址

由new创建的对象和数组,

存取速度快

相对于栈内存较慢

数据大小声明周期必须确定

分配的内存由java虚拟机自动垃圾回收器管理。动态分配内存大小

共享特性

栈中如果有字符串,则直接引用

如果没有,开辟新的空间存入值

每new一次在堆内存中生成一个新的对象。

创建之后值可以改变

String类声明后则不可改变

一、栈内存

基础类型int, short, long, byte, float, double, boolean, char和对象引用

栈的共享特性

String str1 = "abc";
String str2 = "abc";
System.out.println(str1==str2); //true

1、编译器先处理String str1 = "abc";它会在栈中创建一个变量为str1的引用,然后查找栈中是否有abc这个值,如果没找到,就将abc存放进来,然后将str1指向abc。

2、接着处理String str2 = "abc";在创建完b的引用变量后,因为在栈中已经有abc这个值,便将str2直接指向abc。这样,就出现了str1与str2同时均指向abc的情况。

二、堆内存

new、newarray、anewarray和multianewarray等指令建立

要注意:我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的 对象。只有通过new()方法才能保证每次都创建一个新的对象。 由于String类的immutable性质,当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率。

三、 ==内存地址比对

String str1 = "abc";
String str2 = "abc";
System.out.println(str1==str2); //true str1
和str2同时指向 栈内存 中同一个内存空间

String str3 = "abc";
String str4 = new String("abc") ;

System.out.println(str3 == str4); //flase str3值在栈内存中,str4值在堆内存中

String hello = "hello" ;

String hel = "hel" ;

String lo = "lo" ;

System.out.println(hello == "hel" + "lo") ; //true

//两个常量相加,先检测栈内存中是否有hello如有有,指向已有的栈中的hello空间

System.out.println(hello == "hel" + lo) ; //flase

System.out.println(hello == hel + lo) ; //flase

//lo是在常量池中,不检查栈内存,在堆中产生一个新的hello

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值