警告⚠️:本文耗时很长,先做好心理准备
证明:偏向锁、轻量级锁、重量级锁真实存在

由【java并发笔记之java线程模型】链接: https://www.cnblogs.com/yuhangwang/p/11256476.html这篇文章可知:每当java线程创建的时候相对应的os pthread_create()也会创建一个线程,使用synchronized()就必然调用os pthread_mutex_lock() 函数

synchronized关键字锁的状态:无锁、偏向锁、轻量级锁、重量级锁
此篇文章由证明偏向锁是否存在入手,众所周知偏向锁一定会保证线程安全 ,但是实际情况不一定有互斥,偏向锁是synchronized锁的对象没有资源竞争的情况下,不会调用os pthread_mutex_lock() 函数; 但是第一次初始化使用锁的时候确实会调用一次pthread_mutex_lock()进行偏向锁
猜想:偏向锁一定真实存在
求证方法:
  1.修改Linux源码中glibc库中pthread_mutex_lock.c文件中的pthread_mutex_lock() 方法,增加输出当前os id 语句;
  2.java代码中使用synchronized关键字加锁,打印出加锁前线程id(此线程id会转化为真实os 线程Id),1和2两者相互比较;
  3.如果调用os pthread_mutex_lock() os-id 与 java thread-id 相同: 说明锁真的存在, 并且只出现过一次相同为偏向锁
开始求证:
环境搭建:(此处注意linux内核必须与gilbc库相对应,否则编译不成功)
linux:centos 7
Gilbc:Gilbc-2.19 官方gilbc链接http://mirror.hust.edu.cn/gnu/glibc/
避免踩坑,个人环境已经上传百度网盘:请自行下载安装
百度网盘地址:链接:https://pan.baidu.com/s/1LULbCoxm-ooPnZbGG3tdAQ 密码:9jwu

检查环境:
一、首先检查自己的linux有没有java环境:
    通过java、和javac两个命令查看是否可用
    如果没有则检查一下yum当中有哪些JDK可以提供安装命令:
yum search java | grep -i --color jdk

 如果没有安装,  以jdk8为例:

yum install -y java-1.8.0-openjdk.x86_64 java-1.8.0-openjdk-devel.x86_64

二、在检查inux环境要与gIibc版本是否相对应(不对应会导致编译失败)
三、检查有没有安装gcc,用来编译C程序,没有就安装gcc
yum -y install gcc

四、解压glibc

tar -zxvf glibc-2.19.tar.gz

五、修改glibc的源码

修改:pthread_mutex_lock()该方法,加入打印当前os threadId(也就是调用该方法时打印)
路径:{your-dir}/glibc-2.19/nptl/pthread_mutex_lock.c---pthread_mutex_lock()
   pthread_mutex_lock.c文件下 pthread_mutex_lock() 函数下 第66行修改为:
printf(stderr,”msg tid=%lu\n",pthread_self);
  (stderr:文件描述符,pthread_self:当前线程id)

修改完之后,以后所有调用pthread_mutex_lock()函数都会打印出自己的线程id
六、编译刚才修改完的文件:
cd glibc-2.19

  编译完成后要存放的文件位置:

mkdir out

   编译(完成后覆盖系统默认的文件,请提前备份默认文件,以防不测)

cd out
../configure --prefix=/usr --disable-profile --enable-add-ons --with-headers=/usr/include --with-binutils=/usr/bin

  执行:

make
//make 执行完毕之后再执行
make install

   测试 运行:

java

 只有调用了pthread_mutex_lock()会打印出自己的线程id:

  

表示此处修改linux源码glibc库下的pthread_mutex_lock()成功;
接下来就该验证偏向锁到底是不是真实存在的:
java代码:
import java.lang.Thread;

public class ThreadTest {
Object o= new Object();
static {
System.loadLibrary( "TestThreadNative" );
}
public static void main(String[] args) {
//打印出主线程
System.out.println("xxxxxxxxxxxxxxxxxxxxxxxxxxxx");
ThreadTest example4Start = new ThreadTest();
example4Start.start();
}
public void start(){
Thread thread = new Thread(){
@Override
public void run() {
while (true){
try {
sync();
} catch (InterruptedException e) { }
}
}
}; Thread thread2 = new Thread(){
@Override
public void run() {
while (true){
try {
sync();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}; thread.setName("t1");
thread2.setName("t2"); thread.start();
} //.c 文件打印出java threaid 对应的os threadid
public native void tid(); public void sync() throws InterruptedException {
synchronized(o) {
//java threadid 是jvm给的线程id 并不是真是的os 对应的线程id
//System.out.println(Thread.currentThread().getId()); //获取java thread 对应的真实的os thread 打印出id
tid();
}
}
}

获取真实线程id(os 线程Id)的c文件:(对应上边java代码的 native void tid())

#include<pthread.h>
#include<stdio.h>
#include<stdlib.h>
#include "Example4Start.h"
JNIEXPORT void JNICALL Java_Example4Start_tid(JNIEnv *env, jobject c1){
printf("current tid:%lu-----\n",pthread_self());
usleep(700);
}

然后再走一遍生成.class、.h 、so 然后执行(此方法在【java并发笔记之java线程模型】链接: https://www.cnblogs.com/yuhangwang/p/11256476.html中有相对应执行方法,请参考)
执行:
java ThreadTest
显示: 

  

  

实锤:红色的证明偏向锁真实存在,绿色的证明synchronized 1.6之后偏向锁做了优化(只调用了一次os 函数,后面没有调用)
同理轻量级锁及重量级锁也可以这样证明出来 

原创不易,转载请标明出处,谢谢

java并发笔记之证明 synchronized锁 是否真实存在的更多相关文章

  1. java并发笔记之四synchronized 锁的膨胀过程(锁的升级过程)深入剖析

    警告⚠️:本文耗时很长,先做好心理准备,建议PC端浏览器浏览效果更佳. 本篇我们讲通过大量实例代码及hotspot源码分析偏向锁(批量重偏向.批量撤销).轻量级锁.重量级锁及锁的膨胀过程(也就是锁的升 ...

  2. Java并发编程实战 03互斥锁 解决原子性问题

    文章系列 Java并发编程实战 01并发编程的Bug源头 Java并发编程实战 02Java如何解决可见性和有序性问题 摘要 在上一篇文章02Java如何解决可见性和有序性问题当中,我们解决了可见性和 ...

  3. 【Java并发系列04】线程锁synchronized和Lock和volatile和Condition

    img { border: solid 1px } 一.前言 多线程怎么防止竞争资源,即防止对同一资源进行并发操作,那就是使用加锁机制.这是Java并发编程中必须要理解的一个知识点.其实使用起来还是比 ...

  4. java并发编程:深入了解synchronized

    简介 synchronized是Java语言的关键字,可用来给对象和方法或者代码块加锁,当它锁定一个方法或者一个代码块的时候,同一时刻最多只有一个线程执行这段代码.同时它还保证了共享变量的内存可见性. ...

  5. 【Java并发编程实战】-----synchronized

    在我们的实际应用当中可能经常会遇到这样一个场景:多个线程读或者.写相同的数据,访问相同的文件等等.对于这种情况如果我们不加以控制,是非常容易导致错误的.在java中,为了解决这个问题,引入临界区概念. ...

  6. Java 并发编程——volatile与synchronized

    一.Java并发基础 多线程的优点 资源利用率更好 程序设计在某些情况下更简单 程序响应更快 这一点可能对于做客户端开发的更加清楚,一般的UI操作都需要开启一个子线程去完成某个任务,否者会容易导致客户 ...

  7. Java并发(4)- synchronized与CAS

    引言 上一篇文章中我们说过,volatile通过lock指令保证了可见性.有序性以及"部分"原子性.但在大部分并发问题中,都需要保证操作的原子性,volatile并不具有该功能,这 ...

  8. Java并发笔记(二)

    1. 活跃性危险 死锁(最常见) 饥饿 当线程由于无法访问它所需的资源而不能继续执行时,就发生了饥饿.引发饥饿最常见资源就是CPU时钟周期. 活锁 活锁指的是任务或者执行者没有被阻塞,由于某些条件没有 ...

  9. 【Java并发编程实战】—–synchronized

    在我们的实际应用其中可能常常会遇到这样一个场景:多个线程读或者.写相同的数据,訪问相同的文件等等.对于这样的情况假设我们不加以控制,是非常easy导致错误的. 在java中,为了解决问题,引入临界区概 ...

随机推荐

  1. 设计模式之单例模式的几种写法——java

    对于设计模式的使用场景和好处,之前有介绍一篇,今天主要是单例模式的编写方式,直接看代码吧 单例模式之饿汉模式,不会懒加载.线程安全 /** * @Author wangtao * @Descripti ...

  2. vue-resource.js的get和post的正确用法

    在网上看到人家写的vue-resource.js的get方法例子, new Vue({ el:'body', data:{ }, methods:{ get:function(){ this.$htt ...

  3. 你所忽略的,覆盖equals时需要注意的事项《effective java》

    我们都知道Object的equals的比较其实就是==的比较,其实是内存中的存放地址的比较.正常逻辑上:类的每个实例本质上都是唯一的. 在工作中我们实际的业务逻辑往往有可能出现一些相对特殊的需求需要对 ...

  4. 24 margin的用法

    margin塌陷问题 当时说到了盒模型,盒模型包含着margin,为什么要在这里说margin呢?因为元素和元素在垂直方向上margin里面有坑. 我们来看一个例子: html结构: <div ...

  5. UI-grid 表格内容可编辑(enableCellEdit可指定列编辑)

    在网上搜索了很多关于UI-Grid的问题 很遗憾好少啊啊啊 不过有API还是比较欣慰的 官方API:UI Grid 还有一位大佬的翻译的中文API:angularjs ui-grid中文api 行编辑 ...

  6. kubernetes实战篇之helm使用技巧

    系列目录 使用压缩包安装chart 我们使用helm package打包的时候,默认会在当前位置生成一个tgz压缩包,然后helm把它复制到到$HOME/.helm/repository目录下,现在还 ...

  7. SSM框架学习笔记_第1章_SpringIOC概述

    第1章 SpringIOC概述 Spring是一个轻量级的控制反转(IOC)和面向切面(AOP)的容器框架. 1.1 控制反转IOC IOC(inversion of controller)是一种概念 ...

  8. 如何使用 Docker 安装 Jenkins

    说在前面 本篇内容非常简单,仅讲述了如何快速在 Docker 上部署一个 Jenkins 实例,不涉及其他. 本文实验环境: 操作系统:Centos 7.5 Docker Version:18.09. ...

  9. Python中文件的读写操作

    文件操作基本流程: 1. 介绍 计算机系统是由计算机硬件,操作系统,和应用程序三部分组成. 内存 存放不持久 硬盘 可以使数据持久化 文件操作  数据持久化的一种 全栈开发  框架类 2. 文件的操作 ...

  10. C++中 =default,=delete用法

    =default: 用于显式要求编译器提供合成版本的四大函数(构造.拷贝.析构.赋值) 例如: class A{ public: A() = default; A(const A& a) = ...