`

线程同步方法一(ReentrantLock)

 
阅读更多
package com.yli.thread.lock;

/**
 * 不加锁,在两个账户之间转账,多次运行会发现两个账号的总额由变化
 * 
 * @author yli
 * 
 */
public class UnLockTest {
	public static final int TOTAL_AMOUNT = 100000;

	public static void main(String[] args) {
		Bank1 bank = new Bank1("yli", 100000);
		Bank1 targetBank = new Bank1("ylz", 0);
		System.out.println(Thread.currentThread().getName() + "=========>总额="
				+ (bank.getBalance() + targetBank.getBalance()));
		for (int i = 0; i < 20000; i++) {
			TransferBank1 transfer = new TransferBank1(bank, targetBank, 5);
			Thread thread = new Thread(transfer);
			thread.setName("Transfer-" + i);
			thread.start();
		}
	}
}

class TransferBank1 implements Runnable {

	private Bank1 bank;
	private Bank1 targetBank;
	private int amount;

	public TransferBank1(Bank1 bank, Bank1 targetBank, int amount) {
		this.bank = bank;
		this.targetBank = targetBank;
		this.amount = amount;
	}

	@Override
	public void run() {
		this.bank.transferAccount(targetBank, amount);
	}

}

class Bank1 {

	private String name;
	private int balance;

	public Bank1(String name, int balance) {
		this.name = name;
		this.balance = balance;
	}

	public void transferAccount(Bank1 targetBank, int amount) {
		this.balance -= amount;
		targetBank.balance += amount;
		int total = this.getBalance() + targetBank.getBalance();
		System.out.println(Thread.currentThread().getName() + "=========>总额="	+ total);
		if (total != UnLockTest.TOTAL_AMOUNT) {
			Thread.currentThread().interrupt();
			try {
				throw new InterruptedException("================>转账错误,抛出异常");
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getBalance() {
		return balance;
	}

	public void setBalance(int balance) {
		this.balance = balance;
	}

}

 

================================华丽丽的分割线=====================================

 

package com.yli.thread.lock;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * 加锁,两个账户之间转账没有影响
 * @author yli
 *
 */
public class ReentLockTest {
	public static final int TOTAL_AMOUNT = 100000;
	
	public static void main(String[] args) {
		Bank bank = new Bank("yli", TOTAL_AMOUNT);
		Bank targetBank = new Bank("ylz", 0);
		System.out.println(Thread.currentThread().getName() + "=========>总额="	+ (bank.getBalance() + targetBank.getBalance()));
		for (int i = 0; i < 20000; i++) {
			TransferBank transfer = new TransferBank(bank, targetBank, 5);
			Thread thread = new Thread(transfer);
			thread.setName("Transfer-" + i);
			thread.start();
		}
	}
}

class TransferBank implements Runnable {

	private Bank bank;
	private Bank targetBank;
	private int amount;

	public TransferBank(Bank bank, Bank targetBank, int amount) {
		this.bank = bank;
		this.targetBank = targetBank;
		this.amount = amount;
	}

	@Override
	public void run() {
		this.bank.transferAccount(targetBank, amount);
	}

}

class Bank {

	private String name;
	private int balance;
	private Lock transferLock;
	private Condition transferCon;

	public Bank(String name, int balance) {
		this.name = name;
		this.balance = balance;
		this.transferLock = new ReentrantLock();
		this.transferCon = this.transferLock.newCondition();
	}

	public void transferAccount(Bank targetBank, int amount) {
		this.transferLock.lock();

		try {
			while (this.balance < amount) {
				this.transferCon.await();
			}
		} catch (InterruptedException e) {
			e.printStackTrace();
		}

		this.balance -= amount;
		targetBank.balance += amount;
		int total = this.getBalance() + targetBank.getBalance();
		System.out.println(Thread.currentThread().getName() + "=========>总额="	+ total);
		if(total != ReentLockTest.TOTAL_AMOUNT){
			Thread.currentThread().interrupt();
			try {
				throw new InterruptedException("================>转账错误,停止运行");
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}

		this.transferCon.signalAll();

		this.transferLock.unlock();
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getBalance() {
		this.transferLock.lock();
		try {
			return balance;
		} finally {
			this.transferLock.unlock();
		}
	}

	public void setBalance(int balance) {
		this.balance = balance;
	}

}

 

分享到:
评论

相关推荐

    java 多线程同步方法的实例

    Java提供了多种机制来实现线程同步,主要包括synchronized关键字、wait()、notify()和notifyAll()方法以及ReentrantLock等。 1. **synchronized关键字**:这是Java中最基本的线程同步方式。它可以用于修饰方法或...

    Java多线程同步.pdf

    在Java语言中,多线程同步机制的实现可以通过synchronized关键字、ReentrantLock类、 Semaphore类、CountDownLatch类等来实现。 1. 使用synchronized关键字 使用synchronized关键字可以对方法或者代码块进行同步。...

    线程同步面试题深入解析

    线程同步是多线程编程中的关键概念,...总的来说,线程同步是Java多线程编程中不可或缺的一部分,正确理解和使用同步机制能确保程序在并发环境下的正确性和高效性。在面试中,理解并能熟练应用这些知识点是非常重要的。

    简单实现多线程同步示例(模拟购票系统)

    为了实现线程同步,Java提供了多种机制,如synchronized关键字、wait()、notify()和notifyAll()方法,以及Lock接口(如ReentrantLock)等。在这个示例中,我们可能会用到synchronized关键字来修饰购票的方法,这样...

    线程同步解决火车站售票问题

    线程同步是多线程编程中一个重要的概念,它确保了多个线程在访问共享资源时能按照一定的顺序或者规则进行,防止数据的不一致性和资源的竞争。在这个“线程同步解决火车站售票问题”的例子中,我们可以通过线程同步...

    Jni多线程同步事例

    在本例“Jni多线程同步事例”中,我们将探讨如何在JNI层面上实现多线程同步,特别是在一个生产者-消费者模型的场景下。 生产者-消费者模型是一种经典的并发问题,它涉及到两个或多个线程之间的协作。在该模型中,...

    一个简单的线程同步实例

    在Java等编程语言中,线程同步可以通过多种方式实现,本实例将探讨其中的几种常见方法。 首先,我们来看最基本的同步机制——synchronized关键字。synchronized可以用于方法或者代码块,当它包裹的代码被执行时,会...

    操作系统实验 多线程同步与互斥 java编写 有界面

    操作系统实验是计算机科学教育中的重要组成部分,它...总之,理解和掌握多线程同步与互斥的概念是成为一名合格的Java开发者的关键技能之一。通过实际的编程实验,可以加深对这些概念的理解,并锻炼解决并发问题的能力。

    售票窗口模拟线程同步

    在多线程环境中,线程同步是一项关键的技术,用于确保多个线程访问共享资源时的正确性和一致性,避免资源竞争和数据不一致等问题。本文将通过一个具体的案例——售票窗口模拟——来深入探讨线程同步的原理和实现。 ...

    多线程(11)ReentrantLock公平锁与非公平锁(修改)1

    总的来说,这段代码展示了`ReentrantLock`的公平和非公平锁的使用,并通过`CountDownLatch`进行多线程同步。了解这两种锁的区别和应用场景对于优化并发程序的性能和线程调度非常重要。在实际开发中,应根据具体需求...

    多线程同步和通讯完整示例

    在编程领域,尤其是在Java这样的多线程环境中,理解和掌握多线程同步与通讯至关重要。本文将深入探讨这些概念,以及如何使用synchronized关键字、wait-notify机制和Lock接口来实现线程间的同步与通讯。 首先,多...

    java线程同步详解

    Java中的线程同步还有其他工具,例如`java.util.concurrent`包中的`Semaphore`、`ReentrantLock`等,它们提供了更高级的同步控制,如信号量控制、公平锁、非公平锁等。 线程同步的主要目的是防止数据竞争(Data ...

    Java多线程同步具体实例讲解 .doc

    总结起来,Java多线程同步是通过`synchronized`关键字实现的,它可以应用于方法或代码块,保证同一时刻只有一个线程能够执行特定的代码。通过合理使用同步机制,开发者可以有效地管理并发程序中的资源访问,避免数据...

    Java并发之ReentrantLock类源码解析

    ReentrantLock的lock方法使用CAS操作来尝试更新对象内的一个变量,如果更新成功,则将exclusiveOwnerThread变量设置为当前线程,然后lock方法会立刻返回。如果更新不成功,则调用acquire(1)方法。 ReentrantLock的...

    线程同步编程实例

    在多线程编程中,线程同步是一种控制多个线程并发执行的技术,以避免数据竞争和其他潜在的问题。本文将深入探讨线程同步的概念,并通过一个简单的“售票”实例来阐述如何在Java中实现线程同步。 线程同步的目的是...

    某电信项目多线程同步数据实例

    在IT行业中,尤其是在大型系统开发中,多线程同步数据是一个关键的技术环节,尤其是在处理大量实时数据的场景,如电信项目。"某电信项目多线程同步数据实例"的标题揭示了一个具体的应用案例,它表明在该电信项目中,...

    Java多线程同步具体实例.doc

    Java多线程同步是编程中一个非常重要的概念,特别是在并发编程中,用于解决多个线程访问共享资源时可能引发的数据不一致问题。本实例通过一个简单的火车票售票系统来演示同步机制的应用。 在这个实例中,我们创建了...

    线程同步的一个小程序

    线程同步是多线程编程中的一个重要概念,它主要用于解决多个线程并发访问共享资源时可能出现的竞争条件问题。在这个名为“线程同步的一个小程序”中,我们可以推测这是一段用于实验和理解线程同步机制的代码。下面...

    java课程设计线程同步五子棋.zip

    在Java编程语言中,线程同步是一个关键概念,特别是在多线程编程中,它用于确保多个线程在访问共享资源时能正确协调,避免数据不一致性和竞态条件。五子棋游戏是一个典型的多线程应用场景,因为它涉及到两个玩家交替...

Global site tag (gtag.js) - Google Analytics