多线程间的数据共享的几种实现方式比较

需求:设计4个线程,其中两个线程每次对j增加1,另外两个线程对j减少1.

实现数据共享的几种方式比较:

1.使用同一个runnable对象

如果每个线程执行的代码相同,那么可以使用同一个runnable对象,这个runnable有那个共享数据,例如,卖票系统就是这么做的.

如下例所示:

复制代码
...
    public static void main(String[] args) {
        ShareData1 shareData1 = new ShareData1();
        new Thread(shareData1).start();
        new Thread(shareData1).start();
    }

    static class ShareData1 implements Runnable {
        public int count = 100;
        public void run() {
            count--;
            System.out.println("run:"+count);
        }
    }

...
复制代码

 

2.使用不同的runnable对象

如果每个线程执行的代码不同,那么要使用不同的runnable对象,有如下两种方式可以实现runnable对象间的数据共享

1).实现两个runnable对象,将共享数据分别传递给两个不同线程.

复制代码
....
public static void main(String[] args) {
        final ShareData1 shareData1 = new ShareData1();
        new Thread(new MyRunnable1(shareData1)).start();
        new Thread(new MyRunnable1(shareData1)).start();
}
static class MyRunnable1 implements Runnable{
        private ShareData1 shareData1;
        public void run() {
        }
        public MyRunnable1(ShareData1 shareData1){
            this.shareData1 = shareData1;
        }
    }
static class MyRunnable2 implements Runnable{
        private ShareData1 shareData1;
        public void run() {
        }
        public MyRunnable2(ShareData1 shareData1){
            this.shareData1 = shareData1;
        }
    }
static class ShareData1 {
....
}
......
复制代码

 

2).将这些Runnable对象作为一个内部类,将共享数据作为成员变量.

这里用这个方法解决上面的需求,请看下面代码:

复制代码
package com.amos.concurrent;

public class MultiThreadShareData {
    private int j;
    public static void main(String[] args) {
        MultiThreadShareData multiThreadShareData = new MultiThreadShareData();
        for(int i=0;i<2;i++){
            new Thread(multiThreadShareData.new ShareData1()).start();//增加
            new Thread(multiThreadShareData.new ShareData2()).start();//减少
        }
    }
    //自增
    private synchronized void Inc(){
        j++;
        System.out.println(Thread.currentThread().getName()+" inc "+j);
    }
    //自减
    private synchronized void Dec(){
        j--;
        System.out.println(Thread.currentThread().getName()+" dec "+j);
    }
    
    class ShareData1 implements Runnable {
        public void run() {
            for(int i=0;i<5;i++){
                Inc();
            }
        }
    }
    class ShareData2 implements Runnable {
        public void run() {
            for(int i=0;i<5;i++){
                Dec();
            }
        }
    }
}
复制代码

效果:

注:

.上面的代码,首先,是定义了一个全局的变量j,即共享数据;然后,实现Runnable对象,分别去做自增和自减的操作,然后将实现了的Runnable对象作为一个内部类塞给新建的线程;最后循环两遍,实现两个自减和两个自增线程.

.这里要注意的是之所以将自增和自减提出来,是为了方便进行线程安全控制.

3.方法二和方法一的区别在于,方法一是主动将共享数据赋给Runnable对象,方法二则是将数据置为全局变量,然后进行操作.

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值