`
小野bupt
  • 浏览: 14805 次
  • 性别: Icon_minigender_1
文章分类
社区版块
存档分类
最新评论

Vector和List 有关线程安全和线程同步的理解

 
阅读更多

最近在学习的时候,一直很疑惑Vector和List的两个容器的区别。

在网上查了好多资料,知道Vector的许多方法都是线程同步的,在多线程访问的时候,是线程安全的(这里的线程安全是有限的)。而List在多线程访问的时候,是非线程安全的。

原因是由于Vector在好多方法中加入了sychronized关键字修饰。比如addElement方法中:

public synchronized void addElement(E obj) {
        modCount++;
        ensureCapacityHelper(elementCount + 1);
        elementData[elementCount++] = obj;
    }

这样在多线程并发访问Vector实例的时候,保证了某一刻最多只能有一个线程调用Vector中被sychronized修饰的方法。

而反观List,例如ArrayList中的add方法:

  public boolean add(E e) {
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;
        return true;
    }

有可能同一时刻有一个以上的线程访问该方法。

我们知道size++运算过程不是原子操作,有可能被中断。那么如果在size++的过程中被中断,而另外一个线程恰好执行了一次这个方法,就会造成脏数据的产生。

如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。
或者说:一个类或者程序所提供的接口对于线程来说是原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题。
线程安全问题都是由全局变量及静态变量引起的。
若每个线程中对全局变量、静态变量只有读操作,而无写操作,一般来说,这个全局变量是线程安全的;若有多个线程同时执行写操作,一般都需要考虑线程同步,否则的话就可能影响线程安全。
比如上例中一个 ArrayList 类,在添加一个元素的时候,它可能会有两步来完成:1. 在 Items[Size] 的位置存放此元素;2. 增大 Size 的值。
在单线程运行的情况下,如果 Size = 0,添加一个元素后,此元素在位置 0,而且 Size=1;
而如果是在多线程情况下,比如有两个线程,线程 A 先将元素存放在位置 0。但是此时 CPU 调度线程A暂停,线程 B 得到运行的机会。线程B也向此 ArrayList 添加元素,因为此时 Size 仍然等于 0 (注意哦,我们假设的是添加一个元素是要两个步骤哦,而线程A仅仅完成了步骤1),所以线程B也将元素存放在位置0。然后线程A和线程B都继续运行,都增加 Size 的值。
那好,我们来看看 ArrayList 的情况,元素实际上只有一个,存放在位置 0,而 Size 却等于 2。这就是“线程不安全”了。

也就是说像HashTable和Vector这样的容器实现了线程安全,是通过同步关键字实现的。

此外,我们又注意到,不管是Vector还是List容器,都有可能会出现ConCurrenceModificationException的异常。这是因为几乎所有的集合类都有快速失败的(Fail-Fast)校验机制。这是为了确保集合方法一致而设立的保护措施。他的实现原理就是modCount修改计数器。如在遍历列表的时候(使用迭代器的方式),这时候会保存当前的modCount值,当每次迭代器取值的时候,会通过checkForComodification()方法来校验modCount是否发生了改变。如果发生了改变,就会抛出ConCurrenceModificationException的异常。

public E next() {
            checkForComodification();
            int i = cursor;
            if (i >= size)
                throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i + 1;
            return (E) elementData[lastRet = i];
        }
final void checkForComodification() {
            if (modCount != expectedModCount)
                throw new ConcurrentModificationException();
        }

那么,即使是使用Vector仍可能会抛出这种异常,这是不是和它声称的线程安全所相悖呢?

其实这里的异常和线程同步是两码事。因为使用迭代器遍历容器的时候,这是记录了modCount值,然后调用了remove方法,这在单线程中本来就是不允许的,和多线程同步没有关系。

线程同步是为了实现线程安全,而在Vector中则是实现了线程的部分安全。

线程安全性不是一个非真即假的命题。 Vector 的方法都是同步的,并且 Vector 明确地设计为在多线程环境中工作。但是它的线程安全性是有限制的,即在某些方法之间有状态依赖(类似地,如果在迭代过程中 Vector 被其他线程修改,那么由 Vector.iterator() 返回的 iterator会抛出ConcurrentModifiicationException。

以上是个人理解,如有不对的地方请大家批评指正。

分享到:
评论

相关推荐

    经典讲解List和ArrayList和Vector和HashTable和HashMap区别

    - **线程安全性**:`Vector`和`HashTable`是线程安全的,而`ArrayList`和`HashMap`不是。在多线程环境下,`ArrayList`和`HashMap`需要通过同步机制来保证数据一致性。 - **存储方式**:`List`接口的实现(如`...

    Java集合多线程安全.docx

    1. `Vector`:`ArrayList`的线程安全版本,所有操作都是同步的,但是性能较差,因为每次操作都需要获取和释放锁。 2. `HashTable`:`HashMap`的线程安全版本,同样所有操作都是同步的,效率较低,不适合高并发场景。...

    Java中List、ArrayList、Vector及map、HashTable、HashMap分别的区别.

    Vector与ArrayList类似,但它提供了一种线程安全的实现,这意味着在多线程环境中,多个线程可以同时访问和修改Vector,而不会导致数据不一致。然而,由于其同步机制,Vector的性能通常低于ArrayList。 3. Map接口与...

    比较Vector、ArrayList和hashtable hashmap

    - Vector 和 ArrayList 都实现了 List 接口,其中 Vector 是线程安全的,而 ArrayList 不是。ArrayList 在插入和查找性能上通常优于 Vector,因为 Vector 的同步操作会带来额外的性能开销。 - LinkedList 实现了 ...

    Java中ArrayList和Vector的区别共2页.p

    - 在多线程环境中,如果需要线程安全的动态数组,可以选择Vector,但应考虑其性能影响,也可以考虑使用`Collections.synchronizedList(List<T> list)`将ArrayList转换为线程安全的列表,或者使用`...

    map,list,set,stack,queue,vector等区别和特点1

    在Java编程语言中,集合框架是处理对象组织和操作的...在多线程环境中,线程安全的实现如Vector和Hashtable是必要的,但在单线程或性能要求较高的情况下,非线程安全的实现如ArrayList、LinkedList和HashMap通常更优。

    vector集合

    总的来说,`Vector`集合在Java中扮演了一个历史角色,它提供了一种线程安全的动态数组实现,虽然现代编程更多倾向于使用非同步的`ArrayList`,但理解`Vector`的特性和使用方法仍然对Java开发者具有价值。在设计多...

    java中vector的定义以及用法

    与`ArrayList`不同,`Vector`在多线程环境中能确保数据的一致性和安全性,这主要得益于它的方法大多都是同步的(synchronized)。`Vector`继承自`AbstractList`,实现了`List`、`RandomAccess`、`Cloneable`和`...

    Java容器类List、ArrayList、Vector及map、HashTable应用

    ArrayList和Vector的主要区别在于Vector使用了synchronized方法,线程安全,而ArrayList则没有。 ArrayList是Java中最常用的List实现类,它提供了高效的插入、删除和遍历元素的方法。ArrayList基于数组实现,故插入...

    深入Java多线程和并发编程

    - **同步容器类**:如`Vector`、`HashTable`等,这些容器内部采用了同步锁来保证线程安全性,但可能会导致性能瓶颈。需要注意的是,在多线程环境中连续调用这些容器的方法时仍可能出现线程安全问题。 - **并发容器类...

    Java线程安全基础概念解析

    线程安全的实现方法可以分为两类:互斥同步和非阻塞同步。互斥同步是最常见的一种并发正确性保障手段,同步是指在多个线程并发访问共享数据时,保证共享数据在同一个时刻只被一条线程使用。Java中的互斥同步手段有...

    Vector底层结构和源码分析

    它和ArrayList一样,都是基于数组实现的,但是 Vector 是线程同步的,即线程安全的。在本文中,我们将深入分析 Vector 的底层结构和源码,探索它的实现机理和使用场景。 1. Vector 的基本介绍 Vector 类的定义说明...

    ArrayList LinkedList Vector区别

    ArrayList、LinkedList、Vector 是 Java 中常用的数据结构实现类,它们都实现了 List 接口,但它们在存储方式、性能、线程安全性等方面有着不同特点。 首先,ArrayList 和 Vector 都是采用数组方式存储数据的,这种...

    Vector 类实现

    1. **线程安全**:`Vector`类的所有操作都是线程安全的,因为它在每个操作上都使用了同步锁。这意味着在多线程环境中,多个线程可以安全地读写同一个`Vector`实例,而不会导致数据不一致。这是与`ArrayList`的一个...

    10个Java经典的List面试题!.pdf

    Vector是List接口下的一种线程安全的集合实现,Vector的所有方法都是同步的,而ArrayList和LinkedList不是线程安全的。因此,如果需要在多线程环境下使用List,应该选择Vector或使用Collections.synchronizedList()...

    区别和联系-list-map-set-vector

    - **Vector**: Vector 类似于 ArrayList,但它是线程同步的,这意味着多个线程可以安全地访问同一个 Vector 对象。然而,这种同步也会带来性能上的开销。Vector 同样提供了 `add()`、`remove()` 和 `insert()` 方法...

    第17章 - 深入研究容器 - Collection(List,Set,Queue)的性能测试框架(单线程中)(P501)

    Vector与ArrayList类似,但它是线程安全的,但在现代多线程编程中,通常更倾向于使用非同步的ArrayList配合适当的同步机制。 Set接口不允许重复元素,HashSet、LinkedHashSet和TreeSet是其主要实现。HashSet基于...

    实例4 矢量(Vector)对象的操作.rar

    `Vector`类的所有公共方法都是同步的,这意味着它们在多线程环境中是线程安全的。相比之下,`ArrayList`不是线程安全的,所以在多线程环境下使用`ArrayList`时需要手动添加同步机制。 4. 性能比较: - `Vector`...

Global site tag (gtag.js) - Google Analytics