• java核心知识点学习----多线程间的数据共享的几种实现方式比较


    需求:设计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对象,方法二则是将数据置为全局变量,然后进行操作.

  • 相关阅读:
    bootstrap学习(全局CSS样式)(二)
    第八周
    bootstrap学习(全局CSS样式)(一)
    变量的解构赋值 ES6
    ES6的let和const命令
    github之怎么上传本地项目
    浅谈js对象及对象属性
    解决sublime text3 文件名,小框框的办法
    web04--cookie
    web03--session
  • 原文地址:https://www.cnblogs.com/amosli/p/3677567.html
Copyright © 2020-2023  润新知