java synchronized究竟锁住的是什么

  刚学java的时候,仅仅知道synchronized一个线程锁。能够锁住代码,可是它真的能像我想的那样,能够锁住代码吗?
  在讨论之前先看一下项目中常见关于synchronized的使用方法:
 public synchronized  void syncCurrentObject() {
             System.out.println(Thread.currentThread().getName()+"..start.."+"-----"+System.currentTimeMillis());
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"..end.."+"-----"+System.currentTimeMillis());  

    }

这样是否能在多个线程訪问时候,保证仅仅有一个线程进入方法。其他线程堵塞吗?

我用线程池创建三个线程容量,分别启动五个线程:

    public static void syncCurrentObjectTest() {
        ExecutorService exec = Executors.newFixedThreadPool(3);

    //  final GenerateCode gCode = new GenerateCode();

        for (int i = 0; i < 5; i++) {
            exec.execute(new Runnable() {
                @Override
                public void run() {
                    GenerateCode gCode = new GenerateCode();
                    gCode.syncCurrentObject();
                }
            });

        }
        exec.shutdown();
    }

运行效果截图:

依据截图的输入日志说明将synchronized加在方法上并不能让线程安全,而是多个线程并行运行。比方:线程3并没有等线程1运行完毕后再运行。而是线程1休眠的时候。线程3直接获得锁。进行运行。那么在原有的实现上。假设保证线程安全呢?

解决思路:在多个线程调用synchronized修饰的方法时,调用synchronized方法是同一个对象。

详细解决方式是:将GenerateCode 对象创建一次(写成单例更好)。然后调用synchronized修饰方法。

详细改动截图例如以下:

为什么这样改就能够呢?原理是由于对于成员方法。synchronized仅仅能锁住当前对象的线程,其他对象的线程无法锁住。

并且synchronized放在方法和在方法内synchronize(this)是等价的。都仅仅能锁住当前对象。

可是假设想锁住不同对象的多个线程。该怎么做呢?演示样例代码例如以下:

    //直接在静态方法上加synchronized  线程安全
    public synchronized static void syncStatic() {
        //dosomething..
    }  

    //在静态方法上synchronized当前类 线程安全
    public  static void syncCurrentClass() {

            synchronized(GenerateCode.class){
                  //dosomething..
            }
        }

    //在成员方法上synchronized当前类 线程安全
    public   void syncCurrentObjectByThisClass() {
        synchronized(GenerateCode.class){
           //dosomething..
        }
    }

用synchronized锁住当前类字节码,当前类中总是仅仅有一个线程能够进入运行,其他线程进入堵塞。

总结:synchronized能够锁当前对象,也能够锁类。

synchronized锁住当前对象的写法:

public synchronized void a(){

}

public void ab(){

synchronized (this){

}

}

synchronized锁住当前类的写法:

public synchronized static void a(){

}

public static void a(){

synchronized (类名){

}

}

public void ab(){

synchronized (类名){

}

}

我的理解是:当synchronized作用在对象时候,同一个对象中的线程是相互排斥的,仅仅有一个线程运行完毕后。另外一个线程才干获得对象锁得到运行。

假设不是同一个对象,则不会产生相互排斥

当synchronized作用在类时,对于同一个jvm中不同对象的多个线程调用同一个synchronized修饰的方法都是相互排斥的。由于一个jvm仅仅会产生一个class文件。

拓展:

1、假设想让线程相互排斥,synchronized方法是否存在效率问题?

理论应该是存在效率问题的,由于每一个对象都有一个对象锁,当一个线程拿到锁后,其他线程必须堵塞。(未写代码測试)

2、假设是分布式的系统,使用synchronized无效了。由于synchronized最多仅仅能锁住当前JVM的线程。对于其他server的线程无能为力。

那么怎么处理呢?

查了些资料。网上说能够用zookeeper+其他组件完毕分布式锁或者乐观锁。

由于没有详细实践过。仅仅是看了几篇文章。没有发言权,所以感兴趣的朋友能够自行搜索 java+分布式锁

完整測试demo源代码:http://download.csdn.net/download/zl544434558/9495663

时间: 2024-10-26 09:43:56

java synchronized究竟锁住的是什么的相关文章

Synchronized到底锁住的是谁?

先来一道并发编程笔试题 题目:利用5个线程并发执行,num数字累计计数到10000,并打印. /** * Description: * 利用5个线程并发执行,num数字累加计数到10000,并打印. * 2019-06-13 * Created with OKevin. */ public class Count { private int num = 0; public static void main(String[] args) throws InterruptedException {

java synchronized类锁,对象锁详解(转载)

觉得还不错 留个记录,转载自http://zhh9106.iteye.com/blog/2151791 在java编程中,经常需要用到同步,而用得最多的也许是synchronized关键字了,下面看看这个关键字的用法. 因为synchronized关键字涉及到锁的概念,所以先来了解一些相关的锁知识. java的内置锁:每个java对象都可以用做一个实现同步的锁,这些锁成为内置锁.线程进入同步代码块或方法的时候会自动获得该锁,在退出同步代码块或方法时会释放该锁.获得内置锁的唯一途径就是进入这个锁的

JAVA synchronized关键字锁机制(中)

synchronized 锁机制简单的用法,高效的执行效率使成为解决线程安全的首选. 下面总结其特性以及使用技巧,加深对其理解. 特性: 1. Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码.       2. 当一个线程同时访问object的一个synchronized(this)同步代码块时,其它线程仍然可以访问非修饰的方法或代码块.       3. 当多个线程同时访问object的synchronized(this)同步代码

【java并发】(2) Java线程同步:synchronized锁住的是代码还是对象

在Java中,synchronized关键字是用来控制线程同步的,就是在多线程的环境下,控制synchronized代码段不被多个线程同时执行.synchronized既可以加在一段代码上,也可以加在方法上. 关键是,不要认为给方法或者代码段加上synchronized就万事大吉,看下面一段代码: class Sync { public synchronized void test() { System.out.println("test开始.."); try { Thread.sle

Java synchronized 锁住的是整个对象还是只有方法

先上结论: 类方法中,synchronized锁住的是对象this,只有调用同一个对象的方法才需要获取锁.同时,同一个对象中所有加了synchronize的方法只能一次调用一个 静态方法中,synchronized锁的是整个类对象,类似于(X.class),该类中所有加了synchronized的静态方法,一次只能调用一个 1 class Sync { 2 3 public synchronized void test() { 4 System.out.println("test开始..&quo

【转载】Java中的锁机制 synchronized &amp; Lock

参考文章: http://blog.csdn.net/chen77716/article/details/6618779 目前在Java中存在两种锁机制:synchronized和Lock,Lock接口及其实现类是JDK5增加的内容,其作者是大名鼎鼎的并发专家Doug Lea.本文并不比较synchronized与Lock孰优孰劣,只是介绍二者的实现原理. 数据同步需要依赖锁,那锁的同步又依赖谁?synchronized给出的答案是在软件层面依赖JVM,而Lock给出的方案是在硬件层面依赖特殊的

java多线程(二)——锁机制synchronized(同步方法)

synchronized Java语言的关键字,可用来给对象和方法或者代码块加锁,当它锁定一个方法或者一个代码块的时候,同一时刻最多只有一个线程执行这段代码.当两个并发线程访问同一个对象object中的这个加锁同步代码块时,一个时间内只能有一个线程得到执行.另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块.然而,当一个线程访问object的一个加锁代码块时,另一个线程仍然可以访问该object中的非加锁代码块. ——以上来源百度百科 一.方法内的变量为线程安全 “非线程安全”的问题

从 synchronized 到 CAS 和 AQS - 彻底弄懂 Java 各种并发锁

概述Java 中的并发锁大致分为隐式锁和显式锁两种.隐式锁就是我们最常使用的 synchronized 关键字,显式锁主要包含两个接口:Lock 和 ReadWriteLock,主要实现类分别为 ReentrantLock 和 ReentrantReadWriteLock,这两个类都是基于 AQS(AbstractQueuedSynchronizer) 实现的.还有的地方将 CAS 也称为一种锁,在包括 AQS 在内的很多并发相关类中,CAS 都扮演了很重要的角色.我们只需要弄清楚 synchr

浅谈Java中的锁:Synchronized、重入锁、读写锁

Java开发必须要掌握的知识点就包括如何使用锁在多线程的环境下控制对资源的访问限制 ◆ Synchronized ◆ 首先我们来看一段简单的代码: 12345678910111213141516171819 public class NotSyncDemo { public static int i=0; static class ThreadDemo extends Thread { @Override public void run() { for (int j=0;j<10000;j++)