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){ }
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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

핫 AI 도구

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

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

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

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

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

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

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

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

뜨거운 주제











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

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

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

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

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

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

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