String字符串是只读的,不可变的

查看String类的源码,可以发现String类是被final关键字修饰的;

另外还可以看下String类源码中的其它方法实现,随便举个可以修改String值的方法,如字符串拼接方法concat(String str),返回的是一个全新的String对象,而不是在原有的String对象上做修改,代码如下:

    public String concat(String str) {
int otherLen = str.length();
if (otherLen == 0) {
return this;
}
int len = value.length;
char buf[] = Arrays.copyOf(value, len + otherLen);
str.getChars(buf, len);
return new String(buf, true);
}

以上代码中,将原有的字符数组和新字符串数组拷贝到一个新的字符数组中,然后new出一个新的String对象返回;

重载“+”与StringBuilder

先看一段使用“+”拼接字符串的代码如下:

package strings;

public class Test {

    public static void main(String[] args) {
String a = "test";
String b = a; //b保存了a的引用 //这里编译器会为我们自动创建StringBuilder对象,并调用append方法,
//最终调用StringBuilder的toString方法返回一个新的字符串对象
a = a + "1" + "2" + "3"; //生成了一个新的String对象赋给a System.out.println(b==a); //引用a指向了新的字符串对象,不相等
System.out.println(a);
}
}

在使用“+”拼接字符串的时候,编译器会为我们自动创建StringBuilder对象,并调用append方法拼接字符串,最终调用StringBuilder的toString方法返回一个新的字符串对象;

或许你认为既然编译器会为我们创建自动StringBuilder对象就可以任意使用“+”操作符了,实际上编译器在某些情况下为我们优化的程度还是不够,如下代码例子,在循环体内使用“+”操作符;

package strings;

public class Test {

    public static void main(String[] args) {
String a = "";
for(int i = 0; i < 5; i++){
//编译器每次都会生成一个StringBuilder对象,并调用toString方法生成一个新的String对象
//这中间产生了两个临时对象
a +=i;
}
System.out.println(a);
}
}

可以使用eclipse调试进入源码,会发现,编译器每次都会生成一个StringBuilder对象,并调用toString方法生成一个新的String对象,也就是说。每循环体执行一次,就产生了两个临时对象,可见,当循环次数大的时候,会产生一大堆需要垃圾回收的中间对象,而直接使用StringBuilder就没有这种情况,如下代码,为一个正确的例子:

package strings;

public class Test {

    public static void main(String[] args) {
StringBuilder a = new StringBuilder("");
for(int i = 0; i < 5; i++){
a.append(i);
}
System.out.println(a);
}
}

总之,只要涉及字符串的操作,选择StringBuilder总是没错的;

无意识的递归调用

在重写toString方法的时候,如果不注意使用了this关键字,很有可能会调入递归调用的陷阱,如下代码:

package strings;

public class Test {

    @Override
public String toString() { return "addr:" + this;
} public static void main(String[] args) {
Test t = new Test();
System.out.println(t);
}
}

执行的时候,将抛出StackOverflowError异常,因为toString方法里的"addr:" + this语句会调用自身toString方法,导致无穷无尽的递归调用,然后堆栈溢出,抛出异常;

如果你仅仅是想打印下对象的地址,那么可以调用super.toString()方法,因为Object对象的toString方法默认会调用hashCode打印对象地址;

Sting类的方法

关于String类中的方法,可以查看String源码或JDK的API文档,相对来说还是很好理解的,所谓String对象,实质上来说就是一个字符数组;

String类中的方法,大部分都是操作String内部维护的char value[]数组实现的;

主要方法如下,图片参考自java编程思想-4:

字符串格式化输出

关于格式化输出,了解以下几个例子吧:

format方法:可用于PrintString和PritWriter, 如System.out.format, 如果有C语言的printf语法的使用经验的话,学习format语法会非常轻松,基本类似,如下为一个简单例子:

System.out.format("%5d: %2f", 101,1.131452);

Formatter类:printf 风格的格式字符串的解释程序,如下例子:

package strings;

import java.util.Formatter;

public class Test {

    public static void main(String[] args) {
Formatter formatter = new Formatter(System.out);
formatter.format("%5d: %2f", 101, 1.131452);
formatter.close();
}
}

String.format方法:String类的静态方法,在其内部实际上也是通过创建Formatter对象实现的;

System.out.println(String.format("%5d: %2f", 101, 1.131452));

正则表达式

关于正则表达式的具体语法就不细说了,在String类中,涉及正则表达式的主要是以下方法:

matches:判断是否匹配指定的正则表达式规则

split:分割字符串

replaceAll/replaceFirst:将匹配的字符替换为指定的字符串

如下代码为一个简单的使用例子:

package strings;

public class RegexTest {
public static void main(String[] args) { //匹配数字
System.out.println("1314".matches("\\d+")); //按数字分割
String[] splitArr = "asdashh45hiu9jkjaks54d".split("-?\\d+");
for(String str:splitArr){
System.out.print(str + ", ");
} //替换数字为*号
System.out.println();
System.out.println("asdashh45hiu9jkjaks54d".replaceAll("-?\\d+", "*")); }
}
//输出
//true
//asdashh, hiu, jkjaks, d,
//asdashh*hiu*jkjaks*d

JAVA中的字符串小结的更多相关文章

  1. Java中的字符串操作(比较String,StringBuiler和StringBuffer)

    一.前言 刚开始学习Java时,作为只会C语言的小白,就为其中的字符串操作而感到震撼.相比之下,C语言在字节数组中保存一个结尾的\0去表示字符串,想实现字符串拼接,还需要调用strcpy库函数或者自己 ...

  2. [原创]Java中的字符串比较,按照使用习惯进行比较

    java中的字符串比较一般可以采用compareTo函数,如果a.compareTo(b)返回的是小于0的数,那么说明a的unicode编码值小于b的unicode编码值. 但是很多情况下,我们开发一 ...

  3. 理解Java中的字符串类型

    1.Java内置对字符串的支持: 所谓的内置支持,即不用像C语言通过char指针实现字符串类型,并且Java的字符串编码是符合Unicode编码标准,这也意味着不用像C++那样通过使用string和w ...

  4. Java中的字符串驻留

    转自:http://www.cdtarena.com/javapx/201307/9088.html 最近在工作的时候,一句再正常不过的代码String a = “hello” + “world”;被 ...

  5. Java中的字符串拼接

    Java中的字符串拼接 1.设计源码 /** * @Title:IndexOf.java * @Package:com.you.freemarker.model * @Description: * @ ...

  6. java中,字符串类型的时间数据怎样转换成date类型。

    将字符串类型的时间转换成date类型可以使用SimpleDateFormat来转换,具体方法如下:1.定义一个字符串类型的时间:2.创建一个SimpleDateFormat对象并设置格式:3.最后使用 ...

  7. Android学习笔记----Java中的字符串比较

    用习惯了C#.C++,在做字符串比较时想当然地使用如下语句: string str1 = "abcd", str2 = "abcd"; if(str1==str ...

  8. 转载:Java中的字符串常量池详细介绍

    引用自:http://blog.csdn.net/langhong8/article/details/50938041 这篇文章主要介绍了Java中的字符串常量池详细介绍,JVM为了减少字符串对象的重 ...

  9. 《Java入门第三季》第二章 认识 Java 中的字符串

    什么是 Java 中的字符串.1.在Java的世界里,字符串被作为String类型的对象处理. 2.通用的初始化的方式:String s = new String("i love you & ...

随机推荐

  1. PHP类和对象等代码说明

    1.定义和创建类和对象: 定义类要使用class关键字.例如:class 类名{//属性和方法} 创建对象使用new关键字.例如: $p1 = new 类名;,可以基于一个类创建多个对象. 2. 类属 ...

  2. C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题

    (补充:初始化FileStream时使用包含文件共享属性(System.IO.FileShare)的构造函数比使用自定义线程锁更为安全和高效,更多内容可点击参阅) 在开发程序的过程中,难免少不了写入错 ...

  3. Jquery请求Ajax的json数据

      获得单个json对象 $.getJSON("test.js", { name: "John", time: "2pm" }, funct ...

  4. 在Win Server 2012中安装.NET Framework 3.5的问题

    在Windows Server 2012 上安装 SQL Server 2012 时,提示 启用 Windows 功能 NetFx3 时出错,错误代码:-2146498298.请尝试从 Windows ...

  5. Scalaz(15)- Monad:依赖注入-Reader besides Cake

    我们可以用Monad Reader来实现依赖注入(dependency injection DI or IOC)功能.Scala界中比较常用的不附加任何Framework的依赖注入方式可以说是Cake ...

  6. [python拾遗]文件操作

    文件操作 1.open()函数 open()函数主要用于文件处理,一般分为下面3个过程: 1.打开文件 2.操作文件 3.关闭文件 常见的格式示例: f = open('note.txt','r') ...

  7. mysql易混淆知识点

    1,join 和 union join连接属于表之间的水平操作,而union 是表之间的垂直操作.简单讲就是水平操作主要是为了获得列数据,垂直操作是为了获得行数据 cross  join        ...

  8. Javascript单元测试Unit Testing之QUnit

    body{ font: 16px/1.5em 微软雅黑,arial,verdana,helvetica,sans-serif; }           QUnit是一个基于JQuery的单元测试Uni ...

  9. js的动态加载、缓存、更新以及复用(一)

    使用范围: OA.MIS.ERP等信息管理类的项目,暂时不考虑网站. 遇到的问题: 完成一个项目,往往需要引用很多js文件,比如jQuery.js.easyUI等.还有自己写的一些列js文件,那么这些 ...

  10. 【JavaScript】内置对象Math

    Math是具有用于数学常数和函数的属性和方法一内置对象.不是函数对象. 描述编辑 不像其他的全局对象,Math不是一个构造函数.所有属性和方法Math都是静态的.你指的是常数pi为Math.PI你调用 ...