java 字符串

可以证明,字符串操作时家属及程序设计中最常见的行为

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

一. 不变的字符串

1. 字符串不变; 它们的值在创建后不能被更改。 字符串缓冲区支持可变字符串。 因为String对象是不可变的,它们可以被共享

2. String对象是不可变的,每当把String对象作为方法的参数传递时,都会复制一份引用.该引用所指的对象其实一直待在单一的物理位置上,从未动过.

3. 给String对象赋值本质上是改变该String对象引用的指向.

package strings;
//: strings/Immutable.java
import static net.mindview.util.Print.*;

public class Immutable {
  public static String upcase(String s) {
    return s.toUpperCase();//将所有字母改为大写
  }
  public static void main(String[] args) {
    String q = "howdy";
    print(q); // howdy
    String qq = upcase(q);
    print(qq); // HOWDY
    print(q); // howdy
  }
} /* Output:
howdy
HOWDY
howdy
*///:~

三.重载 " + " 与StringBuilder

 1. String对象是不可变的,你可以给一个String对象加任意多的别名.因为String对象具有只读特性,所以指向它的任何引用都不可能改变它的值,因此,也就不会对其它引用有什么影响

2.不可变性会带来一定的效率问题,为String对象重载的"+"操作符就是一个例子.//此种方式每+一次就会多一个String对象

3. String的"+"操作经过编译器优化后是利用的StringBuilder对字符串进行拼接,性能不如直接使用StringBuilder拼接字符串要好.  

package strings;
//: strings/WhitherStringBuilder.java

public class WhitherStringBuilder {
  public String implicit(String[] fields) {
    String result = "";
    for(int i = 0; i < fields.length; i++)
      result += fields[i];
    return result;
  }
  public String explicit(String[] fields) {
    StringBuilder result = new StringBuilder();
    for(int i = 0; i < fields.length; i++)
      result.append(fields[i]);
    return result.toString();
  }
} ///:~
生成的字节码如下

public class strings.WhitherStringBuilder { 
  public strings.WhitherStringBuilder();
    Code:
       0: aload_0    // 将this引用推送至栈顶,即压入栈                
       1: invokespecial #8 //调用<init>方法实例化对象// Method java/lang/Object."<init>":()V
       4: return

  public java.lang.String implicit(java.lang.String[]);
    Code:
       0: ldc           #16 //将编号为#16的字符串推送至栈顶 // String
       2: astore_2     //将栈顶引用类型值存入第三个本地变量
       3: iconst_0      //将int型0推送至栈顶
       4: istore_3      //将栈顶int型数值存入第四个本地变量
       5: goto          32  //无条件跳转到32行
       8: new           #18  //新建一个StringBuilder对象编号#18,并将其引用压入栈顶// class java/lang/StringBuilder
      11: dup           //复制栈顶数值(在这里时引用)并将复制数值压入栈顶
      12: aload_2      //将第三个本地引用推送至栈顶
      13: invokestatic  #20//调用静态方法(这里调用了valueOf())// Method java/lang/String.valueOf:(Ljava/lang/Object;)Ljava/lang/String;
      16: invokespecial #26//调用超类构造方法,实例初始化方法,私有方法 // Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
      19: aload_1       //将第二个引用本地变量(即数组引用)压入栈顶
      20: iload_3       //将第四个int本地变量压入栈顶
      21: aaload        //将引用数组指定索引的值推送至栈顶(即二个本地引用变量所代表的数组的下标为第四个int本地变量的值)
      22: invokevirtual #29//调用实例化方法append// Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      25: invokevirtual #33 //调用实例化方法toString将结果转换为String// Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      28: astore_2          //将栈顶引用类型值存入第三个本地变量
      29: iinc          3, 1 //将第4个int本地变量加1
      32: iload_3          //将第4个int本地变量推至栈顶
      33: aload_1          //将第二个引用类型本地办理推送至栈顶,第二个引用应该是改方法的参数的数组的引用
      34: arraylength      //获得数组的长度并压入栈顶 
      35: if_icmplt     8  //比较栈顶两int数值的大小如果小于0跳转到第8行
      38: aload_2          //将第二个本地引用类型的本地变量推送至栈顶
      39: areturn          //返回栈顶引用型本地变量,并退出方法

  public java.lang.String explicit(java.lang.String[]);
    Code:
       0: new           #18                 // class java/lang/StringBuilder
       3: dup
       4: invokespecial #45                 // Method java/lang/StringBuilder."<init>":()V
       7: astore_2
       8: iconst_0
       9: istore_3
      10: goto          24
      13: aload_2
      14: aload_1
      15: iload_3
      16: aaload
      17: invokevirtual #29                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      20: pop
      21: iinc          3, 1
      24: iload_3
      25: aload_1
      26: arraylength
      27: if_icmplt     13
      30: aload_2
      31: invokevirtual #33                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      34: areturn
}

 四. 无意识的递归

1.容器类都有toString()方法,它生成的结果可以表达容器类自身,以及容器所包含的对象.

2.如果想打印对象的内存地址,在toString()方法中不能使用this关键字的返回字符串,要返回String对象,this关键字会调用toString()方法,从而会产生递归调用 this.toString()->this.toString()  ...             

要想正确打印地址,必须调用Object.toString()方法

package strings;
//: strings/InfiniteRecursion.java
// Accidental recursion.
// {RunByHand}
import java.util.*;

public class InfiniteRecursion {
  public String toString() {
    //!return this.toString(); //这里会递归调用
return super.toString(); //正确的方法时调用Object的toString()方法
}
public static void main(String[] args) { List<InfiniteRecursion> v = new ArrayList<InfiniteRecursion>(); for(int i = 0; i < 10; i++) v.add(new InfiniteRecursion()); System.out.println(v); } } ///:~

六.String上的操作

方法参数,重载版本应用
构造器 重载版本:默认版本,String,StringBuilder,StringBuffer,char数组,byte数组 创建String对象
length()   String中字符的个数
charAt() Int索引 取得String中该索引位置上的char
getChars(),getBytes() 要复制的部分的起点和终点的索引,复制的目标数组,目标数组的起始索引 复制char或byte到一个目标数组中
toCharArray()   生成一个char[],包含String的所有字符
equals(),equalsIgnoreCase() 与之进行比较的String 比较两个String的内容是否相同
compareTo 与之进行比较的String 按词典顺序比较String的内容,比较结果为负数,零或正数.注意,大小写并不等价
contains    
contentEquals() 与之比较的CharSequence或StringBuffer 如果该String与参数的内容完全一致,则返回true
equalsIgnoreCase 与之进行比较的String 忽略大小写,如果两个String的内容相同,则返回true
regionMatcher() 该String的索引偏移量,另一个String及其索引偏移量,要比较的长度.重载版本增加了"忽略大小写"功能 返回boolean结果,以表明所比较区域是否相等
startsWith() 可能的起始String.重载版本在参数中增加了偏移量 返回boolean结果,以表明String是否以此参数起始
endsWith() 该String可能的后缀String 返回boolean结果,以表明此参数在String中的起始索引.lastIndexOf()是从后向前搜索
indexOf(),lastIndexOf() 重载版本包括:char,char与起始索引,String,String与起始索引 如果该String并不包含此参数,就返回-1;否则返回此参数在String中起始的索引.lastIndexOf()是从后向前搜索
substring(subSequence()) 重载版本:起始索引;起始索引+终点坐标 返回一个新的String,以包含参数指定的子字符串
concat() 要连接的String 返回一个新的String对象,内容为起始Stirng连接上参数String
replace() 要替换掉的字符,用来进行替换的新字符,也可以用一个CharSequence来转换另一个CharSequence 返回替换字符后的新String对象.如果没有替换发生,则返回原始的String对象
toLowerCase,toUpperCase()   将字符的大小写改变后,返回一个新String对象.如果没有发生改变,则返回原始的String对象
trim()   将String两端的空白字符删除后,返回一个新的String对象.如果没有改变发生,则返回原始的String对象
valueOf() 重载版本:Object;char[];char[],偏移量,字符个数; boolean; char; int; long; float; double 返回一个表示参数内容的String
intern()   为每个唯一的字符序列生成一个且仅生成一个String引用
posted @ 2019-01-23 12:04  江期玉  阅读(203)  评论(0编辑  收藏  举报