ArrayList的问题以及解决方案

单线程环境下使用ArrayList
    List<String> list =new ArrayList<>();
        list.add("1");
        list.add("2");
        list.add("3");
        for (String element:list){
            System.out.println(element);
        }

我们经常在项目中会这样使用ArrayList,我们来分析一下,当我们执行底层干了些什么

 List<String> list =new ArrayList<>();

底层创建了一个空的数组,初始化值为10,当我们执行add方法的时候,如果超过10.那么就会扩容,扩容值的大小为原值的一半,也就是5个,使用下列方法进行扩容

 elementData = Arrays.copyOf(elementData, newCapacity);

但是ArrayList在单线程下是线程不安全的

多线程下使用ArrayList

        List<String> list1 =new ArrayList<>();
        for (int i=0;i<50;i++){
            new Thread(()->{
                list1.add(UUID.randomUUID().toString().substring(0,8));
                System.out.println("线程名称= "+Thread.currentThread().getName()+"线程id "+Thread.currentThread().getId()+"数组list ="+list1);
            },String.valueOf(i)).start();
        }
截图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-HJmdgD3z-1603943188867)(https://raw.githubusercontent.com/xiaofeifei321/Picture/master/img20201029111328.png)]

分析:

​ 多线程我们观看源码可以知道,在进行写操作的时候,方法上为了保证并发性,并没有添加synchronise修饰,所以并发写的去操作List的时候,就会出现 java.util.ConcurrentModificationException 异常(这里注意一点我们不输出 System.out.println(list1)程序并不会报错,从这个现象看,并发读写时候才会出现问题,并发写的时候没出现问题

1
解决方法
Vector

我们通过阅读源码知道,ArrayList在添加元素的add方法上没有加synchronized锁,但是Vector类上是添加了同步锁的

    public synchronized boolean add(E e) {
        modCount++;
        ensureCapacityHelper(elementCount + 1);
        elementData[elementCount++] = e;
        return true;
    }

这样每一个线程操作,就不会出现线程不安全的问题,但是因为加锁,不会出现线程安全问题,但是会导致并发度降低

采用Collection集和工具类
 List<String> list4= Collections.synchronizedList(new ArrayList<>());

采用集和工具类,在ArrayList外面包装一层同步机制

采用JUC里面的方法
   CopyOnWriteArrayList copyOnWriteArrayList=new CopyOnWriteArrayList();
        copyOnWriteArrayList.add(1);
        copyOnWriteArrayList.add(2);
        copyOnWriteArrayList.add(3);
        copyOnWriteArrayList.add(4);
        System.out.println(copyOnWriteArrayList);

CopyOnWriteArrayList:写入时候复制,主要是一种读写分离的思想,这个思想锁的是说如果多个调用者同时要求相同的资源,他们会同时获取相同的指针指向相同的资源,直到某一个调用者修改资源内容,系统才会复制一份专门的副本给该调用者,而其他调用者所见的最初的资源仍然保持不变,这个调用过程对其他调用者都是透明的,此做法的主要优点就是如果调用者没有修改资源吗,就不会有副本创建,因此多个调用者只是读取操作时候可以共享同一份资源。(废话)

 /**
     * Appends the specified element to the end of this list.
     *
     * @param e element to be appended to this list
     * @return {@code true} (as specified by {@link Collection#add})
     */
    public boolean add(E e) {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            Object[] elements = getArray();
            int len = elements.length;
            Object[] newElements = Arrays.copyOf(elements, len + 1);
            newElements[len] = e;
            setArray(newElements);
            return true;
        } finally {
            lock.unlock();
        }
    }
    


    final void setArray(Object[] a) {
        array = a;
    }
    

CopyOnWriteArrayList是这样干的,当我们往一个容器中添加元素的时候,不直接往当前容器Object[]添加,而是先将Object[]进行copy,复制出一个新的容器,然后在新的容器中进行添加元素的操作,万事之后,将原来容器的引用指向新的容器,这样做的好处是可以提高并发度,当前的容器并不需要加锁。

说明参考

文章为看视频博客学习过程中总结,方便自己以后好复习

https://www.bilibili.com/video/BV18b411M7xz

http://moguit.cn/#/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ArrayList实现了Serializable接口,这意味着ArrayList类可以将其对象序列化成字节流,以便于将其存储到文件或在网络上传输。 writeObject方法是ObjectOutputStream类的一个方法,它将Java对象写入输出流中以进行序列化。当ArrayList被序列化时,writeObject方法将被调用以将ArrayList对象写入输出流中。这个方法将遍历ArrayList对象并将其元素写入输出流中。由于ArrayList的元素可以是任何Java对象,因此这些对象也需要实现Serializable接口才能被序列化。在反序列化时,readObject方法将被调用以从输入流中读取数据并重新构造ArrayList对象及其元素。 因此,ArrayList重写writeObject方法是为了控制序列化ArrayList对象时的行为,例如序列化特定的属性,以及为序列化ArrayList的元素提供额外的安全检查。 ### 回答2: ArrayList重写writeObject方法的原因是因为在默认情况下,ArrayList的元素会使用序列化对象的方式进行存储,即将每个元素逐个写入输出流中。但是ArrayList本身不是实现了Serializable接口的类,所以在序列化时会抛出NotSerializableException异常。 为了解决这个问题ArrayList重写了writeObject方法,实现了自己的序列化方式。在writeObject方法中,ArrayList会将其内部的元素转换为一个数组,然后将该数组进行序列化存储。这样就避免了直接序列化ArrayList本身的问题。 通过重写writeObject方法,ArrayList能够在序列化和反序列化的过程中实现对元素的正确处理,确保ArrayList对象能够被正确地序列化和反序列化。这样,即使ArrayList本身不是一个可序列化的类,也能够正常地进行序列化和反序列化操作。 ### 回答3: ArrayList类重写writeObject方法的原因主要有两个方面: 首先,ArrayList类是实现了Serializable接口的,意味着它可以被序列化成字节序列,以便在网络上传输或者保存到磁盘中等。而writeObject方法是在序列化过程中被调用的方法,它的作用是将对象的状态转换成字节序列的形式。ArrayList类重写writeObject方法主要是为了自定义ArrayList对象的序列化过程。默认情况下,ArrayList对象会按照元素的顺序依次将元素序列化,但是在某些情况下,我们可能希望以其他方式将ArrayList对象转换成字节序列。通过重写writeObject方法,我们可以自定义ArrayList的序列化方式,例如只序列化部分元素或者特定条件下序列化元素等。 其次,ArrayList类是一个动态数组,它可以根据需要自动扩容和收缩。在默认的序列化过程中,writeObject方法会将ArrayList的容量也一同进行序列化,这就意味着在反序列化时需要还原ArrayList的容量设置。然而,在某些情况下,我们可能不希望容量信息也被序列化,因为容量信息并不代表实际元素的数量,它只是ArrayList内部用于优化性能的一个指示值。通过重写writeObject方法,我们可以忽略容量信息的序列化,从而简化反序列化的过程。 总之,ArrayList重写writeObject方法主要是为了自定义序列化过程,包括对元素的序列化顺序和容量信息的处理。这样可以更灵活地控制ArrayList对象在序列化和反序列化过程中的行为。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值