一、synchronized

基本概念:对象锁synchronized(object){….}用法
在以上的代码块中只能由一个线程执行!!!
wait()、notify()是用在这个代码块当中的。wait()可以使当前线程A马上失去对象锁并且沉睡,直到对象调用notify()唤醒该线程。此时持有对象锁的线程B会先行执行完毕,然后再将对象锁交给线程A继续执行。

public class Person {
private String name;
private int age;public Person(String name, int age) {this.name = name;this.age = age;
}public String getName() {return name;
}public void setName(String name) {this.name = name;
}public int getAge() {return age;
}public void setAge(int age) {this.age = age;
}

下面我们开始对Person进行访问,我们想这样一个情景。有两个子线程AB,在A执行到一定条件下,让A暂停,开始执行B,B全部执行完毕,A再继续。。我们可以用一下代码来加以实现

final Person person = new Person("张三",23);Thread t1 = new Thread(new Runnable() {@Overridepublic void run() {synchronized (person){try {for (int i = 0 ; i<10 ; i++) {Thread.sleep(1000);Log.e(person.getName(),"我是A"+i);if (i==5){person.wait();// 此处将A 暂停}}} catch (InterruptedException e) {e.printStackTrace();}}}},"Thread1");Thread t2 = new Thread(new Runnable() {@Overridepublic void run() {synchronized (person){try {for ( int i = 0 ; i < 10 ; i++ ){Thread.sleep(1000);Log.e(person.getName(),"我是B--"+i);if (i==5){person.notify();//  此处重新激活A}}} catch (InterruptedException e) {e.printStackTrace();}}}},"Thread2");t1.start();t2.start();

线程A和线程B是并发执行的。线程A每隔1s打印一条日志。当循环到第6次的时候 wait()放弃对象锁并沉睡。此时线程B获得对象锁开始执行。执行完全部10次循环。notify()唤醒线程A。

01-13 15:45:18.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A0
01-13 15:45:19.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A1
01-13 15:45:20.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A2
01-13 15:45:21.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A3
01-13 15:45:22.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A4
01-13 15:45:23.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A5
01-13 15:45:24.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--0
01-13 15:45:25.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--1
01-13 15:45:26.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--2
01-13 15:45:27.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--3
01-13 15:45:28.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--4
01-13 15:45:29.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--5
01-13 15:45:30.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--6
01-13 15:45:31.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--7
01-13 15:45:32.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--8
01-13 15:45:33.335 15435-15907/com.huaxinzhi.usingthreadpool E/张三: 我是B--9
01-13 15:45:34.335 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A6
01-13 15:45:35.345 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A7
01-13 15:45:36.345 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A8
01-13 15:45:37.345 15435-15906/com.huaxinzhi.usingthreadpool E/张三: 我是A9

二、lock

Lock是java.util.concurrent.locks包下的接口,Lock 实现提供了比使用synchronized 方法和语句可获得的更广泛的锁定操作,因为Lock可以锁定任意一段代码:

public class LockTest {
public static void main(String[] args) {  final Outputter1 output = new Outputter1();  new Thread() {  public void run() {  output.output("zhangsan");  };  }.start();        new Thread() {  public void run() {  output.output("lisi");  };  }.start();  }
}  class Outputter1 {
private Lock lock = new ReentrantLock();// 锁对象
public void output(String name) {  // TODO 线程输出方法  lock.lock();// 得到锁  try {  for(int i = 0; i < name.length(); i++) {  System.out.print(name.charAt(i));  }  } finally {  lock.unlock();// 释放锁  }
}
}

这样就实现了和sychronized一样的同步效果,需要注意的是,用sychronized修饰的方法或者语句块在代码执行完之后锁自动释放,而用Lock需要我们手动释放锁,所以为了保证锁最终被释放(发生异常情况),要把互斥区放在try内,释放锁放在finally内。

三、ReadWriteLock

如果说这就是Lock,那么它不能成为同步问题更完美的处理方式,下面要介绍的是读写锁(ReadWriteLock),我们会有一种需求,在对数据进行读写的时候,为了保证数据的一致性和完整性,需要读和写是互斥的,写和写是互斥的,但是读和读是不需要互斥的,这样读和读不互斥性能更高些,来看一下不考虑互斥情况的代码原型:

public class ReadWriteLockTest {
public static void main(String[] args) {  final Data data = new Data();  for (int i = 0; i < 3; i++) {  new Thread(new Runnable() {  public void run() {  for (int j = 0; j < 5; j++) {  data.set(new Random().nextInt(30));  }  }  }).start();  }         for (int i = 0; i < 3; i++) {  new Thread(new Runnable() {  public void run() {  for (int j = 0; j < 5; j++) {  data.get();  }  }  }).start();  }
}
}  class Data {
private int data;// 共享数据
public void set(int data) {  System.out.println(Thread.currentThread().getName() + "准备写入数据");  try {  Thread.sleep(20);  } catch (InterruptedException e) {  e.printStackTrace();  }  this.data = data;  System.out.println(Thread.currentThread().getName() + "写入" + this.data);
}
public void get() {  System.out.println(Thread.currentThread().getName() + "准备读取数据");  try {  Thread.sleep(20);  } catch (InterruptedException e) {  e.printStackTrace();  }  System.out.println(Thread.currentThread().getName() + "读取" + this.data);
}
}

部分输出结果:

Thread-1准备写入数据
Thread-3准备读取数据
Thread-2准备写入数据
Thread-0准备写入数据
Thread-4准备读取数据
Thread-5准备读取数据
Thread-2写入12
Thread-4读取12
Thread-5读取5
Thread-1写入12

我们要实现写入和写入互斥,读取和写入互斥,读取和读取互斥,在set和get方法加入sychronized修饰符:

public synchronized void set(int data) {...}
public synchronized void get() {...}

部分输出结果:

Thread-0准备写入数据
Thread-0写入9
Thread-5准备读取数据
Thread-5读取9
Thread-5准备读取数据
Thread-5读取9
Thread-5准备读取数据
Thread-5读取9
Thread-5准备读取数据
Thread-5读取9

我们发现,虽然写入和写入互斥了,读取和写入也互斥了,但是读取和读取之间也互斥了,不能并发执行,效率较低,用读写锁实现代码如下:

class Data {
private int data;// 共享数据
private ReadWriteLock rwl = new ReentrantReadWriteLock();
public void set(int data) {  rwl.writeLock().lock();// 取到写锁  try {  System.out.println(Thread.currentThread().getName() + "准备写入数据");  try {  Thread.sleep(20);  } catch (InterruptedException e) {  e.printStackTrace();  }  this.data = data;  System.out.println(Thread.currentThread().getName() + "写入" + this.data);  } finally {  rwl.writeLock().unlock();// 释放写锁  }
}
public void get() {  rwl.readLock().lock();// 取到读锁  try {  System.out.println(Thread.currentThread().getName() + "准备读取数据");  try {  Thread.sleep(20);  } catch (InterruptedException e) {  e.printStackTrace();  }  System.out.println(Thread.currentThread().getName() + "读取" + this.data);  } finally {  rwl.readLock().unlock();// 释放读锁  }
}
}

部分输出结果:

Thread-4准备读取数据
Thread-3准备读取数据
Thread-5准备读取数据
Thread-5读取18
Thread-4读取18
Thread-3读取18
Thread-2准备写入数据
Thread-2写入6
Thread-2准备写入数据
Thread-2写入10
Thread-1准备写入数据
Thread-1写入22

这样,写入和写入互斥,读取和写入互斥,但是读取和读取不互斥,提高了代码效率。

Android中三种锁的基本实现相关推荐

  1. Android 中三种启用线程的方法

    多线程就是分时利用CPU,宏观上让所有线程一起执行 ,也叫并发 先调用start后调用run 1.start()方法来启动线程,真正实现了多线程运行. 这时无需等待run方法体代码执行完毕,可以直接继 ...

  2. Android中三种超实用的滑屏方式汇总(ViewPager、ViewFlipper、ViewFlow)

    现如今主流的Android应用中,都少不了左右滑动滚屏这项功能,(貌似现在好多人使用智能机都习惯性的有事没事的左右滑屏,也不知道在干什么...嘿嘿),由于前段时间项目的需要,所以也对其研究了一下,总的 ...

  3. 【移动开发】Android中三种超实用的滑屏方式汇总(ViewPager、ViewFlipper、ViewFlow)...

    现如今主流的Android应用中,都少不了左右滑动滚屏这项功能,(貌似现在好多人使用智能机都习惯性的有事没事的左右滑屏,也不知道在干什么...嘿嘿),由于前段时间项目的需要,所以也对其研究了一下,总的 ...

  4. Java 中15种锁的介绍:公平锁,可重入锁,独享锁,互斥锁,乐观锁,分段锁,自旋锁等等...

    http://blog.51cto.com/13919357/2339446 Java 中15种锁的介绍 在读很多并发文章中,会提及各种各样锁如公平锁,乐观锁等等,这篇文章介绍各种锁的分类.介绍的内容 ...

  5. java B锁_Java中15种锁的介绍

    原标题:Java中15种锁的介绍 在读很多并发文章中,会提及各种各样锁如公平锁,乐观锁等等,这篇文章介绍各种锁的分类.介绍的内容如下: 1.公平锁 / 非公平锁 2.可重入锁 / 不可重入锁 3.独享 ...

  6. android中几种定位方式详解

    目录 前言: 1.GPS定位 2.NETWORK定位 3.AGPS定位 4.基站定位 5.WIFI定位 6.混合定位 目前,移动端大致通过三种方式来进行设备定位:GPS.基站.wifi.本文就详细的讲 ...

  7. Java 中15种锁的介绍:公平锁,可重入锁,独享锁,互斥锁,乐观锁,分段锁,自旋锁等等

    Java 中15种锁的介绍 在读很多并发文章中,会提及各种各样锁如公平锁,乐观锁等等,这篇文章介绍各种锁的分类.介绍的内容如下: 公平锁 / 非公平锁 可重入锁 / 不可重入锁 独享锁 / 共享锁 互 ...

  8. Android的三种网络联接方式(URL / HttpURLConnection | HttpClient | InetAddress )

    Android SDK 中一些与网络有关的package,如下: 包描述 java.net 提供与联网有关的类,包括流和数据包(datagram)sockets.Internet 协议和常见 HTTP ...

  9. java简述常见的布局极其特点_请简要说明 Android 中五种常见布局的特点。_学小易找答案...

    [简答题]请简要说明有序广播和无序广播的区别 [简答题]请简要说明 Android 程序结构中 AndroidManifest.xml 的功能? [简答题]简述李村站人工办理进路的作业过程. [简答题 ...

最新文章

  1. 查询提升200倍,ClickHouse你值得拥有!
  2. 写Java程序要体现面向对象
  3. tensorflow 动态数组 TensorArray
  4. Runloop循环机制
  5. 只要一点点力气就可以撬起重物?
  6. 上周热点回顾(7.8-7.14)
  7. 【英语学习】【WOTD】ephemeral 释义/词源/示例
  8. 浏览器缓存机制(优化)
  9. python从入门到精通 明日科技 电子书-Python从入门到精通(明日科技出版) 源代码+课件+视频 全套...
  10. memcached php封装类,PHP Memcached + APC + 文件缓存封装_PHP - key
  11. Cookie的过期时间设置
  12. srs2.0安装问题
  13. 您的组织策略阻止我们为您完成此操作,有关详细信息,请联系技术支持
  14. 谷歌翻译(英文PDF文档翻译成中文,免费无限制)
  15. 中国传统节日简介、由来、习俗
  16. 记录百度点击登录无响应及百度搜索显示异常
  17. 最近超火的100句土味情话合集,拿去撩妹撩汉吧!
  18. 电脑端10大图片处理类神器
  19. XSD(XML Schema Definition)用法实例介绍以及C#使用xsd文件验证XML格式
  20. 87金融汇:如何理性看待互联网理财收益波动

热门文章

  1. Maven升级3.8.1后maven-default-http-blocker (http://0.0.0.0/): Blocked mirror for repositories
  2. 应对大厂面试,我是如何学会React Hooks的
  3. python3画直方图出现“Polygon‘ object has no property ‘normed”
  4. Vue核心之MVVM模型
  5. 计算机毕业设计JAVA某市教育局综合信息管理平台mybatis+源码+调试部署+系统+数据库+lw
  6. JS前端取得并解析后台服务器返回的JSON数据的方法
  7. 高德全链路压测平台TestPG的架构与实践
  8. RPA 机器人有哪些优势?
  9. 计算机心理部的活动记录表,《计算机系12月心理活动策划书【荐】.doc》-支持高清全文免费浏览-max文档...
  10. Android几行代码实现监听微信聊天