Java并发编程--2.synchronized
前言
synchronized关键字是互斥锁,也称为内部锁
每个对象都有自己的monitor(锁标记),它是为了分配给线程,最多一个线程可以拥有对象的锁
使用
synchronized修饰成员方法,锁的是当前实例对象
下面是一个例子:
class Thread2 implements Runnable{
private int count;
//修饰成员方法,锁的是调用它的对象,该例中也即是调用它的线程
public synchronized void run() {
for (int i = 0; i < 5; i ++) {
try {
System.out.println(Thread.currentThread().getName() + ":" + (count++));
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
调用:
Thread2 t2 = new Thread2();
new Thread(t2).start();
new Thread(t2).start();
synchronized修饰静态方法,锁的是该类的Class对象
下面是一个例子:
class Thread3 implements Runnable {
private static int count;
//修饰静态方法, 锁的是这个类的所有对象
public static synchronized void getCounter() {
for (int i = 0; i < 5; i ++) {
try {
System.out.println(Thread.currentThread().getName() + ":" + (count++));
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
@Override
public void run() {
getCounter();
}
}
调用:
Thread3 t3_0 = new Thread3();
Thread3 t3_1 = new Thread3();
new Thread(t3_0).start();
new Thread(t3_1).start();
synchronized修饰代码块,锁的是()中的配置对象
下面是一个例子:
public class Synchronized {
private int count;
public void getCount(){
for (int i = 0; i < 5; i ++) {
try {
System.out.println(Thread.currentThread().getName() + ":" + (count++));
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) throws Exception {
Thread1 t1 = new Thread1(new Synchronized());
new Thread(t1).start();
new Thread(t1).start();
}
}
class Thread1 implements Runnable{
private Synchronized s;
public Thread1(Synchronized s) {
this.s = s;
}
@Override
public void run() {
//修饰代码块: 锁的是()中配置的对象
synchronized(s) {
s.getCount();
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
synchronized机制
Jvm需要对两类共享数据进行保护:
1.堆中的实例变量 2.本地方法区类变量
对于对象来说 , 有一个监视器保护实例变量; 对于类来说, 有一个监视器保护类变量
Jvm为每个对象和类关联一个锁,如果某个线程获取了锁, 在它释放锁之前其它的线程时不能获得同样的锁的;
对于一个对象,Jvm会维护一个加锁计数器,线程每获得一次该对象, 计数器+1, 每释放一次, 计数器-1,当计数器=0时,锁就完全释放了
死锁
当线程需要持有多个锁时, 就有可能发生死锁的情况, 比如下面这个情形:
A线程首先获得lock1,在获得lock2; B线程首先获得lock2,在获得lock1,
当A线程获得lock1时,B线程获得lock2, 于是A线程等待lock2, B线程等待lock1,
两个线程会无限的等待,这就发生了死锁现象
下面是一个例子:
public class Deadlock {
//监视器对象
private final Object lock1 = new Object();
private final Object lock2 = new Object();
public void instanceMethod1(){
synchronized(lock1){
System.out.println("线程1: 获得lock1,等待lock2");
synchronized(lock2) {
System.out.println("线程1:获得lock2");
}
}
}
public void instanceMethod2(){
synchronized(lock2){
System.out.println("线程2: 获得lock2,等待lock1");
synchronized(lock1){
System.out.println("线程2: 获得lock1");
}
}
}
public static void main(String[] args){
final Deadlock dld = new Deadlock();
Runnable r1 = new Runnable(){
@Override
public void run(){
while(true){
dld.instanceMethod1();
try{
System.out.println("线程1: 睡眠");
Thread.sleep(1000);
}
catch (InterruptedException ie){
}
}
}
};
Runnable r2 = new Runnable(){
@Override
public void run(){
while(true) {
dld.instanceMethod2();
try {
System.out.println("线程2: 睡眠");
Thread.sleep(1000);
}
catch (InterruptedException ie){
}
}
}
};
Thread thdA = new Thread(r1);
Thread thdB = new Thread(r2);
thdA.start();
thdB.start();
}
}
控制台输出:
线程1: 获得lock1,等待lock2
线程2: 获得lock2,等待lock1
避免发生死锁
生产中,死锁现象一旦发生,很可能会造成灾难性的后果,我们在编码中应该避免死锁现象发生
1、尽量不要编写在同一时刻需要持有多个锁的代码; 2、创建和使用一个大锁来代替若干小锁,并把这个锁用于互斥,而不是用作单个对象的对象级别锁;
锁的优化
synchronize采取独占的方式,它属于悲观锁,它假设了最坏的情况,如果持有锁的线程延迟,其他等待程序就会测试,程序停滞不前
锁自旋
在等待锁时,线程不会立即进入阻塞状态,而是先等一段时间看锁是否被释放
偏向锁
一个线程获得了锁,如果在接下来没有别的线程获得该锁,这个锁会偏向第一个获得它的线程,使一个线程多次获得锁的代价更低
锁膨胀
多次调用粒度太小的锁, 不如一次调用粒度大的锁
轻量级锁
如果存在锁的竞争,除了互斥量的开销外,还会发生CAS操作,在同步期间如果没有锁竞争,使用轻量级锁避免互斥量的开销
Java并发编程--2.synchronized的更多相关文章
- Java并发编程:synchronized
Java并发编程:synchronized 虽然多线程编程极大地提高了效率,但是也会带来一定的隐患.比如说两个线程同时往一个数据库表中插入不重复的数据,就可能会导致数据库中插入了相同的数据.今天我们就 ...
- Java并发编程:Synchronized及其实现原理
Java并发编程系列: Java 并发编程:核心理论 Java并发编程:Synchronized及其实现原理 Java并发编程:Synchronized底层优化(轻量级锁.偏向锁) Java 并发编程 ...
- Java并发编程:Synchronized底层优化(偏向锁、轻量级锁)
Java并发编程系列: Java 并发编程:核心理论 Java并发编程:Synchronized及其实现原理 Java并发编程:Synchronized底层优化(轻量级锁.偏向锁) Java 并发编程 ...
- 【转】Java并发编程:synchronized
一.什么时候会出现线程安全问题? 在单线程中不会出现线程安全问题,而在多线程编程中,有可能会出现同时访问同一个资源的情况,这种资源可以是各种类型的资源:一个变量.一个对象.一个文件.一个数据库表等,而 ...
- 4、Java并发编程:synchronized
Java并发编程:synchronized 虽然多线程编程极大地提高了效率,但是也会带来一定的隐患.比如说两个线程同时往一个数据库表中插入不重复的数据,就可能会导致数据库中插入了相同的数据.今天我们就 ...
- 【转】Java并发编程:Synchronized及其实现原理
一.Synchronized的基本使用 Synchronized是Java中解决并发问题的一种最常用的方法,也是最简单的一种方法.Synchronized的作用主要有三个:(1)确保线程互斥的访问同步 ...
- 【Java并发编程】synchronized相关面试题总结
目录 说说自己对于synchronized关键字的了解 synchronized关键字的三种使用 synchronized关键字的底层原理 JDK1.6之后对synchronized关键字进行的优化 ...
- Java并发编程实例(synchronized)
此处用一个小程序来说明一下,逻辑是一个计数器(int i):主要的逻辑功能是,如果同步监视了资源i,则不输出i的值,但如果没有添加关键字synchronized,因为是两个线程并发执行,所以会输出i的 ...
- Java并发编程:synchronized、Lock、ReentrantLock以及ReadWriteLock的那些事儿
目录 前言 synchronized用法 修饰方法 修饰实例方法 修饰静态方法 同步代码块 引出Lock Lock用法 子类:ReentrantLock 读写分离锁:ReadWriteLock Loc ...
- Java并发编程:synchronized和锁优化
1. 使用方法 synchronized 是 java 中最常用的保证线程安全的方式,synchronized 的作用主要有三方面: 确保线程互斥的访问代码块,同一时刻只有一个方法可以进入到临界区 保 ...
随机推荐
- 如果你的shiro没学明白,那么应该看看这篇文章,将shiro整合进springboot
最近在做项目的时候需要用到shiro做认证和授权来管理资源 在网上看了很多文章,发现大多数都是把官方文档的简介摘抄一段,然后就开始贴代码,告诉你怎么怎么做,怎么怎么做 相信很多小伙伴即使是跟着那些示例 ...
- hdu 2612
Find a way Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- Var与Dynamic的区别
1.var与dynamic的区别 C#中的很多关键词用法比较容易混淆,var和dynamic就是其中一组,但其实它们是有本质的区别的.var 在编译阶段已经确定类型,在初始化时候,必须提供初始化的 ...
- 【 js 片段 】移动端适配简易步骤
手机端全部适配简易步骤,不确保没有bug. 1.HTML 的 head 部分中加入如下代码: <meta name="viewport" content="widt ...
- 对MVC模式与MVVM模式的认识
MVC模式与MVVM模式目的一样,主要是分离模型(model)和视图(view),具体介绍如下. 名词介绍 MVC模式:指的是模型(Model)-- 视图(View)-- 控制器(Controller ...
- PHP通用分页类page.php[仿google分页]
<?php /** ** 通用php分页类.(仿Google样式) ** 只需提供记录总数与每页显示数两个参数.(已附详细使用说明..) ** 无需指定URL,链接由程序生成.方便用于检索结果分 ...
- jQuery源码学习笔记一
学习jQuery源码,我主要是通过妙味视频上学习的.这里将所有的源码分析,还有一些自己弄懂过程中的方法及示例整理出来,供大家参考. 我用的jquery v2.0.3版本. var rootjQuery ...
- 转 mysqli 事务常用方法
原文:mysqli 事务常用方法 1. //打开(true)或关闭(false)本次数据库连接的自动命令提交事务模式 //参数如果设置为 FALSE,则表示关闭 auto-commit.如果设置为 T ...
- html打造动画【系列3】- 小猫笑脸动画
猫咪容器 咱们每次画一个图片,肯定先要确定一个容器,几确定一下图形的位置和大小. <div class="mao_box"> <div class="m ...
- 深入理解net core中的依赖注入、Singleton、Scoped、Transient(二)
相关文章: 深入理解net core中的依赖注入.Singleton.Scoped.Transient(一) 深入理解net core中的依赖注入.Singleton.Scoped.Transient ...