IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 简单理解synchronized代码块锁对象的选取 -> 正文阅读

[Java知识库]简单理解synchronized代码块锁对象的选取

简单理解synchronized代码块锁对象的选取

1.自述

我先简单的说一哈自己的理解:

对于这个锁对象的选取应该满足一个最基本条件:唯一性。

我们可以想象一哈这个锁,他的作用是锁住资源(就是锁住当前线程操作的数据信息),如果这个锁有内容变化的情况,会导致同一个锁在不同时刻内容不一样,这样的结果是这一个锁前后不一致,没有了唯一性,具体解释结合下图:

在这里插入图片描述

我用简单的方式描述过程:

1.线程一和线程二都去抢cpu的时间片,现在线程一抢到了,线程二没抢到,那线程一去执行任务,线程二停在那准备抢下一次的cpu时间片。

2.线程一执行任务到代码块处,会根据你给的锁(就是给一个对象)制定一个标志,这个标志的作用是:在线程一操作代码块内部的代码时,整个代码块资源不对外开放(就是其他线程摸不到这个资源),可以这样想象一哈,这个代码块就是一个房间,锁就是这个房间的门锁,代码块中执行的代码涉及的数据就是房间的东西。

3.当线程一任务执行到一半,时间片用完了,线程一又和线程二抢cpu时间片。

4.如果线程一抢到,线程一会继续(线程一执行到一半没有时间片时的记录开始继续)执行任务,由于线程一第一次执行任务有了一个锁,现在线程一会拿着现在生成的锁去和之前的锁比较,如果一样,线程一自己有开这个锁的东西,就进入房间继续执行任务。

5.如果线程二抢到,线程一停在那准备抢下一次的cpu时间片,线程二执行到代码块,会生成一个锁,这个锁会和之前线程一创建的锁比较:如果锁不一样,线程二可以直接替换之前的锁,用自己的锁进入房间,进行任务执行;如果锁一样,线程二需要线程一打开锁的钥匙,但线程二没有这个钥匙,他打不开这个锁,进不了这个房间,所以他没法执行任务。

6.不管线程二有没有执行任务,等时间片用完,他又和线程一抢时间片

2.看代码运行结果

(锁不住)执行的任务代码:

public class Run implements Runnable{
    private int i=0;
    //设置可变的锁
    String nn=new String();
    //设置具有唯一性的锁
    final int ff=0;
    @Override
    public void run() {
        int j=0;
        //当前nn这个字符串对象就是一个可变锁
        synchronized (nn){
            //一个任务要执行完需要一个线程执行10次下面的步骤
            while (j < 10) {
                //记录当前是那个线程在做
                System.out.println(i + "    当前线程name:" + Thread.currentThread().getName());
                i++;
                j++;
                //设置新的值,让他变成可变的锁
                nn="sd"+nn;
            }
        }
    }
}
public class Run implements Runnable{
    private int i=0;
    //设置可变的锁
    String nn=new String();
    //设置具有唯一性的锁
    final int ff=0;
    @Override
    public void run() {
        int j=0;
        //每次new的对象地址变了,锁不住
        synchronized (new Object()){
            //一个任务要执行完需要一个线程执行10次下面的步骤
            while (j < 10) {
                //记录当前是那个线程在做
                System.out.println(i + "    当前线程name:" + Thread.currentThread().getName());
                i++;
                j++;
                //设置新的值,让他变成可变的锁
                nn="sd"+nn;
            }
        }
    }
}

线程测试代码:

public class ThreadTest {
    public static void main(String[] args) {
        Run run=new Run();
       Thread thread=new Thread(run,"1");
       Thread thread1=new Thread(run,"2");
       Thread thread2=new Thread(run,"3");
        thread.start();
        thread1.start();
        thread2.start();
    }
}

测试结果:可以看到资源完全没锁住,线程一执行过程中穿插有线程二和线程三对资源数据的修改,在线程一刚读到资源数据(我这里用的int i作为资源数据),时间片用完,线程二抢到时间片,线程二会去拿资源数据并执行i++操作后时间片用完,线程一抢到时间片,线程一拿到的是线程二执行之前的值,在此基础上执行i++,所以下方会出现两个i的值为6,这也解释了7这个数值为何在13之后这种现象。

在这里插入图片描述

当把执行任务的锁改成具有唯一性的锁后:

public class Run implements Runnable{
    private int i=0;
    //设置可变的锁
    String nn=new String();
    //设置具有唯一性的锁
    final int ff=0;
    @Override
    public void run() {
        int j=0;
          //当前ff这个int是基本数据类型,要封装成int对象数据才行,是一个唯一锁
        synchronized ((Integer)ff){
            //一个任务要执行完需要一个线程执行10次下面的步骤
            while (j < 10) {
                //记录当前是那个线程在做
                System.out.println(i + "    当前线程name:" + Thread.currentThread().getName());
                i++;
                j++;
                //设置新的值,让他变成可变的锁
                nn="sd"+nn;
            }
        }
    }
}

执行结果:可以发现线程一中没有穿插线程二和线程三,这里就说明在线程一没有执行完之前线程二和线程三执行时,压根就没打开过这个锁,也就是碰都没碰到这个i值的i++操作,直到线程一执行完后解开锁,线程二执行时又创这个锁,他有钥匙,线程一和线程三没法开锁。

在这里插入图片描述

总结:

我自己认为这里的锁要满足具有唯一性这个特征才会成功把资源锁起来;

唯一性的判断:只要对象的地址不发生改变,那他就具有唯一性;

举个例子:

我们常用的锁有

this:指的我当前的任务对象,由于该对象只创建了一次,所以地址唯一,可以将资源锁起来;

Object.class:这个是object类的模板类,在Java的jvm类加载机制中我们知道类加载只加载一次,那这个类对象是唯一的,也可以锁;

new Object().getClass():这个和上边的Object.class道理一样的。

最后希望我的理解思路可以帮到你,如果我哪里有错误的地方,烦请大佬指点一二,共同进步

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-01-01 13:44:53  更:2022-01-01 13:45:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 8:03:54-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码