本文仅仅是为了说明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. spring 事务的七中传播行为五中隔离

    事务的传播行为(七): public enum Propagation { REQUIRED(TransactionDefinition.PROPAGATION_REQUIRED), SUPPORTS ...

  2. A_Pancers团队作业4—基于原型的团队项目需求调研与分析

    任务1:实施团队项目软件用户调研活动. (1)用户调研对象:我们的项目软件是基于安卓系统的音乐播放器,以设计出操作简单的音乐播放器为目的,所以本次用户调研的对象主要以身边的老人为主,对他们听音乐,听戏 ...

  3. Django 编写模板并渲染的示例

    >>> from django.template import Template, Context >>> raw_template = ""& ...

  4. [JavaScript] 给input标签传值

    body: <input type="text" style="width: 240px;" name="orgname" id=&q ...

  5. spark中的combineByKey函数的用法

    一.函数的源码 /** * Simplified version of combineByKeyWithClassTag that hash-partitions the resulting RDD ...

  6. CentOS安装glibc异常Protected multilib versions

    安装失败 在执行yum install glibc.i686 libstdc++.i686 libcurl.i686安装命令时出现Protected multilib versions 解决方案 在命 ...

  7. 随机森林RF、XGBoost、GBDT和LightGBM的原理和区别

    目录 1.基本知识点介绍 2.各个算法原理 2.1 随机森林 -- RandomForest 2.2 XGBoost算法 2.3 GBDT算法(Gradient Boosting Decision T ...

  8. day042 css 选择器

    css(Cascading Style Sheet) 层叠样式表,定义了如何显示HTML元素,给HTML设置样式,让它看起来更好 一.css语法 css样式包括两部门,选择器(给谁的样式)和声明(什么 ...

  9. Petrozavodsk Winter Camp, Day 8, 2014, Rectangle Count

    给一个n*m的格点图,问其中有多少个矩形? $ \sum_{x=1}^{nm} \sum_{ab=x} [a + b \leq n](n - a - b + 1)\sum_{cd=x} [c + d ...

  10. promise学习总结

    什么是Promise Promise是异步编程的一种解决方案,它有三种状态,分别是pending-进行中.resolved-已完成.rejected-已失败 当Promise的状态又pending转变 ...