java线程安全String -可变不可变 StringBuilder StringBuffer

-classpath不支持下划线

proxy-revalidation代理服务器 重新生效

Expires是RFC 2616(HTTP/1.0)协议中和网页缓存相关字段。用来控制缓存的失效日期,要注意的是,HTTP/1.0有一个功能比较弱的缓存控制机制:Pragma,使用HTTP/1.0的缓存将忽略Expires和Cache-Control头。

 

 

可变 vs 不可变

不可变的对象具有非常多的优势,比如简单,安全等。但是,对于每一个不同的值,都需要该类的一个对象。而且,生成很多对象带来的问题就是可能导致频繁的垃圾回收。所以,在选择可变类还是不可变类时,应该综合考虑后再做抉择。

 

通常而言,可变对象可以避免创建大量的中间对象。一个非常经典的例子就是链接大量的短String对象为一个长的String对象。如果使用不可变String类,链接的过程将产生大量的,适合立即被垃圾回收的中间String对象,这将消耗大量的CPU性能和内存空间。此时,使用一个可变的StringBuilder或StringBuffer才是正确的。

java线程安全String -可变不可变 StringBuilder StringBuffer_第1张图片

 

 

除了上述情况,可变对象在其他场景下可能用于不可变对象。比如,传递一个可变的对象到方法内部,利用该对象可以收集多个结果,而不用在多个循环层次中跳进跳出。

在迭代时移除List中的元素

 

首先,看一下在迭代过程中移除List中元素的代码:

java线程安全String -可变不可变 StringBuilder StringBuffer_第2张图片

 

这个示例代码的输出结果是:

[b,d]

 

 

这个示例代码中存在一个非常严重的错误。当一个元素被移除时,该List的大小(size)就会缩减,同时也改变了索引的指向。所以,在迭代的过程中使用索引,将无法从List中正确地删除多个指定的元素。

 

你可能知道解决这个错误的方式之一是使用迭代器(iterator)。而且,你可能认为Java中的foreach语句与迭代器(iterator)是非常相似的,但实际情况并不是这样。我们考虑一下如下的示例代码:

java线程安全String -可变不可变 StringBuilder StringBuffer_第3张图片

 

 

这个示例代码会抛出来一个ConcurrentModificationException。我们应该修改成如下所示:

java线程安全String -可变不可变 StringBuilder StringBuffer_第4张图片

 

next()方法必须在remove()方法之前被调用。在 foreach循环中,编译器使得 remove()方法先于next()方法被调用,这就导致了ConcurrentModificationException 异常。具体细节可以查看ArrayList.iterator()的源码。

在Java中,对应哈希表的的类是HashMap而不是Hashtable。HashMap与Hashtable之间的最核心区别就是:HashMap是非同步的,Hashtable是同步的。

字符串对象的两个构建方式

 

Java中的字符串对象具有两个常见的创建方式:

 

java线程安全String -可变不可变 StringBuilder StringBuffer_第5张图片

它们之间的区别是什么呢?我们再看一下如下的代码:

java线程安全String -可变不可变 StringBuilder StringBuffer_第6张图片

 

 

 

你可能感兴趣的:(java,String)