目录
使用同步方法
使用同步语句或块
什么是同步?
为啥需要同步?
总结
首页 Java java教程 一个例子看懂Java中synchronized关键字到底怎么用

一个例子看懂Java中synchronized关键字到底怎么用

Oct 08, 2022 pm 04:02 PM
java

本篇文章给大家带来了关于java的相关知识,其中主要介绍了关于synchronized关键字的相关问题,包括了使用同步方法、使用同步语句或块以及什么是同步、为什么需要同步的相关内容,下面一起来看一下,希望对大家有帮助。

推荐学习:《java视频教程

在平时开发中,synchronized关键字经常遇到,你知道synchronized怎么用吗?本文给大家介绍一下。

我们有两种方法使用同步:

  • 使用同步方法
  • 使用同步语句或块

使用同步方法

要使方法同步,只需将synchronized关键字添加到其声明中:

public class SynchronizedDemo {

    private int i = 0;

    public synchronized void add() {
        i++;
    }

    public synchronized void del() {
        i--;
    }

    public synchronized int getValue() {
        return i;
    }
}
登录后复制

如上代码显示,一共有三个同步方法:

  • add()
  • del()
  • getValue()

每个方法同一个对象同一时刻只会被调用一次,比如一个线程在调用add()时,其他线程都会被阻塞,直到第一个线程处理完add()方法。

使用同步语句或块

    public void del(int value){

        synchronized(this){
            this.i -= value;
        }
    }
登录后复制

如上代码,synchronized加在了一个{}代码前,这个就代表是一个同步代码块。

以上就是synchronized关键字两种使用方法,下面我们来简单的介绍一下同步相关的概念。

什么是同步?

同步是一个控制多个线程访问任何共享资源的进程,可以避免不一致的结果。使用同步的主要目的是避免线程的不一致行为,防止线程干扰。

在java中可以使用synchronized 关键字实现同步的效果,synchronized只能应用于方法和块,不能应用于变量和类。

为啥需要同步?

首先我们来看一段代码:

public class SynchronizedDemo {

    int i;

    public void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();
        synchronizedDemo.increment();
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

每当调用increment()方法时计算值都会加1:

调用2次就会加2,调用3次就会加3,调用4次就会加4:

public class SynchronizedDemo {

    int i;

    public void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();
        synchronizedDemo.increment();
        synchronizedDemo.increment();
        synchronizedDemo.increment();
        synchronizedDemo.increment();
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

现在我们扩展一下上面的例子,创建一个线程去调用10次increment()方法:

public class SynchronizedDemo {

    int i;

    public void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();
        Thread thread = new Thread(() -> {
            for (int i = 1; i <= 10; i++) {
                synchronizedDemo.increment();
            }
        });
        thread.start();
        try {
            thread.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

此时计算的结果正如我们预料的那样,结果为10.

这是单线程的情况,一切都是如此的美好,但是事实真的如此吗?如果是多线程环境,会是什么样的?

下面我们来演示一下多线程的情况!

public class SynchronizedDemo {

    int i;

    public void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();

        Thread thread1 = new Thread(() -> {
            for (int i = 1; i <= 1000; i++) {
                synchronizedDemo.increment();
            }
        });

        Thread thread2 = new Thread(() -> {
            for (int i = 1; i <= 1000; i++) {
                synchronizedDemo.increment();
            }
        });

        thread1.start();
        thread2.start();
        try {
            thread1.join();
            thread2.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

如上代码,我们创建了两个线程 thread1 和 thread2,每个线程调用1000次increment(),理论上最终打印的值应该是2000,因为thread1调用increment()1000次后值会变成1000,thread2调用increment()1000次后值会变成2000.

我们执行一下,看看结果:

结果和我们想的不一样,小于2000,我们再执行一下:

结果还是小于2000.

这是为什么呢?

因为多线程支持并行处理,因此,两个线程总是有可能同时获取计数器的值,因此都得到相同的计数器值,所以在这种情况下,不是递增计数器的值两次,只增加一次。

那么,如何避免这种情况呢?

使用 synchronized 关键字即可解决。

我们只需要将increment()方法加上synchronized就可以了:

public class SynchronizedDemo {

    int i;

    public synchronized void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();

        Thread thread1 = new Thread(() -> {
            for (int i = 1; i <= 1000; i++) {
                synchronizedDemo.increment();
            }
        });

        Thread thread2 = new Thread(() -> {
            for (int i = 1; i <= 1000; i++) {
                synchronizedDemo.increment();
            }
        });

        thread1.start();
        thread2.start();
        try {
            thread1.join();
            thread2.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

这个时候我们再执行一下:

可以看到,值为2000.

我们把计算次数提高到10000次:

public class SynchronizedDemo {

    int i;

    public synchronized void increment() {
        i++;
    }

    public static void main(String[] args) {
        SynchronizedDemo synchronizedDemo = new SynchronizedDemo();

        Thread thread1 = new Thread(() -> {
            for (int i = 1; i <= 10000; i++) {
                synchronizedDemo.increment();
            }
        });

        Thread thread2 = new Thread(() -> {
            for (int i = 1; i <= 10000; i++) {
                synchronizedDemo.increment();
            }
        });

        thread1.start();
        thread2.start();
        try {
            thread1.join();
            thread2.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("计算值为:" + synchronizedDemo.i);
    }
}
登录后复制

执行结果为:

可以看出,一个小小的synchronized竟然那么简单的就解决了这个问题。

这个背后的原理就是线程1执行increment()方法时,因为有synchronized,所以会自动将此方法加锁,而此时只有线程1拥有这把锁,其他线程只能等待,直到线程1释放这把锁,线程2才能参与调用。

同理,当线程2去调用increment()时,线程2拿到锁,线程1进入等待,直到线程2释放锁,就这样,直到计算完毕,在此过程中,不会出现计算错误的情况。

总结

  • synchronized 关键字是使块或方法同步的唯一方法。
  • synchronized 关键字提供了锁的特性,它确保线程之间不会出现竞争条件。被锁定后,线程只能从主存中读取数据,读取数据后,它会刷新写操作,然后才能释放锁。
  • synchronized 关键字还有助于避免程序语句的重新排序。

推荐学习:《java视频教程

以上是一个例子看懂Java中synchronized关键字到底怎么用的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Java 中的完美数 Java 中的完美数 Aug 30, 2024 pm 04:28 PM

Java 完美数指南。这里我们讨论定义,如何在 Java 中检查完美数?,示例和代码实现。

Java 中的随机数生成器 Java 中的随机数生成器 Aug 30, 2024 pm 04:27 PM

Java 随机数生成器指南。在这里,我们通过示例讨论 Java 中的函数,并通过示例讨论两个不同的生成器。

Java中的Weka Java中的Weka Aug 30, 2024 pm 04:28 PM

Java 版 Weka 指南。这里我们通过示例讨论简介、如何使用weka java、平台类型和优点。

Java 中的史密斯数 Java 中的史密斯数 Aug 30, 2024 pm 04:28 PM

Java 史密斯数指南。这里我们讨论定义,如何在Java中检查史密斯号?带有代码实现的示例。

Java Spring 面试题 Java Spring 面试题 Aug 30, 2024 pm 04:29 PM

在本文中,我们保留了最常被问到的 Java Spring 面试问题及其详细答案。这样你就可以顺利通过面试。

突破或从Java 8流返回? 突破或从Java 8流返回? Feb 07, 2025 pm 12:09 PM

Java 8引入了Stream API,提供了一种强大且表达力丰富的处理数据集合的方式。然而,使用Stream时,一个常见问题是:如何从forEach操作中中断或返回? 传统循环允许提前中断或返回,但Stream的forEach方法并不直接支持这种方式。本文将解释原因,并探讨在Stream处理系统中实现提前终止的替代方法。 延伸阅读: Java Stream API改进 理解Stream forEach forEach方法是一个终端操作,它对Stream中的每个元素执行一个操作。它的设计意图是处

Java 中的时间戳至今 Java 中的时间戳至今 Aug 30, 2024 pm 04:28 PM

Java 中的时间戳到日期指南。这里我们还结合示例讨论了介绍以及如何在java中将时间戳转换为日期。

创造未来:面向零基础的 Java 编程 创造未来:面向零基础的 Java 编程 Oct 13, 2024 pm 01:32 PM

Java是热门编程语言,适合初学者和经验丰富的开发者学习。本教程从基础概念出发,逐步深入讲解高级主题。安装Java开发工具包后,可通过创建简单的“Hello,World!”程序实践编程。理解代码后,使用命令提示符编译并运行程序,控制台上将输出“Hello,World!”。学习Java开启了编程之旅,随着掌握程度加深,可创建更复杂的应用程序。

See all articles