Java Java시작하기 Java를 동기화하는 방법

Java를 동기화하는 방법

Nov 12, 2019 pm 05:59 PM
java 동기식

Java를 동기화하는 방법

동기화를 달성해야 하는 이유

java는 다중 스레드 동시성을 허용합니다. control 에서는 여러 스레드가 공유 가능한 리소스 변수를 동시에 조작(데이터 추가, 삭제, 수정, 확인 등)할 경우 부정확한 데이터 및 서로 충돌이 발생하므로 이를 방지하기 위해 동기화 잠금을 추가합니다. 작업이 완료되기 전에 스레드를 호출하여 변수의 고유성과 정확성을 보장합니다.

1. 예시

예를 들어 은행 계좌를 두 개의 스레드로 동시에 운영하는 경우 하나에는 100위안이 걸리고 기타 예금 돈 100 위안. 원래 계정에 블록이 0개 있다고 가정하면 출금 스레드와 입금 스레드가 동시에 발생하면 어떻게 될까요? 출금에 실패했으며 계좌 잔액은 100입니다. 출금이 성공되었으며 계좌잔고는 0입니다. 그러나 어느 잔액이 어느 잔액에 해당합니까? 명확하게 말하기 어렵기 때문에 다중 스레드 동기화 문제가 발생합니다.

2. 동기화가 되지 않는 상황

예를 들어 은행 계좌를 두 개의 스레드로 동시에 운영한다면 하나는 100블록을 차지합니다. , 각각 100 위안을 절약합니다. 원래 계정에 블록이 0개 있다고 가정하면 출금 스레드와 입금 스레드가 동시에 발생하면 어떻게 될까요? 출금에 실패하면 계좌잔고는 100이 되고, 출금에 성공하면 계좌잔고는 0이 됩니다. 그러나 어느 잔액이 어느 잔액에 해당합니까? 명확하게 말하기 어렵기 때문에 다중 스레드 동기화 문제가 발생합니다.

public class Bank {
   private int count =0;//账户余额
   
   //存钱
   public  void addMoney(int money){
       count +=money;
       System.out.println(System.currentTimeMillis()+"存进:"+money);
   }
    
    //取钱
    public  void subMoney(int money){
        if(count-money < 0){
            System.out.println("余额不足");
            return;
        }
        count -=money;
        System.out.println(+System.currentTimeMillis()+"取出:"+money);
    }
    
    //查询
    public void lookMoney(){
        System.out.println("账户余额:"+count);
    }
}
로그인 후 복사
package threadTest;
public class SyncThreadTest {
public static void main(String args[]){
final Bank bank=new Bank();
Thread tadd=new Thread(new Runnable() {
    
    @Override
    public void run() {
        // TODO Auto-generated method stub
        while(true){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            bank.addMoney(100);
            bank.lookMoney();
            System.out.println("\n");
            
        }
    }
});
Thread tsub = new Thread(new Runnable() {
    
    @Override
    public void run() {
        // TODO Auto-generated method stub
        while(true){
            bank.subMoney(100);
            bank.lookMoney();
            System.out.println("\n");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }    
        }
    }
});
tsub.start();
tadd.start();
}
}
로그인 후 복사

실행 결과:

1502542307917取出:100
账号余额:100
1502542308917存进:100
1502542308917取出:100
账号余额:0
账号余额:0
1502542309917存进:100
账号余额:0
1502542309917取出:100
账号余额:0
로그인 후 복사

두 스레드가 동시에 비동기화된 메서드에 액세스하므로 스레드가 아닌 안전 문제가 발생합니다. 동시에 개체의 인스턴스 변수는 스레드가 아닌 안전 문제를 일으킬 수 있습니다.

해결책: public void run() 앞에 동기화 키워드를 추가하기만 하면 됩니다.

3. 동기화 방법

동기화 키워드 수정 방법

그건 동기화 키워드로 수정된 메소드가 있습니다. Java의 모든 객체에는 내장 잠금이 있으므로 이 키워드로 메서드를 수정하면 내장 잠금이 전체 메서드를 보호합니다. 이 메서드를 호출하기 전에 내장 잠금을 얻어야 합니다. 그렇지 않으면 차단됩니다.

코드:

public synchronized void save(){}
로그인 후 복사

참고: 동기화된 키워드는 정적 메서드도 수정할 수 있습니다. 이때 정적 메서드가 호출되면 전체 클래스가 잠깁니다.

public class Bank {
private int count =0;//账户余额
//存钱
public  synchronized void addMoney(int money){
count +=money;
System.out.println(System.currentTimeMillis()+"存进:"+money);
}
//取钱
public  synchronized void subMoney(int money){
if(count-money < 0){
    System.out.println("余额不足");
    return;
}
count -=money;
System.out.println(+System.currentTimeMillis()+"取出:"+money);
}
//查询
public void lookMoney(){
System.out.println("账户余额:"+count);
}
}
로그인 후 복사

실행 결과:

余额不足
账号余额:0
1502543814934存进:100
账号余额:100
1502543815934存进:100
账号余额:200
1502543815934取出:100
账号余额:100
로그인 후 복사

스레드 동기화는 이런 방식으로 이루어집니다

동기화된 코드 블록#🎜🎜 #

즉, 동기화된 키워드로 수정된 명령문 블록입니다.

이 키워드로 수정된 명령문 블록은 동기화를 달성하기 위해 기본 제공 잠금과 함께 자동으로 추가됩니다.

코드:

synchronized(object){ 
}
로그인 후 복사

Note : 동기화 오버헤드가 높은 작업이므로 동기화 내용을 최소화해야 합니다.

일반적으로 전체 메소드를 동기화할 필요는 없으며, 동기화된 코드 블록을 사용하여 키 코드를 동기화하면 됩니다.

public class Bank {
private int count =0;//账户余额
//存钱
public  void addMoney(int money){
synchronized (this) {
    count +=money;
}
System.out.println(System.currentTimeMillis()+"存进:"+money);
}
//取钱
public   void subMoney(int money){
synchronized (this) {
    if(count-money < 0){
        System.out.println("余额不足");
        return;
    }
    count -=money;
}
System.out.println(+System.currentTimeMillis()+"取出:"+money);
}
//查询
public void lookMoney(){
System.out.println("账户余额:"+count);
}
}
로그인 후 복사

실행 결과는 다음과 같습니다.

余额不足
账户余额:0
余额不足
账户余额:100
1502544966411存进:100
账户余额:100
1502544967411存进:100
账户余额:100
1502544967411取出:100
账户余额:100
1502544968422取出:100
로그인 후 복사

이것도 스레드 동기화를 달성하며, 메서드 동기화보다 작업 효율성이 높으므로 비용이 많이 드는 작업입니다. 동기화된 콘텐츠는 최소화되어야 합니다. 일반적으로 전체 메소드를 동기화할 필요는 없으며, 동기화된 코드 블록을 사용하여 키 코드를 동기화하면 됩니다.

스레드 동기화를 달성하기 위해 특수 도메인 변수(휘발성) 사용

a.휘발성 키워드는 멤버 변수에 액세스하는 자유로운 방법을 제공합니다. 잠금 메커니즘;

b. 휘발성을 사용하여 멤버 변수를 수정하는 것은 해당 필드가 다른 스레드에 의해 업데이트될 수 있음을 가상 머신에 알리는 것과 같습니다.

c. 레지스터의 값을 사용하는 대신 재계산하기 위해

d.휘발성은 원자 연산을 제공하지 않으며 최종 유형 변수를 수정하는 데 사용할 수도 없습니다.

Bank.java 코드는 다음과 같습니다.

package com.thread.demo;
/**
* Created by HJS on 2017/8/12.
*/
public class Bank {
private volatile int count =0;//账户余额
//存钱
public  void addMoney(int money){
synchronized (this) {
    count +=money;
}
System.out.println(System.currentTimeMillis()+"存进:"+money);
}
//取钱
public   void subMoney(int money){
synchronized (this) {
    if(count-money < 0){
        System.out.println("余额不足");
        return;
    }
    count -=money;
}
System.out.println(+System.currentTimeMillis()+"取出:"+money);
}
//查询
public void lookMoney(){
System.out.println("账户余额:"+count);
}
}
로그인 후 복사

실행 결과:

余额不足
账户余额:0
余额不足
账户余额:100
1502546287474存进:100
账户余额:100
1502546288474存进:100
1502546288474取出:100
账户余额:100
로그인 후 복사

이때, 순서가 또 엉망이 되어버렸음을 나타냅니다. 또 다른 동기화 문제가 있습니다. 휘발성은 원자적 작업을 보장할 수 없기 때문에 동기화된 것을 대체할 수 없습니다. 게다가 휘발성은 컴파일러가 코드를 최적화하는 것을 방해하므로 사용할 수 없다면 적용하지 마세요. 그 원칙은 스레드가 휘발성 수정 변수에 액세스하려고 할 때마다 캐시에서 읽는 대신 메모리에서 읽으므로 각 스레드에서 액세스하는 변수 값이 동일하다는 것입니다. 이렇게 하면 동기화가 보장됩니다.

재진입 잠금을 사용하여 스레드 동기화 달성

동기화를 지원하기 위해 새로운 java.util.concurrent 패키지가 JavaSE5.0에 추가되었습니다. ReentrantLock 클래스는 Lock 인터페이스를 구현하는 재진입 상호 배타적 잠금입니다. 이 잠금은 동기화된 메서드 및 블록을 사용하는 것과 동일한 기본 동작 및 의미를 가지며 해당 기능을 확장합니다.

ReenreantLock 클래스의 일반적으로 사용되는 메서드는 다음과 같습니다.

ReentrantLock(): ReentrantLock 인스턴스 만들기

lock(): 잠금 획득 #🎜 🎜## 🎜🎜#unlock(): 잠금 해제

참고: ReentrantLock()에도 공정한 잠금을 생성할 수 있는 생성 방법이 있지만 효율성이 크게 떨어질 수 있으므로 권장하지 않습니다. 프로그램의.

Bank.java 코드는 다음과 같이 수정됩니다.

public class Bank {
private  int count = 0;// 账户余额
//需要声明这个锁
private Lock lock = new ReentrantLock();
// 存钱
public void addMoney(int money) {
lock.lock();//上锁
try{
    count += money;
    System.out.println(System.currentTimeMillis() + "存进:" + money);
}finally{
    lock.unlock();//解锁
}
}
// 取钱
public void subMoney(int money) {
lock.lock();
try{
    if (count - money < 0) {
        System.out.println("余额不足");
        return;
    }
    count -= money;
    System.out.println(+System.currentTimeMillis() + "取出:" + money);
}finally{
    lock.unlock();
}
}
// 查询
public void lookMoney() {
System.out.println("账户余额:" + count);
}
}
로그인 후 복사

실행 결과:

余额不足
账户余额:0
1502547439892存进:100
账户余额:100
1502547440892存进:100
账户余额:200
1502547440892取出:100
账户余额:100
로그인 후 복사

참고: 잠금 개체 및 동기화 키워드 선택에 관해: #🎜 🎜#

a 둘 중 어느 것도 사용하지 않는 것이 가장 좋으며, 사용자가 모든 잠금 관련 코드를 처리할 수 있도록 java.util.concurrent 패키지에서 제공하는 메커니즘을 사용하는 것이 좋습니다.

b.如果synchronized关键字能满足用户的需求,就用synchronized,因为它能简化代码。

c.如果需要更高级的功能,就用ReentrantLock类,此时要注意及时释放锁,否则会出现死锁,通常在finally代码释放锁 。

使用局部变量实现线程同步

代码如下:

public class Bank {
private static ThreadLocal<Integer> count = new ThreadLocal<Integer>(){
@Override
protected Integer initialValue() {
    // TODO Auto-generated method stub
    return 0;
}
};
// 存钱
public void addMoney(int money) {
count.set(count.get()+money);
System.out.println(System.currentTimeMillis() + "存进:" + money);
}
// 取钱
public void subMoney(int money) {
if (count.get() - money < 0) {
    System.out.println("余额不足");
    return;
}
count.set(count.get()- money);
System.out.println(+System.currentTimeMillis() + "取出:" + money);
}
// 查询
public void lookMoney() {
System.out.println("账户余额:" + count.get());
}
}
로그인 후 복사

运行结果如下:

复制代码
余额不足
账户余额:0
余额不足
1502547748383存进:100
账户余额:100
账户余额:0
余额不足
账户余额:0
1502547749383存进:100
账户余额:200
로그인 후 복사

看了运行效果,一开始一头雾水,怎么只让存,不让取啊?看看ThreadLocal的原理:

如果使用ThreadLocal管理变量,则每一个使用该变量的线程都获得该变量的副本,副本之间相互独立,这样每一个线程都可以随意修改自己的变量副本,而不会对其他线程产生影响。现在明白了吧,原来每个线程运行的都是一个副本,也就是说存钱和取钱是两个账户,知识名字相同而已。所以就会发生上面的效果。

ThreadLocal 类的常用方法

ThreadLocal() : 创建一个线程本地变量     

get() : 返回此线程局部变量的当前线程副本中的值     

initialValue() : 返回此线程局部变量的当前线程的"初始值"     

set(T value) : 将此线程局部变量的当前线程副本中的值设置为value

注:ThreadLocal与同步机制

a.ThreadLocal与同步机制都是为了解决多线程中相同变量的访问冲突问题。 

b.前者采用以"空间换时间"的方法,后者采用以"时间换空间"的方式。

php中文网,大量的免费Java入门教程,欢迎在线学习!

위 내용은 Java를 동기화하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

자바의 완전수 자바의 완전수 Aug 30, 2024 pm 04:28 PM

Java의 완전수 가이드. 여기서는 정의, Java에서 완전 숫자를 확인하는 방법, 코드 구현 예제에 대해 논의합니다.

자바의 웨카 자바의 웨카 Aug 30, 2024 pm 04:28 PM

Java의 Weka 가이드. 여기에서는 소개, weka java 사용 방법, 플랫폼 유형 및 장점을 예제와 함께 설명합니다.

Java의 스미스 번호 Java의 스미스 번호 Aug 30, 2024 pm 04:28 PM

Java의 Smith Number 가이드. 여기서는 정의, Java에서 스미스 번호를 확인하는 방법에 대해 논의합니다. 코드 구현의 예.

Java Spring 인터뷰 질문 Java Spring 인터뷰 질문 Aug 30, 2024 pm 04:29 PM

이 기사에서는 가장 많이 묻는 Java Spring 면접 질문과 자세한 답변을 보관했습니다. 그래야 면접에 합격할 수 있습니다.

Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Java 8 Stream foreach에서 나누거나 돌아 오시겠습니까? Feb 07, 2025 pm 12:09 PM

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

Java의 날짜까지의 타임스탬프 Java의 날짜까지의 타임스탬프 Aug 30, 2024 pm 04:28 PM

Java의 TimeStamp to Date 안내. 여기서는 소개와 예제와 함께 Java에서 타임스탬프를 날짜로 변환하는 방법에 대해서도 설명합니다.

캡슐의 양을 찾기위한 Java 프로그램 캡슐의 양을 찾기위한 Java 프로그램 Feb 07, 2025 am 11:37 AM

캡슐은 3 차원 기하학적 그림이며, 양쪽 끝에 실린더와 반구로 구성됩니다. 캡슐의 부피는 실린더의 부피와 양쪽 끝에 반구의 부피를 첨가하여 계산할 수 있습니다. 이 튜토리얼은 다른 방법을 사용하여 Java에서 주어진 캡슐의 부피를 계산하는 방법에 대해 논의합니다. 캡슐 볼륨 공식 캡슐 볼륨에 대한 공식은 다음과 같습니다. 캡슐 부피 = 원통형 볼륨 2 반구 볼륨 안에, R : 반구의 반경. H : 실린더의 높이 (반구 제외). 예 1 입력하다 반경 = 5 단위 높이 = 10 단위 산출 볼륨 = 1570.8 입방 단위 설명하다 공식을 사용하여 볼륨 계산 : 부피 = π × r2 × h (4

Spring Tool Suite에서 첫 번째 Spring Boot 응용 프로그램을 실행하는 방법은 무엇입니까? Spring Tool Suite에서 첫 번째 Spring Boot 응용 프로그램을 실행하는 방법은 무엇입니까? Feb 07, 2025 pm 12:11 PM

Spring Boot는 강력하고 확장 가능하며 생산 가능한 Java 응용 프로그램의 생성을 단순화하여 Java 개발에 혁명을 일으킨다. Spring Ecosystem에 내재 된 "구성에 대한 협약"접근 방식은 수동 설정, Allo를 최소화합니다.

See all articles