java锁对象

在Java5中,专门提供了锁对象,利用锁可以方便的实现资源的封锁,用来控制对竞争资源并发访问的控制,这些内容主要集中在java.util.concurrent.locks 包下面,里面有三个重要的接口Condition、Lock、ReadWriteLock。

Condition: 
Condition 将 Object 监视器方法(wait、notify 和 notifyAll)分解成截然不同的对象,以便通过将这些对象与任意 Lock 实现组合使用,为每个对象提供多个等待 set (wait-set)。

Lock: 
Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。

ReadWriteLock: 
ReadWriteLock 维护了一对相关的锁定,一个用于只读操作,另一个用于写入操作。

有关锁的介绍,API文档解说很多,看得很烦,还是看个例子再看文档比较容易理解。

package cn.thread;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 *
 *
 * @author 林计钦
 * @version 1.0 2013-7-25 上午10:33:37
 */
public class LockTest {
    public static void main(String[] args) {
        LockTest test = new LockTest();

        // 创建并发访问的账户
        MyCount myCount = test.new MyCount("95599200901215522", 10000);
        // 创建一个锁对象
        Lock lock = new ReentrantLock();
        // 创建一个线程池
        ExecutorService pool = Executors.newCachedThreadPool();
        // 创建一些并发访问用户,一个信用卡,存的存,取的取,好热闹啊
        User u1 = test.new User("张三", myCount, -4000, lock);
        User u2 = test.new User("张三他爹", myCount, 6000, lock);
        User u3 = test.new User("张三他弟", myCount, -8000, lock);
        User u4 = test.new User("张三", myCount, 800, lock);
        // 在线程池中执行各个用户的操作
        pool.execute(u1);
        pool.execute(u2);
        pool.execute(u3);
        pool.execute(u4);
        // 关闭线程池
        pool.shutdown();
    }

    /**
     * 信用卡的用户
     */
    class User implements Runnable {
        private String name; // 用户名
        private MyCount myCount; // 所要操作的账户
        private int iocash; // 操作的金额,当然有正负之分了
        private Lock myLock; // 执行操作所需的锁对象

        User(String name, MyCount myCount, int iocash, Lock myLock) {
            this.name = name;
            this.myCount = myCount;
            this.iocash = iocash;
            this.myLock = myLock;
        }

        public void run() {
            String string;
            if(iocash>0){
                string="存款";
            }else{
                string="取款";
            }

            // 获取锁
            myLock.lock();
            // 执行现金业务
            System.out.println(name + "正在操作" + myCount + "账户," +
                    string+"金额为" + iocash + ",当前金额为" + myCount.getCash());
            myCount.setCash(myCount.getCash() + iocash);
            System.out.println(name + "操作" + myCount + "账户成功,"+
                    string + "金额为" + iocash + ",当前金额为" + myCount.getCash());
            System.out.println("============");
            // 释放锁,否则别的线程没有机会执行了
            myLock.unlock();
        }
    }

    /**
     * 信用卡账户,可随意透支
     */
    class MyCount {
        private String oid; // 账号
        private int cash; // 账户余额

        MyCount(String oid, int cash) {
            this.oid = oid;
            this.cash = cash;
        }

        public String getOid() {
            return oid;
        }

        public void setOid(String oid) {
            this.oid = oid;
        }

        public int getCash() {
            return cash;
        }

        public void setCash(int cash) {
            this.cash = cash;
        }

        @Override
        public String toString() {
            return "MyCount{" + "账号=‘" + oid + ‘\‘‘ + ", 余额=" + cash + ‘}‘;
        }
    }
}

张三正在操作MyCount{账号=‘95599200901215522‘, 余额=10000}账户,取款金额为-4000,当前金额为10000
张三操作MyCount{账号=‘95599200901215522‘, 余额=6000}账户成功,取款金额为-4000,当前金额为6000
============
张三他弟正在操作MyCount{账号=‘95599200901215522‘, 余额=6000}账户,取款金额为-8000,当前金额为6000
张三他弟操作MyCount{账号=‘95599200901215522‘, 余额=-2000}账户成功,取款金额为-8000,当前金额为-2000
============
张三他爹正在操作MyCount{账号=‘95599200901215522‘, 余额=-2000}账户,存款金额为6000,当前金额为-2000
张三他爹操作MyCount{账号=‘95599200901215522‘, 余额=4000}账户成功,存款金额为6000,当前金额为4000
============
张三正在操作MyCount{账号=‘95599200901215522‘, 余额=4000}账户,存款金额为800,当前金额为4000
张三操作MyCount{账号=‘95599200901215522‘, 余额=4800}账户成功,存款金额为800,当前金额为4800
============

从上面的输出可以看到,利用锁对象太方便了,比直接在某个不知情的对象上用锁清晰多了。

但一定要注意的是,在获取了锁对象后,用完后应该尽快释放锁,以便别的等待该锁的线程有机会去执行。

时间: 2024-10-07 17:34:56

java锁对象的相关文章

java多线程(二)锁对象

转载请注明出处:http://blog.csdn.net/xingjiarong/article/details/47679007 在上一篇博客中,我们讨论了Race Condition现象以及它产生的原因,现在我们知道它是不好的一种现象了,那么我们有什么方法避免它呢.最直接有效的方式就是放弃多线程,直接改为使用单线程但操作数据,但是这是不优雅的,因为我们知道有时候,多线程有它自己的优势.在这里我们讨论两种其他的方法--锁对象和条件对象. 锁对象 java SE5.0之后为实现多线程的互斥引入了

Java多线程的两种实现方式的区别以及深刻同步问题中的锁对象

首先我们知道创建线程有两种方式: 1.继承Thread类:2.实现Runnable接口. 但是这两这并非完全一样的.下面谈谈区别: 因为Java并不支持多继承的(接口是可以多继承接口的.不过一般我们不提),但支持多实现.当一个类继承了父类,就不能再继承Thread类,只能通过实现接口的形式创建线程. 继承Runnable接口更加符合面向对象的思想.线程分为两部分,一是线程对象,二是线程任务.继承Thread类,线程对象和线程任务(run方法内的代码)耦合在一起.一旦创建了Thread类的子类对象

【java】锁对象状态的改变会导致非线程安全

问题描述 使用synchronized在一个非final对象上加了锁之后,在synchronized体(同步代码块)中,将该对象的值(状态)改变之后,会导致线程不安全,即其他线程会拿到改变之后对象的锁,从而进入同步代码块. 场景设计 public class TestLock extends Thread{ private AAA aaa; public TestLock(AAA aaa) { this.aaa = aaa; } @Override public void run() { thi

Java 多线程, 同步访问, 线程锁,锁对象,ReentrantLock,synchronized

1.为什么要同步访问数据? 当两个或以上的线程需要共享对同一数据的存取,可能会发生共享数据的讹误. 2.实现同步的方式 2.1 ReentrantLock类 School类: class School{ private int stuNum; private Lock lock; private Condition condition; public School(int stuNum) { this.stuNum = stuNum; lock = new ReentrantLock(); co

java 锁!

问题:如何实现死锁. 关键: 1 两个线程ta.tb 2 两个对象a.b 3 ta拥有a的锁,同时在这个锁定的过程中,需要b的锁:tb拥有b的锁,同时在这个锁定的过程中,需要a的锁: 关键的实现难点是3, —— 所以说,死锁也不是那么容易出现的吧.. 实现方式synchronized.Lock 等等 死锁例子1  采用了不同类的两个对象. 原理是: 两个线程尝试进入同一个需要对象锁的方法 package basic.thread; public class DL { public static

java锁和同步

Java 语言设计中的一大创新就是:第一个把跨平台线程模型和锁模型应用到语言中去,Java 语言包括了跨线程的关键字synchronized 和 volatile,使用关键字和java类库就能够简单的实现线程间的同步.在简化与平台无关的并发程序开发时,它没有使并发程序的编写工作变得繁琐,反而使它变得更容易了. 在这一章,我们详细介绍锁的技术和概念,java中提供了两种锁,一个是使用关键字的锁,还有一种类库提供的锁. synchronized关键字锁 synchronized关键字能够作为函数的修

Java 锁

当多个线程同时访问同一部分代码进行数据的修改时,有可能会发生数据不安全(单线程除外),其中一种简单的方式可以通过Java自带的关键字解决:synchronized 同步方法 public synchronized void method_name(){ 代码块 } 同步代码块 synchronized(锁对象){ 代码块 }

java 锁2

并发,其实是多线程才有的场景... java 多线程? 锁? 现在看来,即使已经工作了4.5年,这仍然不是一个简单的问题. 其实java 本身有提供锁的机制. 比如 Object对象的 wait .notify 方法.synchronized 的原理不过是直接调用对应的对象的 wait方法罢了! 看tomcat 源码的时候,多线程的地方就是直接用到了 wait .notify等方法 —— 这些方法真高级, 一般哪里会用得着???!!! 1 synchronized 其实这个算是容易学的东西了,相

java锁Lock的应用

package jun.lock; public class BankCard { protected String cardId;//银行卡号 protected int balance = 8000;//余额 public String getCardId() { return cardId; } public void setCardId(String cardId) { this.cardId = cardId; } public int getBalance() { return ba