java核心知识点学习----多线程间的数据共享的几种实现方式比较
2023-09-14 08:58:56 时间
需求:设计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(); } } } }
效果:
注:
1.上面的代码,首先,是定义了一个全局的变量j,即共享数据;然后,实现Runnable对象,分别去做自增和自减的操作,然后将实现了的Runnable对象作为一个内部类塞给新建的线程;最后循环两遍,实现两个自减和两个自增线程.
2.这里要注意的是之所以将自增和自减提出来,是为了方便进行线程安全控制.
3.方法二和方法一的区别在于,方法一是主动将共享数据赋给Runnable对象,方法二则是将数据置为全局变量,然后进行操作.
相关文章
- Java多线程详解_java支持多线程
- java h2数据库_JAVA 项目中使用 H2 数据库
- java分布式事务框架_Java分布式事务,及解决方案
- 用好JAVA中的函数式接口,轻松从通用代码框架中剥离掉业务定制逻辑
- java数组的声明_Java数组定义常用方法[通俗易懂]
- java StringBuffer和StringBuilder
- java核心技术 – 17个重要的知识点
- 怎么才能学好Java?你对Java开发了解吗?Java开发就业方向有哪些?Java开发工程师都需要学习哪些内容?
- java的类型转换异常_类型转换异常英文
- Java递归详解_java难不难学
- idea导入eclipse项目的时候,Java图标变成黄色小J了,怎么解决?
- java并发编程(1):Java多线程-基本线程类-基础知识复习笔记
- 干货:Java并发编程必懂知识点解析详解编程语言
- Java多线程知识点详解编程语言
- 实现使用Java实现Redis消息队列(redis消息队列java)
- Java实现Redis事务管理(redis事务java)
- 解析Java中的Linux路径(java中linux路径)
- 策略研究Java 利用Redis设置过期策略(redisjava过期)
- 时间设置Redis中Java如何设置过期时间(redisjava过期)
- 策略解决Redis Java过期策略问题.(redisjava过期)
- Java无法连接Redis服务器(java连不上redis)
- 互操作Java与Redis缓存互操作实践(redis缓存与java)
- Linux下搭建 Java 开发环境(linux搭建java环境)
- Java程序如何在Linux上顺利部署?快来了解一下!(java部署到Linux)
- Oracle收购了Java一个伟大的转折点(java被oracle吗)