JAVA中写时复制(Copy-On-Write)Map实现

本文深入探讨了写时复制(Copy-On-Write)技术在Java并发编程中的应用,特别是其在容器类如CopyOnWriteArrayList和CopyOnWriteHashMap中的实现。文章详细解释了写时复制如何通过创建副本进行修改来实现读写分离,从而提高并发性,同时也讨论了这种技术的内存开销和数据一致性问题。

 

1,什么是写时复制(Copy-On-Write)容器?

写时复制是指:在并发访问的情景下,当需要修改JAVA中Containers的元素时,不直接修改该容器,而是先复制一份副本,在副本上进行修改。修改完成之后,将指向原来容器的引用指向新的容器(副本容器)。

 

2,写时复制带来的影响

①由于不会修改原始容器,只修改副本容器。因此,可以对原始容器进行并发地读。其次,实现了读操作与写操作的分离,读操作发生在原始容器上,写操作发生在副本容器上。

②数据一致性问题:读操作的线程可能不会立即读取到新修改的数据,因为修改操作发生在副本上。但最终修改操作会完成并更新容器,因此这是最终一致性。

 

3,在JDK中提供了CopyOnWriteArrayList类和CopyOnWriteArraySet类,但是并没有提供CopyOnWriteMap的实现。因此,可以参考CopyOnWriteArrayList自己实现一个CopyOnWriteHashMap

这里主要是实现 在写操作时,如何保证线程安全。

复制代码

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;


public class CopyOnWriteMap<K, V> implements Map<K, V>, Cloneable{

    private volatile Map<K, V> internalMap;
    
    public CopyOnWriteMap() {
        internalMap = new HashMap<K, V>(100);//初始大小应根据实际应用来指定
    }
    
    @Override
    public V put(K key, V value) {
        synchronized (this) {
            Map<K, V> newMap = new HashMap<K, V>(internalMap);//复制出一个新HashMap
            V val = newMap.put(key, value);//在新HashMap中执行写操作
            internalMap = newMap;//将原来的Map引用指向新Map
            return val;
        }
    }
    
    @Override
    public void putAll(Map<? extends K, ? extends V> m) {
        synchronized (this) {
            Map<K, V> newMap = new HashMap<K, V>(internalMap);
            newMap.putAll(m);
            internalMap = newMap;
        }
        
    }
    
    @Override
    public V get(Object key) {
        V result = internalMap.get(key);
        return result;
    }
    ......//other methods inherit from interface Map
}

复制代码

从上可以看出,对于put() 和 putAll() 而言,需要加锁。而读操作则不需要,如get(Object key)。这样,当一个线程需要put一个新元素时,它先锁住当前CopyOnWriteMap对象,并复制一个新HashMap,而其他的读线程因为不需要加锁,则可继续访问原来的HashMap。

 

4,应用场景

CopyOnWrite容器适用于读多写少的场景。因为写操作时,需要复制一个容器,造成内存开销很大,也需要根据实际应用把握初始容器的大小。

不适合于数据的强一致性场合。若要求数据修改之后立即能被读到,则不能用写时复制技术。因为它是最终一致性。

总结:写时复制技术是一种很好的提高并发性的手段。

 

5,为什么会出现COW?

集合类(ArrayList、HashMap)上的常用操作是:向集合中添加元素、删除元素、遍历集合中的元素然后进行某种操作。当多个线程并发地对一个集合对象执行这些操作时就会引发ConcurrentModificationException,比如线程A在for-each中遍历ArrayList,而线程B同时又在删除ArrayList中的元素,就可能会抛出ConcurrentModificationException,可以在线程A遍历ArrayList时加锁,但由于遍历操作是一种常见的操作,加锁之后会影响程序的性能,因此for-each遍历选择了不对ArrayList加锁而是当有多个线程修改ArrayList时抛出ConcurrentModificationException,因此,这是一种设计上的权衡。

为了应对多线程并发修改这种情况,一种策略就是本文的主题“写时复制”机制;另一种策略是:线程安全的容器类:

ArrayList--->CopyOnWriteArrayList

HashMap--->ConcurrentHashMap

而ConcurrentHashMap并不是从“复制”这个角度来应对多线程并发修改,而是引入了分段锁(JDK7);CAS、锁(JDK11)解决多线程并发修改的问题。

Java中,实现文件的读操作可以通过多种方式完成,具体取决于需求的复杂性和性能要求。以下是几种常见的实现方法: ### 使用字节流进行文件读 最基础的文件读方式是使用`FileInputStream`和`FileOutputStream`类。这种方式适用于处理二进制数据,例如图片或视频文件。以下是一个简单的示例: ```java void uploadFile(File file) { String path = "C:\\Users\\uploadfile\\" + file.getName(); try (FileInputStream fis = new FileInputStream(file); FileOutputStream fos = new FileOutputStream(path)) { byte[] buffer = new byte[1024]; int len; while ((len = fis.read(buffer)) != -1) { fos.write(buffer, 0, len); } } catch (IOException e) { e.printStackTrace(); } } ``` ### 使用字符流进行文本文件读 对于文本文件的读操作,可以使用`FileReader`和`FileWriter`类。这种方式更适合处理文本数据,因为它能够处理字符编码问题。以下是示例代码: ```java void downloadFile(File file) { try (FileInputStream fis = new FileInputStream(file)) { byte[] buffer = new byte[1024]; int len; while ((len = fis.read(buffer)) != -1) { String content = new String(buffer, 0, len); System.out.print(content); } } catch (IOException e) { e.printStackTrace(); } } ``` ### 使用缓冲流提高效率 为了提高读效率,可以使用缓冲流`BufferedReader`和`BufferedWriter`。这些类提供了缓冲功能,减少了对磁盘的访问次数,从而提高了性能。以下是示例代码: ```java FileReader fileReader = new FileReader("G:/b/b.text"); BufferedReader bufferedReader = new BufferedReader(fileReader); String s2; while ((s2 = bufferedReader.readLine()) != null) { System.out.println(s2); } bufferedReader.close(); FileWriter fileWriter = new FileWriter("G:/b/b.text"); BufferedWriter bufferedWriter = new BufferedWriter(fileWriter); bufferedWriter.newLine(); bufferedWriter.write(123); bufferedWriter.flush(); bufferedWriter.close(); ``` ### 使用NIO进行高效文件操作 Java NIO(New IO)包提供了更高效的文件读方式,特别是对于大文件处理。`Files`类中的方法可以简化文件操作。以下是示例代码: ```java import java.nio.file.*; public class NIOExample { public static void main(String[] args) { Path source = Paths.get("source.txt"); Path target = Paths.get("target.txt"); try { Files.copy(source, target, StandardCopyOption.REPLACE_EXISTING); } catch (IOException e) { e.printStackTrace(); } } } ``` ### 使用内存映射文件 对于非常大的文件,可以使用内存映射文件技术,这允许将文件的一部分直接映射到内存中,从而实现快速访问。以下是示例代码: ```java import java.io.*; import java.nio.*; import java.nio.channels.*; public class MemoryMappedFile { public static void main(String[] args) throws IOException { RandomAccessFile file = new RandomAccessFile("largefile.dat", "rw"); FileChannel channel = file.getChannel(); MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, 1024 * 1024); // Perform read/write operations on buffer channel.close(); file.close(); } } ``` ### 使用并行流和异步IO 对于需要高并发处理的场景,可以考虑使用并行流和异步IO。这些高级特性可以进一步提升应用程序的性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值