java 的String, Stringbuffer, StringBuilder的区别以及对final关键字的理解

本文参考了一些文章,链接如下:

http://www.cnblogs.com/xudong-bupt/p/3961159.html

http://www.cnblogs.com/mengdd/archive/2012/12/26/2834711.html

但是我也看了好一会儿才完全弄明白,比如文章中的“对象不可变”有点难理解,高手请忽略。


java中String、StringBuffer、StringBuilder是编程中经常使用的字符串类,他们之间的区别也是经常在面试中会问到的问题。现在总结一下,看看他们的不同与相同。

1.可变与不可变

  String类中使用字符数组保存字符串,如下就是,因为有“final”修饰符,所以可以知道string对象是不可变的。

    private final char value[];

我的理解:String对象不可变:比如 String aa = new String("hello"); 

//新建一个String对象,指向一个地址,地址上的值是"hello";  

aa = "world";

//新取一段地址空间(不是原来的),值是"world",aa指向这个新的地址

aa += "!";

//新取一段地址空间(不是原来的),值是"world!",aa指向这个新的地址

//所以说String值每修改一次,内存里就会新建一个字符串,String对象指向的地址也会发生改变


StringBuilder与StringBuffer都继承自AbstractStringBuilder类,在AbstractStringBuilder中也是使用字符数组保存字符串,如下就是,可知这两种对象都是可变的。

    char[] value;

2.是否多线程安全

  String中的对象是不可变的,也就可以理解为常量,显然线程安全

  AbstractStringBuilder是StringBuilder与StringBuffer的公共父类,定义了一些字符串的基本操作,如expandCapacity、append、insert、indexOf等公共方法。

  StringBuffer对方法加了同步锁或者对调用的方法加了同步锁,所以是线程安全的。看如下源码:

复制代码
1 public synchronized StringBuffer reverse() {
2     super.reverse();
3     return this;
4 }
5 
6 public int indexOf(String str) {
7     return indexOf(str, 0);        //存在 public synchronized int indexOf(String str, int fromIndex) 方法
8 }
复制代码

  StringBuilder并没有对方法进行加同步锁,所以是非线程安全的

 3.StringBuilder与StringBuffer共同点

  StringBuilder与StringBuffer有公共父类AbstractStringBuilder(抽象类)。

  抽象类与接口的其中一个区别是:抽象类中可以定义一些子类的公共方法,子类只需要增加新的功能,不需要重复写已经存在的方法;而接口中只是对方法的申明和常量的定义。

  StringBuilder、StringBuffer的方法都会调用AbstractStringBuilder中的公共方法,如super.append(...)。只是StringBuffer会在方法上加synchronized关键字,进行同步。

  最后,如果程序不是多线程的,那么使用StringBuilder效率高于StringBuffer。



Final

final关键字可以理解为终态。

 

final修饰类

  final修饰的类不允许被继承。

  一个类不能既是final的,又是abstract的。因为abstract的主要目的是定义一种约定,让子类去实现这种约定,而final表示该类不能被继承,两者矛盾。

 

final修饰方法

  final修饰方法,表示该方法不能被子类中的方法覆写Override。

 

final修饰变量

  final成员变量表示常量,只能被赋值一次,赋值后值不再改变。

  当final修饰一个原生数据类型时,表示该原生数据类型的值不能发生变化;

  如果final修饰一个引用类型时,表示该引用类型不能再指向其他对象了,但该引用所指向的对象的内容是可以发生变化的。

  本质上是一回事,因为引用的值是一个地址,final要求值,即地址的值不发生变化。

  final修饰一个成员变量(属性),必须要显示初始化。

  这里有两种初始化方式,一种是在变量声明的时候初始化;第二种方法是在声明变量的时候不赋初值,但是要在这个变量所在的类的所有的构造函数中对这个变量赋初值。

  当函数的参数类型声明为final时,说明该参数是只读型的。


举例:final int a; a = 10; //如果再给a=12;则会报错,final是不可修改的

final String b = new String("hello");  //如果再给 b +="world";则会报错,说明String对象每次修改,地址会改变,而final不允许改变

final StringBuffer c=new StringBuffer("hello");
c.append("world");  //这样写是对的,c的值是"helloworld"


哎呀,看来得好好学习了,java这些基础的都不会,以后可怎么找工作啊,加油~20170531






你可能感兴趣的:(Java)