java线程学习之线程同步synchronized

简介: java线程学习之线程同步synchronized

synchronized学习

线程安全

java支持多个线程访问同一个对象或者对象的成员变量,在并发编程中,这种被多个线程并发访问的资源称为临界资源。并且每个线程可以拥有对这个临界资源的拷贝,所以在程序执行过程中看到的变量不一定是最新的,无论何时只要有多于一个的线程访问给定的共享变量,而且其中某个线程会写入该变量,此时必须用同步来协调线程对该变量的访问,以保证某一时刻只有一个线程进入访问该共享资源。

线程同步

synchronized介绍

java提供了内置锁synchronized关键字来保证我们对共享资源的同步,synchronized可以修饰代码块或者方法,它本质上是一种互斥锁或者说独占锁,也就是当一个线程获取到锁之后,其他线程只能够等待当前线程释放锁之后才能进入同步方法或者代码块。每个对象都有一个锁标记monitor(或者叫监视器),当线程拥有了这个锁标记才能够访问这个资源,没有获取到锁标记的线程只能进入锁池,系统会为每个对象都创建一个互斥锁,这个锁是分配给线程使用的,为了防止打断原子操作,每个锁都只能分配给一个线程,因此叫做互斥锁。

对象锁和类锁

对象锁说明

当synchronized作用于对象或者实例方法时,称之为对象锁。

1、当一个线程访问某个对象的synchronized同步方法或者同步代码块时,也就获取到了当前对象的对象锁,其他线程如果想要访问此同步方法或者同步代码块,便需要阻塞等待,直到前一个线程从同步代码块或者同步方法中退出。

2、访问同一个类的不同实例对象的同步代码块或同步方法时,不存在阻塞等待获取对象锁的问题,因为他们获取的是各自实例的对象级别锁,相互之间没有影响。

3、使用synchronized(obj)同步语句块时,可以获取指定对象上的对象锁。obj为对象的引用,如果获取到了obj的对象锁,在并发访问obj时,便会在synchronized代码处阻塞等待,直到获取该obj对象的对象锁,当obj为this时便是获取到当前对象的对象锁。

类锁说明

当synchronized作用于类或者静态方法时称之为类锁。

1、类锁是某个类的Class对象,每个Class对象在虚拟机中只有一个,所以类锁也只有一个。

2、类锁是所有实例共享,用于控制对static成员变量或者static方法的并发访问。

synchronized使用

synchronized关键字可用来修饰方法或者代码块。

1、修饰方法,分为实例方法和静态方法

1.1 修饰实例方法,对象锁
public  synchronized void objectMethods(){
   .....
    }
1.2 修饰静态方法,类锁
public static  synchronized void staticMethods(){
   .....
 }

2、修饰代码块

2.1 obj为对象的引用 对象锁
public  void objectMethods(){
      synchronized (obj){
    ......
    }
    }
2.2  Object 为某个类 类锁
public  void classLock(){
    synchronized (Object.class){
  ......
  }
  }

synchronized 特性

互斥性

同时只有一个线程能够访问synchronized方法或者同步代码块。

可重入性

synchronized是可重入锁,通俗解释可重入锁就是当一个线程获取到了某个对象锁或者类锁之后,这个线程在未释放锁之前,再调用该锁的其他synchronized方法或代码块时,不用再次重新获得锁。

如下代码:

public class SynReentrant {
    public static void main(String[] args) {
        SynReentrant synReentrant = new SynReentrant();
        synReentrant.method();
    }
    public synchronized void method(){
        System.out.println("method,currentThread:"+Thread.currentThread().getId());
        synchronized (this){
            System.out.println("this,currentThread:"+Thread.currentThread().getId());
            method2();
        }
    }
    public synchronized void method2(){
        System.out.println("method2,currentThread:"+Thread.currentThread().getId());
    }
}

结果:

method,currentThread:1
this,currentThread:1
method2,currentThread:1

主线程调用method()时获得了synReentrant 对象锁,进入同步method、method2及同步代码块中没有阻塞,根据synchronized互斥性,可证synchronized具有可重入性,也就是说synchronized是可重入锁。

可重入锁的机制:JVM会为每个锁关联一个计数器和持有者线程,当计数器为0时说明当前锁没有被任何线程持有,当某个线程请求获取当前锁并成功时,当前锁的计数器加1,如果同一个线程再次获取这个锁计数器将递增,当线程退出同步代码块时,计数器将递减,直到计数器为0,线程释放当前锁。

可见性

  1. 线程每次获取到锁时都要拷贝一份共享资源到线程本地内存中
  2. 线程每次释放锁时,都要将本地内存中的共享资源刷新到共享内存中去
  3. 共享资源的可见性保证了当共享资源变化时,所有对该共享资源的操作的线程都能感受到变化,所有对共享资源的操作都是以共享内存为准。

原子性

对共享资源的一组操作,要么成功要么失败,不会出现部分成功部分失败的情况。

目录
相关文章
|
6月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
315 1
|
6月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
321 1
|
7月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
302 0
|
7月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
482 16
|
8月前
|
缓存 并行计算 安全
关于Java多线程详解
本文深入讲解Java多线程编程,涵盖基础概念、线程创建与管理、同步机制、并发工具类、线程池、线程安全集合、实战案例及常见问题解决方案,助你掌握高性能并发编程技巧,应对多线程开发中的挑战。
|
8月前
|
数据采集 存储 前端开发
Java爬虫性能优化:多线程抓取JSP动态数据实践
Java爬虫性能优化:多线程抓取JSP动态数据实践
|
9月前
|
Java API 调度
从阻塞到畅通:Java虚拟线程开启并发新纪元
从阻塞到畅通:Java虚拟线程开启并发新纪元
455 83
|
9月前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
382 0
|
9月前
|
存储 Java 调度
Java虚拟线程:轻量级并发的革命性突破
Java虚拟线程:轻量级并发的革命性突破
491 83
下一篇
开通oss服务