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

作者:lqh 时间:2022-02-16 19:30:47 

 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,多线程,同步
0
投稿

猜你喜欢

  • Android 应用启动欢迎界面广告的实现实例

    2023-01-26 07:34:11
  • C# Timer控件学习之使用Timer解决按钮幂等性问题

    2021-07-07 11:26:09
  • 浅谈Java中的n种随机数产生办法

    2023-12-22 10:36:29
  • Android使用PowerImageView实现播放强大的ImageView动画效果

    2022-06-16 19:55:54
  • spring aop注解配置代码实例

    2022-02-16 21:13:33
  • java StringBuilder类的详解及简单实例

    2022-04-15 01:24:44
  • Springcloud seata nacos环境搭建过程图解

    2022-11-15 00:34:14
  • Java8中Lambda表达式使用和Stream API详解

    2022-04-27 13:25:23
  • java控制台输出图书馆管理系统

    2022-06-13 01:29:29
  • Springcloud Config配置中心使用与相关介绍

    2021-07-13 05:15:17
  • SpringBoot创建maven多模块项目实战代码

    2023-11-11 04:05:32
  • Java中的"goto"语句妙用

    2023-10-10 03:54:32
  • spring boot整合redis主从sentinel方式

    2021-10-01 20:13:47
  • C#同步和异步调用方法实例

    2022-09-11 21:20:50
  • 带你了解Java Maven的打包操作

    2022-08-03 15:23:33
  • C# 开发step步骤条控件详解

    2021-07-01 00:53:14
  • java对象拷贝详解及实例

    2023-01-20 08:27:00
  • HttpsURLConnection上传文件流(实例讲解)

    2022-03-02 16:54:47
  • C语言实现两个矩阵相乘

    2023-07-22 12:41:20
  • SpringMVC结构简介及常用注解汇总

    2023-10-25 09:16:59
  • asp之家 软件编程 m.aspxhome.com