Lock可以更好的解决线程同步问题,使之更面向对象,并且ReadWriteLock在处理同步时更强大,那么同样,线程间仅仅互斥是不够的,还需要通信,本篇的内容是基于上篇之上,使用Lock如何处理线程通信。

那么引入本篇的主角,Condition,Condition 将 Object 监视器方法(wait、notify 和 notifyAll)分解成截然不同的对象,以便通过将这些对象与任意 Lock 实现组合使用,为每个对象提供多个等待 set (wait-set)。其中,Lock 替代了 synchronized 方法和语句的使用,Condition 替代了 Object 监视器方法的使用。下面将之前写过的一个线程通信的例子替换成用Condition实现(Java线程(三)),代码如下:

  1. public class ThreadTest2 {
  2. public static void main(String[] args) {
  3. final Business business = new Business();
  4. new Thread(new Runnable() {
  5. @Override
  6. public void run() {
  7. threadExecute(business, "sub");
  8. }
  9. }).start();
  10. threadExecute(business, "main");
  11. }
  12. public static void threadExecute(Business business, String threadType) {
  13. for(int i = 0; i < 100; i++) {
  14. try {
  15. if("main".equals(threadType)) {
  16. business.main(i);
  17. } else {
  18. business.sub(i);
  19. }
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. }
  23. }
  24. }
  25. }
  26. class Business {
  27. private boolean bool = true;
  28. private Lock lock = new ReentrantLock();
  29. private Condition condition = lock.newCondition();
  30. public /*synchronized*/ void main(int loop) throws InterruptedException {
  31. lock.lock();
  32. try {
  33. while(bool) {
  34. condition.await();//this.wait();
  35. }
  36. for(int i = 0; i < 100; i++) {
  37. System.out.println("main thread seq of " + i + ", loop of " + loop);
  38. }
  39. bool = true;
  40. condition.signal();//this.notify();
  41. } finally {
  42. lock.unlock();
  43. }
  44. }
  45. public /*synchronized*/ void sub(int loop) throws InterruptedException {
  46. lock.lock();
  47. try {
  48. while(!bool) {
  49. condition.await();//this.wait();
  50. }
  51. for(int i = 0; i < 10; i++) {
  52. System.out.println("sub thread seq of " + i + ", loop of " + loop);
  53. }
  54. bool = false;
  55. condition.signal();//this.notify();
  56. } finally {
  57. lock.unlock();
  58. }
  59. }
  60. }

在Condition中,用await()替换wait(),用signal()替换notify(),用signalAll()替换notifyAll(),传统线程的通信方式,Condition都可以实现,这里注意,Condition是被绑定到Lock上的,要创建一个Lock的Condition必须用newCondition()方法。

这样看来,Condition和传统的线程通信没什么区别,Condition的强大之处在于它可以为多个线程间建立不同的Condition,下面引入API中的一段代码,加以说明。

  1. class BoundedBuffer {
  2. final Lock lock = new ReentrantLock();//锁对象
  3. final Condition notFull  = lock.newCondition();//写线程条件
  4. final Condition notEmpty = lock.newCondition();//读线程条件
  5. final Object[] items = new Object[100];//缓存队列
  6. int putptr/*写索引*/, takeptr/*读索引*/, count/*队列中存在的数据个数*/;
  7. public void put(Object x) throws InterruptedException {
  8. lock.lock();
  9. try {
  10. while (count == items.length)//如果队列满了
  11. notFull.await();//阻塞写线程
  12. items[putptr] = x;//赋值
  13. if (++putptr == items.length) putptr = 0;//如果写索引写到队列的最后一个位置了,那么置为0
  14. ++count;//个数++
  15. notEmpty.signal();//唤醒读线程
  16. } finally {
  17. lock.unlock();
  18. }
  19. }
  20. public Object take() throws InterruptedException {
  21. lock.lock();
  22. try {
  23. while (count == 0)//如果队列为空
  24. notEmpty.await();//阻塞读线程
  25. Object x = items[takeptr];//取值
  26. if (++takeptr == items.length) takeptr = 0;//如果读索引读到队列的最后一个位置了,那么置为0
  27. --count;//个数--
  28. notFull.signal();//唤醒写线程
  29. return x;
  30. } finally {
  31. lock.unlock();
  32. }
  33. }
  34. }

这是一个处于多线程工作环境下的缓存区,缓存区提供了两个方法,put和take,put是存数据,take是取数据,内部有个缓存队列,具体变量和方法说明见代码,这个缓存区类实现的功能:有多个线程往里面存数据和从里面取数据,其缓存队列(先进先出后进后出)能缓存的最大数值是100,多个线程间是互斥的,当缓存队列中存储的值达到100时,将写线程阻塞,并唤醒读线程,当缓存队列中存储的值为0时,将读线程阻塞,并唤醒写线程,这也是ArrayBlockingQueue的内部实现。下面分析一下代码的执行过程:

1. 一个写线程执行,调用put方法;

2. 判断count是否为100,显然没有100;

3. 继续执行,存入值;

4. 判断当前写入的索引位置++后,是否和100相等,相等将写入索引值变为0,并将count+1;

5. 仅唤醒读线程阻塞队列中的一个;

6. 一个读线程执行,调用take方法;

7. ……

8. 仅唤醒写线程阻塞队列中的一个。

这就是多个Condition的强大之处,假设缓存队列中已经存满,那么阻塞的肯定是写线程,唤醒的肯定是读线程,相反,阻塞的肯定是读线程,唤醒的肯定是写线程,那么假设只有一个Condition会有什么效果呢,缓存队列中已经存满,这个Lock不知道唤醒的是读线程还是写线程了,如果唤醒的是读线程,皆大欢喜,如果唤醒的是写线程,那么线程刚被唤醒,又被阻塞了,这时又去唤醒,这样就浪费了很多时间。

示例:

package lock.demo11;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class ReadKafkaThread extends Thread { private int i = 0;
@Override
public void run() {
while(true) {
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
} execute();
} } public void execute() {
lock.lock();
try {
if(!readKafka) {
condition.await();//this.wait();
}
} catch(InterruptedException e) { } finally {
lock.unlock();
}
System.out.println("你好:" + i++);
} public volatile boolean readKafka = false;
private Lock lock = new ReentrantLock();
private Condition condition = lock.newCondition();
public void openReadKafka() {
lock.lock();
try {
readKafka = true;
condition.signal();//this.notify();
} finally {
lock.unlock();
} } public void closeReadKafka() {
readKafka = false;
} } package lock.demo11; import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.TimeUnit; public class Client { public static void main(String[] args) throws InterruptedException {
ReadKafkaThread rkt = new ReadKafkaThread();
rkt.start(); print("1 main sleep 1");
TimeUnit.SECONDS.sleep(1);
rkt.openReadKafka(); print("2 main sleep 2");
TimeUnit.SECONDS.sleep(2);
rkt.closeReadKafka(); print("3 main sleep 3");
TimeUnit.SECONDS.sleep(3);
rkt.openReadKafka(); print("4 main sleep 4");
TimeUnit.SECONDS.sleep(4);
rkt.closeReadKafka(); print("5 main sleep 5");
TimeUnit.SECONDS.sleep(5);
rkt.closeReadKafka(); print("6 main sleep 6");
TimeUnit.SECONDS.sleep(6);
rkt.openReadKafka();
print("7 main sleep 7");
TimeUnit.SECONDS.sleep(7);
rkt.closeReadKafka(); print("8 main sleep 8");
TimeUnit.SECONDS.sleep(8);
rkt.openReadKafka(); print("9 main sleep 9");
TimeUnit.SECONDS.sleep(9);
rkt.closeReadKafka(); } public static void print(String str) {
System.out.println(str + " 时间:" + new SimpleDateFormat("yyyy-MM-dd hh:mm:ss").format(new Date()));
}
}

结果:

1 main sleep 1 时间:2017-01-10 11:25:11
2 main sleep 2 时间:2017-01-10 11:25:12
你好:0
你好:1
3 main sleep 3 时间:2017-01-10 11:25:14
你好:2
4 main sleep 4 时间:2017-01-10 11:25:17
你好:3
你好:4
你好:5
你好:6
5 main sleep 5 时间:2017-01-10 11:25:21
6 main sleep 6 时间:2017-01-10 11:25:26
7 main sleep 7 时间:2017-01-10 11:25:32
你好:7
你好:8
你好:9
你好:10
你好:11
你好:12
你好:13
8 main sleep 8 时间:2017-01-10 11:25:39
你好:14
9 main sleep 9 时间:2017-01-10 11:25:47
你好:15
你好:16
你好:17
你好:18
你好:19
你好:20
你好:21
你好:22

Android实例] android获取web服务器端session并验证登陆的更多相关文章

  1. [Android实例] Android Studio插件-自动根据布局生成Activity等代码1.4 (开源)(申明:来源于网络)

    [Android实例] Android Studio插件-自动根据布局生成Activity等代码1.4 (开源)(申明:来源于网络) 地址:http://www.eoeandroid.com/thre ...

  2. [Android实例] Android之断点续传下载

    在我们做开发的时候经常遇到的就是下载了,现在下载的方法有很多很多,那么怎么做到断点续传下载呢!很多人都头疼这个问题,如果我们没有很好的逻辑真不是很容易解决啊.我参考了一下前辈们的资料了整理了一个项目, ...

  3. [Android实例] Android网络收音机项目(内含源码)

    ======================帖子内容===================================最近喜欢听广播,但是搜索了一下,苦于网上没有Android的网络收音机项目的例 ...

  4. [Android实例] Android 6.0RecyclerView SwipeRefreshLayout 下拉刷新 上拉加载

    这是Android 6.0的 SwipeRefreshLayout 实现下拉刷新和RecyclerView的上拉加载更多,以及添加分割线等 Android <ignore_js_op> r ...

  5. android客户端向服务器端验证登陆方法的实现2

    一.在上一篇文章中,我只是提到了其中一种方法来实现登陆 大家可以参见: http://www.apkbus.com/android-45004-1-1.html      android获取web服务 ...

  6. Android实例剖析笔记(四)

    摘要:分析NoteEditor这个类和以及Content Provider机制 NoteEditor深入分析 首先来弄清楚“日志编辑“的状态转换,通过上篇文章的方法来做下面这样一个实验,首先进入“日志 ...

  7. Android 实例子源代码文件下载地址380个合集

      android 城市列表特效 - 触摸查找源码 .rar: http://www.t00y.com/file/64337887 android 日记系统源码(数据库的基本操作) .rar: htt ...

  8. Android开发16——获取网络资源之基础应用

    一.项目背景在Android开发中有一项非常广泛的应用:Android项目获取另一个web项目的资源或者返回的数据.本博文介绍了获取另一个web项目的资源.有一个web项目,在其WebRoot文件夹下 ...

  9. 45个android实例源码

    分享45个android实例源码,很好很强大http://www.apkbus.com/android-20978-1-1.html andriod闹钟源代码http://www.apkbus.com ...

随机推荐

  1. 使用CSS3制作漂亮的按钮

    我给大家介绍一下如何使用CSS3来制作一个圆角阴影.渐变色的漂亮的按钮,它不需要任何图片和javascript脚本,只需要CSS3就可以轻松实现按钮效果,并且可以适用于任意HTML元素,想div,sp ...

  2. JAVA编译异常处理:java.lang.OutOfMemoryError: PermGen space

    在Intellij开发工具中编译JAVA项目,出现以下错误: 六月 21, 2016 6:28:07 下午 org.apache.tomcat.util.modeler.BaseModelMBean ...

  3. ubuntu sh脚本双击运行

    自从13.04以后,双击sh脚本文件就已经默认是geidt打开了,要想运行,从nautilus-->文件-->首选项-->行为-->可执行文件 有三个选项,默认是第二个,如果想 ...

  4. Linux远程文件传输

    linux系统中,难免会遇到一些要将某文件通过网络传送给其他主机的情况,而恰好两台主机 都是linux系统的时候,我们就可以直接使用scp命令来传输文件到另一台主机了. scp命令用于在网络中安全的传 ...

  5. ES6入门之Symbol

    ES5对象属性名都是字符串容易造成属性名的冲突. eg:var a = { name: 'lucy'}; a.name = 'lili';这样就会重写属性 ES6引入了一种新的原始数据类型Symbol ...

  6. 监听Activity进入后台(最小化),并根据时间判断是否超时,此解决办法可用于超时重登陆

    通过重写一个继承自Activity的基类中的onUserLeaveHint()方法,当用户按Home键等操作使程序进入后台时即开始计时,当用户使程序恢复为前台显示时执行onResume()方法,在其中 ...

  7. 深入理解Java内存模型(六)——final

    与前面介绍的锁和volatile相比较,对final域的读和写更像是普通的变量访问.对于final域,编译器和处理器要遵守两个重排序规则: 在构造函数内对一个final域的写入,与随后把这个被构造对象 ...

  8. Linq 和 EF Contains示例

    List<int> unitIDList=new List<int>(); //此处添加int元素 var query = DB.ElecConsumers.Where(c = ...

  9. 查看linux系统常用的命令,Linux查看系统配置常用命令

    一.linux CPU大小  cat /proc/cpuinfo |grep "model name" && cat /proc/cpuinfo |grep &qu ...

  10. C#高级编程(第9版) -C#5.0&.Net4.5.1 书上的示例代码下载链接

    http://www.wrox.com/WileyCDA/WroxTitle/Professional-C-5-0-and-NET-4-5-1.productCd-1118833031,descCd- ...