JAVA多线程-Lock入门

synchronized是java中的一个关键字,也就是说是Java语言内置的特性。那么为什么会出现Lock呢?

  在前面文章中,我们了解到如果一个代码块被synchronized修饰了,当一个线程获取了对应的锁,并执行该代码块时,其他线程便只能一直等待,等待获取锁的线程释放锁,而这里获取锁的线程释放锁只会有两种情况:

  1)获取锁的线程执行完了该代码块,然后线程释放对锁的占有;

  2)线程执行发生异常,此时JVM会让线程自动释放锁。

  那么如果这个获取锁的线程由于要等待IO或者其他原因(比如调用sleep方法)被阻塞了,但是又没有释放锁,其他线程便只能干巴巴地等待,试想一下,这多么影响程序执行效率。

  因此就需要有一种机制可以不让等待的线程一直无期限地等待下去(比如只等待一定的时间或者能够响应中断),通过Lock就可以办到。

  再举个例子:当有多个线程读写文件时,读操作和写操作会发生冲突现象,写操作和写操作会发生冲突现象,但是读操作和读操作不会发生冲突现象。

  但是采用synchronized关键字来实现同步的话,就会导致一个问题:

  如果多个线程都只是进行读操作,所以当一个线程在进行读操作时,其他线程只能等待无法进行读操作。

  因此就需要一种机制来使得多个线程都只是进行读操作时,线程之间不会发生冲突,通过Lock就可以办到。

  另外,通过Lock可以知道线程有没有成功获取到锁。这个是synchronized无法办到的。

  总结一下,也就是说Lock提供了比synchronized更多的功能。但是要注意以下几点:

  1)Lock不是Java语言内置的,synchronized是Java语言的关键字,因此是内置特性。Lock是一个类,通过这个类可以实现同步访问;

       2)Lock和synchronized有一点非常大的不同,采用synchronized不需要用户去手动释放锁,当synchronized方法或者synchronized代码块执行完之后,系统会自动让线程释放对锁的占用;而Lock则必须要用户去手动释放锁,如果没有主动释放锁,就有可能导致出现死锁现象。

我们先看下lock的简单用法:

先看下不加lock的账户代码:

public class Account {

    public   void add(BigDecimal amount , String name) {

        System.out.println(" 开始充值,线程名: " + Thread.currentThread().getName());

        try {

            Thread.sleep(2000);

            System.out.println(" 结束充值,线程名: " + Thread.currentThread().getName());

        } catch (InterruptedException e) {

            e.printStackTrace();

        }finally {


        }

    }

}

看下充值代码:

package com.ck.thread;



import java.math.BigDecimal;



public class CzThread extends Thread{


    private Account account;

    private String accountName;
  

    public CzThread(Account account) {

        this.account = account;

    }  

   

    @Override

    public void run() {

        account.add(new BigDecimal("100"), accountName);

    }


    public Account getAccount() {

        return account;

    }


    public void setAccount(Account account) {

        this.account = account;

    }


    public String getAccountName() {

        return accountName;

    }


    public void setAccountName(String accountName) {

        this.accountName = accountName;

    }
   

}

 

再看下测试类:

package com.ck.thread;





public class MainThread {



    public static void main(String[] args) throws InterruptedException {

        Account account = new Account();

       

        CzThread cz1 = new CzThread(account);

        cz1.setName("thread1");

        cz1.start();

       

        CzThread cz2 = new CzThread(account);

        cz2.setName("thread2");

        cz2.start();

    }

}

运行结果:

开始充值,线程名: thread2

 开始充值,线程名: thread1

 结束充值,线程名: thread1

 结束充值,线程名: thread2

 

我们通过结果可以看出来,账户类是不安全的,然后我们修改下账户类:

package com.ck.thread;



import java.math.BigDecimal;

import java.util.concurrent.locks.Lock;

import java.util.concurrent.locks.ReentrantLock;



public class Account {



private Lock lock = new ReentrantLock();



public   void add(BigDecimal amount , String name) {

    lock.lock();

    System.out.println(" 开始充值,线程名: " + Thread.currentThread().getName());

    try {

        Thread.sleep(2000);

        System.out.println(" 结束充值,线程名: " + Thread.currentThread().getName());

    } catch (InterruptedException e) {

        e.printStackTrace();

    }finally {

        lock.unlock();

    }

}



}

我们加上了lock锁,然后看下运行结果:

开始充值,线程名: thread1

 结束充值,线程名: thread1

 开始充值,线程名: thread2

 结束充值,线程名: thread2

我们看到lock的第一个作用与synchronized一样,具有保护线程安全的作用。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值