本文仅仅是为了说明synchronized关键字同步的是对象不是方法,列子的确有失偏颇。

一.明确一点synchronized同步的是对象不是方法也不是代码块 

我有关synchronized同步的是对象讨论的博客在这里:https://www.cnblogs.com/SAM-CJM/p/9798263.html

只要明确了synchronized同步的是对象那么,底下的问题就好解决了。

二.问题的导入

首先我有一个班级,班级中有学生。那么我们可以这样来模拟这个问题,代码如下:

import java.util.ArrayList;

public class Class {
//班级类
ArrayList<Student> students;
public Class(ArrayList<Student> students){
this.students=students;
} } //学生类
class Student{
private String name;
private int ID; public Student(String name,int ID){
this.ID=ID;
this.name=name;
}
}

然后我们有一个班级事务处理类,来处理事务,这个类继承了Thread类代码入下:

import java.util.ArrayList;
import java.util.Arrays; public class ClassText extends Thread{//班级事务处理类 Class class1;//待处理事务的班级 //构造器
public ClassText(Class class1){
this.class1=class1;
} @Override//没有同步run方法
public void run(){
super.run();
addStudent(new Student("张三",15));
} //添加学生,没有同步该方法
public void addStudent(Student student){
System.out.println("添加前现在有"+class1.students.size()+"个学生");
class1.students.add(student);
System.out.println("添加后现在有"+class1.students.size()+"个学生");
} public static void main(String[] args) {
Class c=new Class(new ArrayList<>(Arrays.asList(new Student("李四",20),new Student("赵牛",20))));
ClassText ct1=new ClassText(c);
ClassText ct2=new ClassText(c);
ct1.start();
ct2.start();
} }

显然上面的代码是错误的,他没同步两个线程,那么我们看看结果是什么样子的:

果然出现了线程不安全的情况。

那么,我们马上同步化我们的addStudent方法或者是run方法,代码如下:

这里同步addStudent方法

    //添加学生,同步化该方法
public synchronized void addStudent(Student student){
System.out.println("添加前现在有"+class1.students.size()+"个学生");
class1.students.add(student);
System.out.println("添加后现在有"+class1.students.size()+"个学生");
}

结果如下:

还是不同步的。

那么这到底是为什么呢?

三.解决方法

其实我们无论在ClassText类里面的哪个方法加synchronized使其同步化都是没有用的,因为你同步化的是你的ClassText对象,而我们要同步的是处理的是在ClassText类中组合Class对象,因为我们是对他的进行共享资源的操作,那么问题来了,怎么对不是本身对象进行一个同步化操作呢?还是使用synchronized同步代码块只不过同步对象不再是this了而是共享数据处理的对象,修改代码如下:

    //添加学生
public void addStudent(Student student){
//同步化修改共享数据的对象
synchronized (class1){
System.out.println("添加前现在有"+class1.students.size()+"个学生");
class1.students.add(student);
System.out.println("添加后现在有"+class1.students.size()+"个学生");
}
}

结果如下:没毛病了!

当然你把添加学生的添加函数放到Class类中,同步改方法也是没问题的

所以还是那一条,同步化的对象,是需要同步化的对象。

深入理解Java并发synchronized同步化的代码块不是this对象时的操作的更多相关文章

  1. Java的synchronized的同步代码块和同步方法的区别

    synchronized同步方法和同步代码块的区别 同步方法默认使用this或者当前类做为锁. 同步代码块可以选择以什么来加锁,比同步方法更精确,我们可以选择只有会在同步发生同步问题的代码加锁,而并不 ...

  2. Java并发——synchronized关键字

    前言: 只要涉及到Java并发那么我们就会考虑线程安全,实际上能够实现线程安全的方法很多,今天先介绍一下synchronized关键字,主要从使用,原理介绍 一.synchronized的使用方法 1 ...

  3. 精通java并发-synchronized关键字和锁

    目前CSDN,博客园,简书同步发表中,更多精彩欢迎访问我的gitee pages synchronized关键字和锁 示例代码 public class MyThreadTest2 { public ...

  4. Java并发-Synchronized关键字

    一.多线程下的i++操作的并发问题 package passtra; public class SynchronizedDemo implements Runnable{ private static ...

  5. 深入理解Java并发框架AQS系列(一):线程

    深入理解Java并发框架AQS系列(一):线程 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念 一.概述 1.1.前言 重剑无锋,大巧不工 读j.u.c包下的源码,永远无法绕开的经典 ...

  6. 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念

    深入理解Java并发框架AQS系列(一):线程 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念 一.AQS框架简介 AQS诞生于Jdk1.5,在当时低效且功能单一的synchroni ...

  7. 深入理解Java并发框架AQS系列(四):共享锁(Shared Lock)

    深入理解Java并发框架AQS系列(一):线程 深入理解Java并发框架AQS系列(二):AQS框架简介及锁概念 深入理解Java并发框架AQS系列(三):独占锁(Exclusive Lock) 深入 ...

  8. java 并发——synchronized

    java 并发--synchronized 介绍 在平常我们开发的过程中可能会遇到线程安全性的问题,为了保证线程之间操作数据的正确性,我们第一想到的可能就是使用 synchronized 并且 syn ...

  9. java 为什么wait(),notify(),notifyAll()必须在同步(Synchronized)方法/代码块中调用?

    wait()作用:该方法用来将当前线程置入休眠状态,直到接到通知或被中断为止.条件:在调用wait()之前,线程必须要获得该对象的对象级别锁,即只能在同步方法或同步块中调用wait()方法.进入wai ...

随机推荐

  1. 【题解】Luogu P2153 [SDOI2009]晨跑

    原题传送门 一眼应该就能看出是费用流 因为每个交叉路口只能通过一次,所以我们进行拆点,连一条流量为1费用为0的边 再按照题目给的边(是单向边)建图 跑一下MCMF就行了 拆点很套路的~ #includ ...

  2. topcoder srm 590 div1 (max_flow_template)

    problem1 link 对于每一个,找到其在目标串中的位置,判断能不能移动即可. problem2 link 如果最后的$limit$为$11=(1011)_{2}$,那么可以分别计算值为$(10 ...

  3. topcoder srm 570 div1

    problem1 link 找到周期,每个周期的增量是相同的. problem2 link 对于分给某一个公司的有$c$个联通分量,其中$k$个联通分量只有1个节点,$c$个联通分量一共有$x$个节点 ...

  4. 1、代理服务器及haproxy基础

    1.web站点架构 前端一台主机提供app server,当用户请求到达时,如果要存储结构化数据,就需要找一台主机做database server.当业务达到一定程度时,要把web server.存储 ...

  5. OceanBase 2.0让百万支付不是梦?

    小蚂蚁说: 你们都很关心的 “OB双11大促实战分享” 专题来啦!本系列将为你系统性的介绍OceanBase支撑蚂蚁双11背后的技术原理和实战分享.从平台到架构,再到实现,一起来探索蚂蚁双11这场神秘 ...

  6. (14)线程- Event事件和守护线程Daemon

    <一>Event事件 线程Event基本和进程的Event语法是一样的 # wait() 动态给程序加阻塞 # set() 将内部属性改成True # clear() 将内部属性改成Fal ...

  7. MongoDB数据库基本命令

    MongoDB的启动 先找到MongoDB所在的路径,然后执行以下代码 mongod --dbpath=E:\Program Files\MongoDB\Server\3.4\data\db --db ...

  8. Python字符编码的发展、cmd寻找路径

    字符编码的发展: ASCII(American Standard Code for Information Interchange,美国标准信息交换代码)是基于拉丁字母的一套电脑编码系统,主要用于显示 ...

  9. day15_python_1124

    03序列化模块 04加密模块 05 os sys 模块 06 collections 模块 # 03 序列化模块 # 网络传输数据:字节 bytes# 文件写入内容:bytes , str # dic ...

  10. 学习Linux系统中命令的简单方法

    如果说如何快速学习.了解Linux的话,我的答案是学命令.背命令!为何呢?对于一名新手来说,去学习Linux的思想.了解Linux的架构.明白Linux中“一切皆文件”概念虽然说是没有错,是对的.但是 ...