一. this 锁

同步函数其实用到的锁就是 this 锁,为什么他用到的是 this 锁呢?为了证实这个结论我 们本节将会有两个实验性的程序来作为支撑,说服自己和读者证明同步函数用到的就是 this 锁好了,请看下第一个程序

需求:

我们定义一个类,其中有两个方法,均加了同步锁,假设函数的同步不是 this 锁,我们 如果启动一个线程调用方法 A,另外一个线程调用用方法 B,A 方法和 B 方法里均是死循环, 按照同步函数不是 this 锁的逻辑,两个函数中的逻辑将会被同时执行,但是情况是否是这样, 我们需要通过代码进行实验

public class ClassA {
public synchronized void A()
{
System.out.println("AAAAAAAAAAAAAAAAA");
while (true)
{
} }
public synchronized void B()
{
System.out.println("BBBBBBBBBBBBBBBBB");
while (true)
{
}
}
}
public class MethodSynchronizedTest {

    public static void main(String[] args) {
final ClassA clazz = new ClassA();
// 启动一个线程
new Thread(new Runnable() {
public void run() {
clazz.A();// 调用A方法
}
}).start(); // 启动另一个线程
new Thread(new Runnable() {
public void run() {
clazz.B();// 调用B方法
}
}).start(); }
}

分别启动了两个线程,分别用来执行 ClassA 中的两个方法 A 和 B,两个方法都是加了锁 的,也就是说某个线程尽到方法 A 中其他线程就不能进入 A,但是另一个线程应该能进入 B, 但是我们等了半天方法 B 仍然没有输出,因此我们得出一个结论,他们的锁是同一个,至于 是哪一个锁呢?答案就是 this 锁;但是这个例子似乎还不过瘾,那么我们继续修正叫号的程 序,让读者有一个更加直观的理解;

需求: 我们增加一个方法,也用来进行叫号,在 run 方法中也进行叫号,第一个线程调用 run 方法中的逻辑,第二个方法调用函数中的逻辑;

public class TicketWindow3 implements Runnable {

    private int max_value = 0;

    private Object lock = new Object();

    private boolean flag = true;

    @Override
public void run() {
if (flag) {
while (true) {
//同步函数其实用到的锁就是 this 锁, this锁针对的是对象
//静态锁,锁是类的字节码信息,因此如果一个类的函数为静态方法,那么我们需要通过该类的 class 信息进行加锁;
synchronized (lock) {
if (max_value > 500) {
break;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println(Thread.currentThread().getName() + ":lock..." + max_value++);
}
}
} else {
while (true)
if (ticket())
break;
} } private synchronized boolean ticket() {
if (max_value > 500) {
return true;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println(Thread.currentThread().getName() + ": method.." + max_value++);
return false;
} public void change() throws InterruptedException {
Thread.sleep(30);// 读者可以自行思考为什么要sleep
this.flag = false;
} }
public class Bank3 {

    public static void main(String[] args) throws InterruptedException {
TicketWindow3 tw3 = new TicketWindow3();
Thread t1 = new Thread(tw3);
Thread t2 = new Thread(tw3);
t1.start();
tw3.change();
t2.start();
}
}

文字说明: 可能到输出性信息,其中会有 501 这样的信息输出,为什么会这样呢?因为上述的代码 两处业务逻辑同步锁是两把锁,如果您将 lock 换成 this,这个现象就不会出现,读者可 以自己进行测试;

2 static 锁

如果我们的方法 ticket 是一个静态方法,再次测试一下您会发现,还是会出现 501 这 样的输出信息,根据之前的描述读者可能会第一时间想到他们两个用到的锁不是同一把锁, 因此我们将代码在次做了修改

public class TicketWindow3 implements Runnable {

    private static int max_value = 0;

    private boolean flag = true;

    @Override
public void run() {
if (flag) {
while (true) {
//同步函数其实用到的锁就是 this 锁, this锁针对的是对象
//静态锁,锁是类的字节码信息,因此如果一个类的函数为静态方法,那么我们需要通过该类的 class 信息进行加锁;
synchronized (TicketWindow3.class) {
if (max_value > 500) {
break;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println(Thread.currentThread().getName() + ":lock..." + max_value++);
}
}
} else {
while (true)
if (ticket())
break;
} } private synchronized static boolean ticket() {
if (max_value > 500) {
return true;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
}
System.out.println(Thread.currentThread().getName() + ": method.." + max_value++);
return false;
} public void change() throws InterruptedException {
Thread.sleep(30);// 读者可以自行思考为什么要sleep
this.flag = false;
} }

静态锁,锁是类的字节码信息,因此如果一个类的函数为静态方法,那么我们需要通过 该类的 class 信息进行加锁;

this 锁与 static 锁的更多相关文章

  1. 类锁和对象锁,synchronized修饰static方法与非static方法的区别

    当synchronized修饰一个static方法时,多线程下,获取的是类锁(即Class本身,注意:不是实例), 作用范围是整个静态方法,作用的对象是这个类的所有对象. 当synchronized修 ...

  2. 可重入锁 公平锁 读写锁、CLH队列、CLH队列锁、自旋锁、排队自旋锁、MCS锁、CLH锁

    1.可重入锁 如果锁具备可重入性,则称作为可重入锁. ========================================== (转)可重入和不可重入 2011-10-04 21:38 这 ...

  3. Java核心知识点学习----线程中如何创建锁和使用锁 Lock,设计一个缓存系统

    理论知识很枯燥,但这些都是基本功,学完可能会忘,但等用的时候,会发觉之前的学习是非常有意义的,学习线程就是这样子的. 1.如何创建锁? Lock lock = new ReentrantLock(); ...

  4. Java锁之自旋锁详解

    锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchronized 和 ReentrantLock等等 ) .这些已经写好提供的锁为我们开发提供了便利,但是锁的具体性质以及类 ...

  5. java synchronized类锁,对象锁详解(转载)

    觉得还不错 留个记录,转载自http://zhh9106.iteye.com/blog/2151791 在java编程中,经常需要用到同步,而用得最多的也许是synchronized关键字了,下面看看 ...

  6. mysql的锁--行锁,表锁,乐观锁,悲观锁

    一 引言--为什么mysql提供了锁 最近看到了mysql有行锁和表锁两个概念,越想越疑惑.为什么mysql要提供锁机制,而且这种机制不是一个摆设,还有很多人在用.在现代数据库里几乎有事务机制,aci ...

  7. Hibernate、乐观锁和悲观锁

    悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁.传统的关系型数据 ...

  8. Java类锁和对象锁实践(good)

    一.前言 之前对类锁和对象锁是否是互斥的不是太确定,因此决定编写相关的程序进行实践一下.编写前对相关定义约定约定如下: 1. 类锁:在代码中的方法上加了static和synchronized的锁,或者 ...

  9. 【试验局】ReentrantLock中非公平锁与公平锁的性能测试

    硬件环境: CPU:AMD Phenom(tm) II X4 955 Processor Memory:8G SSD(128G):/ HDD(1T):/home/ 软件环境: OS:Ubuntu14. ...

随机推荐

  1. 使用JS与CSS3的翻转实现3D翻牌效果

    之前我们有讨论过使用CSS3如何实现网页水平翻转的效果,而这次我们介绍的是翻转效果更深一层的应用——3D翻牌效果. 这里我们需要使用flip中轴翻转实现,又因为是3D效果,如果希望呈现一定的3D视角, ...

  2. vs 2013 编译cocos2d-x-3.9

    下载地址:链接: https://pan.baidu.com/s/1IkQsMU6NoERAAQLcCUMcXQ 提取码: p1pb 下载完成后,解压 进入build 目录 使用vs 2013 打开工 ...

  3. [原][osg][osgearth]倾斜摄影1.介绍

    总体介绍: 倾斜摄影就是将拍好的数据,三角网格化再附上贴图. 目前流行处理软件: Street Factory.PIX4DMapper.smart3D 后期开发平台:超图 Skyline smart3 ...

  4. vuex中的辅助函数 mapState,mapGetters, mapActions, mapMutations

    1.导入辅助函数 导入mapState可以调用vuex中state的数据 导入mapMutations可以调用vuex中mutations的方法 四个辅助函数 各自对应自己在vuex上的自己 2.ma ...

  5. Android接入微信SDK之一:发起微信授权登录

    1.重要的事情首先说! 包名.应用签名.app id 三者都必须和在腾讯上申请的一致!!!否则将不能成功. 包名:就是在腾讯上申请的包名 应用签名:使用微信官网提供的<签名生成工具>(这个 ...

  6. conda-使用手册

    conda remove -n tf --all ##删除环境 conda env export -- name ##首先导出配置文件: conda env create -f name.yml ## ...

  7. Codeforces 1006 F - Xor-Paths

    F - Xor-Path 思路: 双向搜索dfs 如果普通的搜索复杂度是n 那么双向搜索复杂度是√n 代码: #include<bits/stdc++.h> using namespace ...

  8. 记录python接口自动化测试--把测试结果写进excel文件(第九目)

    python中一般使用xlrd(excel read)来读取Excel文件,使用xlwt(excel write)来生成Excel文件(可以控制Excel中单元格的格式),需要注意的是,用xlrd读取 ...

  9. 通过 rufus 创建启动U盘,安装 VMWare Esxi

    现在谁还用光盘安装系统啊. 做出启动盘后,U盘启动进行安装才是王道. https://www.starwindsoftware.com/blog/create-an-esxi-6-5-installa ...

  10. Spring之Spel表达式

    正常业务场景一般不用这个技术,但需要知道有这么个东西支持Spring. 记忆力不好,抄了些套路代码便于以后用到. package com.paic.phssp.springtest.spel; imp ...