使用Runnable实现数据共享

使用Runnable实现数据共享,供大家参考,具体内容如下

先上代码:

public class TestThread {
  private static final Logger logger = LoggerFactory.getLogger(TestThread.class);

  private final class MyRunnable implements Runnable {
    private int i;
    public MyRunnable() {
      this.i = 10;
    }
    public void run() {
      while(i > 0) {
        synchronized (this) {
          if (i > 0) {
            i--;
            logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i);
          }
        }
      }
    }
  }

  @Test
  public void testRunable() throws InterruptedException{
    MyRunnable myRunnable = new MyRunnable();
    Thread th1 = new Thread(myRunnable);
    Thread th2 = new Thread(myRunnable);
    th1.start();
    th2.start();
    th1.join();
    th2.join();
  }
}

楼上的代码很简单,模拟一个售票系统。通过两个Thread对象开启两条线程同时运行一个MyRunnable实例。

几个注意点:

1. 没有加上synchronised关键词的话,即

public void run() {
      while(i > 0) {
        if (i > 0) {
          i--;
          logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i);
        }
      }
    }

系统的运行结果:

Thread-1 buy one ticket, 8 left.
Thread-2 buy one ticket, 8 left.
Thread-2 buy one ticket, 6 left.
Thread-1 buy one ticket, 6 left.
Thread-2 buy one ticket, 5 left.
Thread-1 buy one ticket, 4 left.
Thread-2 buy one ticket, 3 left.
Thread-1 buy one ticket, 2 left.
Thread-2 buy one ticket, 1 left.
Thread-1 buy one ticket, 0 left. 

可以看到,缺少同步的程序输出明显有问题。

2. 在进入同步代码块之后,还需要对i的值再进行一次判断,即,如果不加if判断:

public void run() {
      while(i > 0) {
        synchronized (this) {
          i--;
          logger.debug("{} buy one ticket, {} left. ", Thread.currentThread().getName(), i);

        }
      }
    }

程序的运行结果为:

Thread-2 buy one ticket, 9 left.
Thread-2 buy one ticket, 8 left.
Thread-2 buy one ticket, 7 left.
Thread-2 buy one ticket, 6 left.
Thread-2 buy one ticket, 5 left.
Thread-2 buy one ticket, 4 left.
Thread-2 buy one ticket, 3 left.
Thread-2 buy one ticket, 2 left.
Thread-2 buy one ticket, 1 left.
Thread-2 buy one ticket, 0 left.
Thread-1 buy one ticket, -1 left. 

可以看出,出现了“多卖”的现象, 所以需要在进入同步代码块中再进行一次if判断。

总结

synchronised用于互斥访问共享变量, 并在同步代码块中使用if判断更新共享变量。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • java实现Runnable接口适合资源的共享

    本文为大家分享了java实现Runnable接口适合资源的共享,供大家参考,具体内容如下 Java当中,创建线程通常用两种方式: 1.继承Thread类 2.实现Runnable接口 但是在通常的开发当中,一般会选择实现Runnable接口,原因有二: 1.避免单继承的局限,在Java当中一个类可以实现多个接口,但只能继承一个类 2.适合资源的共享 原因1我们经常听到,但是2是什么呢?下面用一个例子来解释: 有5张票,分两个窗口卖: 继承Thread类: public class ThreadD

  • 使用Runnable实现数据共享

    使用Runnable实现数据共享,供大家参考,具体内容如下 先上代码: public class TestThread { private static final Logger logger = LoggerFactory.getLogger(TestThread.class); private final class MyRunnable implements Runnable { private int i; public MyRunnable() { this.i = 10; } pub

  • Java中Thread和Runnable创建线程的方式对比

    目录 一.通过继承Thread创建线程 二.实现Runnable创建检查 三.比较两种创建方式 3.1.多继承 3.2. 数据共享 3.3.线程池 四.源码分析 一.通过继承Thread创建线程 通过继承Thread类,创建一个线程,在主线程中,调用start,让线程处于runnable状态,让系统去运行线程的方法. public class MyThread extends Thread { @Override public void run() { System.out.println("执

  • AngularJS控制器之间的数据共享及通信详解

    AngularJS 本身已经提供了像指令 Directive 和 服务 Service 一类的方式,来实现数据和代码的共享和复用,但在实际的项目开发中,或许是处于懒惰,亦或是为了便利,总会想在两个控制器之间,直接进行数据的共享通信,或者是函数与方法的调用,这里我们就看看有哪些方法可以满足这个要求. 单例服务 单例服务是 AngularJS 本身支持的数据和代码共享方式,因为是单例的,所有的控制器访问的便是同一份数据.比如,下面的 Service 便可以实现: angular .module('a

  • 详解Java中多线程异常捕获Runnable的实现

    详解Java中多线程异常捕获Runnable的实现 1.背景: Java 多线程异常不向主线程抛,自己处理,外部捕获不了异常.所以要实现主线程对子线程异常的捕获. 2.工具: 实现Runnable接口的LayerInitTask类,ThreadException类,线程安全的Vector 3.思路: 向LayerInitTask中传入Vector,记录异常情况,外部遍历,判断,抛出异常. 4.代码: package step5.exception; import java.util.Vector

  • Java中Runnable和Thread的区别分析

    Thread类是在java.lang包中定义的.一个类只要继承了Thread类同时覆写了本类中的run()方法就可以实现多线程操作了,但是一个类只能继承一个父类,这是此方法的局限, 下面看例子: 复制代码 代码如下: package org.thread.demo; class MyThread extends Thread{ private String name; public MyThread(String name) { super(); this.name = name; } publ

  • Runnable.com 在线测试代码片分享网站

    代码片段是开发者每天都要面对的东西,甚至有时候查找代码片段的时间比编写新代码的时间还要多.因为如果找到能够在项目中直接使用的代码片段,这意味着你无需绞尽脑汁"重新发明轮子",且可以令工作效率成倍提升. 但是互联网中存在大量的代码片段,它们分布在成千上万个的网站中,并且你无法确定搜索到的这些代码能否成功运行,是否符合你的需要. Runnable的诞生就是为了解决这一问题.Runnable是一个一站式的代码片段集合网站,你不仅可以搜索代码,还可以编辑.运行这些代码片段,以确保它们的正确性与

  • java多线程编程之使用runnable接口创建线程

    1.将实现Runnable接口的类实例化. 2.建立一个Thread对象,并将第一步实例化后的对象作为参数传入Thread类的构造方法. 最后通过Thread类的start方法建立线程.下面的代码演示了如何使用Runnable接口来创建线程: package mythread;public class MyRunnable implements Runnable{ public void run() {  System.out.println(Thread.currentThread().get

  • Java中的Runnable,Callable,Future,FutureTask的比较

    Java中的Runnable,Callable,Future,FutureTask的比较 Java中存在Runnable.Callable.Future.FutureTask这几个与线程相关的类或者接口,在Java中也是比较重要的几个概念,我们通过下面的简单示例来了解一下它们的作用于区别. Runnable 其中Runnable应该是我们最熟悉的接口,它只有一个run()函数,用于将耗时操作写在其中, 该函数没有返回值 .然后使用某个线程去执行该runnable即可实现多线程,Thread类在调

  • java线程之使用Runnable接口创建线程的方法

    实现Runnable接口的类必须使用Thread类的实例才能创建线程.通过Runnable接口创建线程分为两步: 1. 将实现Runnable接口的类实例化. 2. 建立一个Thread对象,并将第一步实例化后的对象作为参数传入Thread类的构造方法. 最后通过Thread类的start方法建立线程. 下面的代码演示了如何使用Runnable接口来创建线程: 复制代码 代码如下: package mythread; public class MyRunnable implements Runn

  • java 多线程Thread与runnable的区别

    java 多线程Thread与runnable的区别 java中实现多线程的方法有两种:继承Thread类和实现runnable接口 1,继承Thread类,重写父类run()方法 public class thread1 extends Thread { public void run() { for (int i = 0; i < 10000; i++) { System.out.println("我是线程"+this.getId()); } } public static

随机推荐