温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

在java项目中如何实现同步多线程

发布时间:2020-11-18 15:15:55 来源:亿速云 阅读:153 作者:Leah 栏目:编程语言

本篇文章给大家分享的是有关在java项目中如何实现同步多线程,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

 java多线程的同步方法实例代码

先看一个段有关银行存钱的代码:

class Bank {
  private int sum;
  public void add(int num){
    sum = sum + num;
    try {
      Thread.sleep(10);
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
    System.out.println("total num is : " + sum);
  }
}
class Custom implements Runnable{
  private Bank b = new Bank();

  @Override
  public void run() {
    for(int i = 3 ; i > 0 ; i--)
      b.add(100);
  }
}
public class BankDemo{
  public static void main(String[] args) {
    Custom custom = new Custom();
    Thread t1 = new Thread(custom);
    Thread t2 = new Thread(custom);
    t1.start();
    t2.start();
  }
}

此代码的运行结果为:

total num is : 100
total num is : 300
total num is : 400
total num is : 500
total num is : 500
total num is : 600

可以看出sum的值与预期的效果不太一样;造成这种现象的原因有两个:

1.程序存在两个以上的子线程;

2.子线程中存在多条语句操作同一变量;

上述例子中:创建了两个子线程·t1 和 t2,分别向银行中存钱。但是可以看出银行的实力随着Custom的创建,只创建了一个对象。也就是说我们只操作一个数据变量即为银行中钱的总数sum;当两个子线程开启的时候run方法中调用了bank的add方法,而add方法中有两个语句都在操作sum一个sum的增加,一个是打印sum,当两个子线程抢占cpu执行各自的程序的时候会出现:

当t1执行到add以后,t2抢到了cpu的执行权,执行也是执行了add语句,随后打印出sum的值,这时候由于sum增加了两次,所以打印出来的sum值为200。类推,假如这个时候t1又抢回了cpu的执行权,因此又打印出一次200。

显然这种现象是我们不希望产生的。我们希望一个线程存完钱然后打印出结果,之后才允许下一次添加操作。这就是多线程会产生的问题,线程不安全。

我们应尽量避免这种现象的发生,Java给我们提供了三种方法来解决这个问题:

第一种:同步代码块

//private Object obj = new Object();
  public void add(int num) {
    synchronized (this) {
      sum = sum + num;
      try {
        Thread.sleep(10);
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
      System.out.println("total num is : " + sum);
    }
  }

将多线程中需要操作同一数据对象的语句使用同步代码块包含。同步代码块的原理就是:

1.java中每个对象都有一个内置锁;

2.当程序运行到同步代码块的时候首先会获取指定对象的锁,这个锁对于多个线程来说是唯一的。我们可以创建任意一个对象(obj)让他当作同步代码块的锁。

3.当程序中只有一个只有一个锁的话我们还可以使用this,this代表当前执行代码所操作的实例对象的锁。即拥有add方法的类的对象,即bank。

4.两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。

这样就可以操作同一个数据的多条语句只能在“同一段时间”只能被一个子线程所操作。

第二种 同步函数

 public synchronized void add(int num) {
      sum = sum + num;
      try {
        Thread.sleep(10);
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
      System.out.println("total num is : " + sum);
    }

除了同步代码块以外我们还可以将需要同步的操作抽象成一个函数,然后将这个函数用synchronized修饰,形成同步方法。比如上述例子中的add方法中的语句都在操作sum对象。我们就可以将add方法使用synchronized修饰。这样也能达到代码同步的效果。

同步方法使用的锁其实就是 this。

值得一提的是:同步方法和同步代码块,在开发程序的时候我们更推荐使用同步代码块。

1.同步代码块可以绑定任意对象,而同步函数只能绑定该类对象this

2.如果多个线程使用同一个锁的话,那么两者均可以使用,如果存在多个锁的(比如,在一个对象的同步方法里面调用另外一个对象的同步方法,则获取了两个对象的同步锁),只能使用同步代码块。
静态方法的同步
同步方法

public synchronized static void add(int num){}

同步代码块:

public synchronized void add(int num){
  synchronized (Bank.Class) {
  }
}

静态方法的默认同步锁是当前方法所在类的.class 对象,注意this与static不可以连用,所以不能使用this.Class

以上就是在java项目中如何实现同步多线程,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注亿速云行业资讯频道。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI