同步容器真的是安全的吗?

有人认为Vector中的方法都进行了同步处理,那么一定就是线程安全的,事实上这可不一定。看下面这段代码:

public class Test {
    static Vector<Integer> vector = new Vector<Integer>();
    public static void main(String[] args) throws InterruptedException {
        while(true) {
            for(int i=0;i<10;i++)
                vector.add(i);
            Thread thread1 = new Thread(){
                public void run() {
                    for(int i=0;i<vector.size();i++)
                        vector.remove(i);
                };
            };
            Thread thread2 = new Thread(){
                public void run() {
                    for(int i=0;i<vector.size();i++)
                        vector.get(i);
                };
            };
            thread1.start();
            thread2.start();
            while(Thread.activeCount()>10)   {
                 
            }
        }
    }
}
在我机器上运行的结果:

正如大家所看到的,这段代码报错了:数组下标越界。

  也许有朋友会问:Vector是线程安全的,为什么还会报这个错?很简单,对于Vector,虽然能保证每一个时刻只能有一个线程访问它,但是不排除这种可能:

  当某个线程在某个时刻执行这句时:

for(int i=0;i<vector.size();i++)
    vector.get(i);

假若此时vector的size方法返回的是10,i的值为9

  然后另外一个线程执行了这句:

for(int i=0;i<vector.size();i++)
    vector.remove(i);

将下标为9的元素删除了。

  那么通过get方法访问下标为9的元素肯定就会出问题了。

  因此为了保证线程安全,必须在方法调用端做额外的同步措施,如下面所示:

public class Test {
    static Vector<Integer> vector = new Vector<Integer>();
    public static void main(String[] args) throws InterruptedException {
        while(true) {
            for(int i=0;i<10;i++)
                vector.add(i);
            Thread thread1 = new Thread(){
                public void run() {
                    synchronized (Test.class) {   //进行额外的同步
                        for(int i=0;i<vector.size();i++)
                            vector.remove(i);
                    }
                };
            };
            Thread thread2 = new Thread(){
                public void run() {
                    synchronized (Test.class) {
                        for(int i=0;i<vector.size();i++)
                            vector.get(i);
                    }
                };
            };
            thread1.start();
            thread2.start();
            while(Thread.activeCount()>10)   {
                 
            }
        }
    }
}
参考:http://www.cnblogs.com/dolphin0520/p/3933404.html
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在Java中,同步是指多个线程按照一定的顺序访问共享资源的过程。线程安全问题可以发生在多个线程同时访问和修改共享资源时,可能导致不可预期的结果或数据不一致的情况。为了避免线程安全问题,可以采取以下几种方式: 1. 使用synchronized关键字:synchronized关键字可以用于修饰方法或代码块,确保同一时间只有一个线程访问被修饰的方法或代码块。通过加锁机制,保证了多个线程按顺序访问共享资源。 2. 使用ReentrantLock类:ReentrantLock是Java提供的一种显示锁机制,它可以替代synchronized关键字实现对共享资源的访问控制。它提供了更灵活的锁定方式,例如可重入性、公平性等特性。 3. 使用volatile关键字:volatile关键字用于修饰变量,确保对该变量的读写操作对所有线程可见。它可以避免由于指令重排序等原因导致的数据不一致性。 4. 使用并发容器:Java提供了一些并发容器,如ConcurrentHashMap、ConcurrentLinkedQueue等,它们内部实现了线程安全的机制,可以直接使用而无需额外的同步措施。 5. 使用线程安全的类:Java中提供了一些线程安全的类,如AtomicInteger、AtomicLong等,它们提供了原子操作,可以在多线程环境中保证操作的原子性。 6. 合理设计程序:在编写多线程程序时,要注意合理设计,避免多个线程同时修改同一块共享资源,可以通过分离共享数据、使用局部变量、避免全局变量等方式来减少并发访问的冲突。 需要根据具体的场景和需求选择适合的线程安全措施。同时,了解并发编程的原理和常见的线程安全问题也是很重要的。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

赶路人儿

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值