一、同步方法

public synchronized void methodAAA(){
    //….
}

锁定的是调用这个同步方法的对象

测试:
a、不使用这个关键字修饰方法,两个线程调用同一个对象的这个方法。
目标类:

public class TestThread {
   public  void execute(){  //synchronized,未修饰
        for(int i=0;i<100;i++){
            System.out.println(i);
      }
   }
}

线程类:

public class ThreadA implements Runnable{
    TestThread test=null;
    public ThreadA(TestThread pTest){  //对象有外部引入,这样保证是同一个对象
        test=pTest;
    }
    public void run() {
        test.execute();
    }
}

调用:

TestThread test=new TestThread();
Runnable runabble=new ThreadA(test);
Thread a=new Thread(runabble,"A");
a.start();
Thread b=new Thread(runabble,"B");
b.start();

结果:
输出的数字交错在一起。说明不是同步的,两个方法在不同的线程中是异步调用的。

b、修改目标类,增加synchronized修饰

public class TestThread {
    public synchronized  void execute(){  //synchronized修饰
        for(int i=0;i<100;i++){
            System.out.println(i);
        }
    }
}

结果:
输出的数字是有序的,首先输出A的数字,然后是B,说明是同步的,虽然是不同的线程,但两个方法是同步调用的。
注意:上面虽然是两个不同的线程,但是是同一个实例对象。下面使用不同的实例对象进行测试。

c、每个线程都有独立的TestThread对象。
目标类:

public class TestThread {
    public synchronized void execute(){  //synchronized修饰
        for(int i=0;i<100;i++){
            System.out.println(i);
        }
    }
}

线程类:

public class ThreadA implements Runnable{
    public void run() {
        TestThread test=new TestThread();
        test.execute();
    }
}

调用:

Runnable runabble=new ThreadA();
Thread a=new Thread(runabble,"A");
a.start();
Thread b=new Thread(runabble,"B");
b.start();

结果:
输出的数字交错在一起。说明虽然增加了synchronized 关键字来修饰方法,但是不同的线程调用各自的对象实例,两个方法仍然是异步的。

引申:
对于这种多个实例,要想实现同步即输出的数字是有序并且按线程先后顺序输出,我们可以增加一个静态变量,对它进行加锁(后面将说明锁定的对象)。

修改目标类:

public class TestThread {
     private static Object lock=new Object(); //必须是静态的。
     public  void execute(){
         synchronized(lock){
             for(int i=0;i<100;i++){
                 System.out.println(i);
             }
         }
     }
}

二、同步代码块

public void method(SomeObject so){
    synchronized(so)
       //…..
    }
}

锁定一个对象,其实锁定的是该对象的引用(object reference)
谁拿到这个锁谁就可以运行它所控制的那段代码。当有一个明确的对象作为锁时,就可以按上面的代码写程序,但当没有明确的对象作为锁,只是想让一段代码同步时,可以创建一个特殊的instance变量(它必须是一个对象)来充当锁(上面的解决方法就是增加了一个状态锁)。

a、锁定一个对象,它不是静态的
private byte[] lock = new byte[0]; // 特殊的instance变量
目标类:

public class TestThread {
     private Object lock=new Object();
     public  void execute(){
         synchronized(lock){  //增加了个锁,锁定了对象lock,在同一个类实例中,是线程安全的,但不同的实例还是不安全的。因为不同的实例有不同对象锁lock.
             for(int i=0;i<100;i++){
                 System.out.println(i);
             }
        }
    }
}  

其实上面锁定一个方法,等同于下面的:

public void execute(){
    synchronized(this){   //同步的是当然对象
        for(int i=0;i<100;i++){
            System.out.println(i);
        }
    }
}

b、锁定一个对象或方法,它是静态的
这样锁定,它锁定的是对象所属的类

public synchronized  static void execute(){
    //...
}
等同于
public class TestThread { public static void execute(){ synchronized(TestThread.class){ // } } }

测试:

目标类:

public class TestThread {
     private static Object lock=new Object();
     public synchronized static void execute(){  //同步静态方法
         for(int i=0;i<100;i++){
             System.out.println(i);
         }
     }
     public static void execute1(){
         for(int i=0;i<100;i++){
            System.out.println(i);
        }
    }
    public void test(){
        execute();     //输出是有序的,说明是同步的
        //execute1();  //输出是无须的,说明是异步的
    }
}
线程类:调用不同的方法,于是建立了两个线程类
 public class ThreadA implements Runnable{
     public void run() {
         TestThread.execute();//调用同步静态方法
     }
 }
 public class ThreadB implements Runnable{
     public void run() {
         TestThread test=new TestThread();
         test.test();//调用非同步非静态方法
    }
}
调用:
Runnable runabbleA=new ThreadA();
Thread a=new Thread(runabbleA,"A");
a.start();
Runnable runabbleB=new ThreadB();
Thread b=new Thread(runabbleB,"B");
b.start();
注意:
用synchronized 来锁定一个对象的时候,如果这个对象在锁定代码段中被修改了,则这个锁也就消失了。看下面的实例:

目标类:
 public class TestThread {
    private static final class TestThreadHolder {
       private static TestThread theSingleton = new TestThread();
       public static TestThread getSingleton() {
            return theSingleton;
       }
       private TestThreadHolder() {
       }
    }
    private Vector ve =null;
    private Object lock=new Object();
    private TestThread(){
        ve=new Vector();
        initialize();
    }
    public static TestThread getInstance(){
        return TestThreadHolder.getSingleton();
    }
    private void initialize(){
        for(int i=0;i<100;i++){
            ve.add(String.valueOf(i));
        }
    }
    public void reload(){
        synchronized(lock){
            ve=null;
            ve=new Vector();
                        //lock="abc";
            for(int i=0;i<100;i++){
                ve.add(String.valueOf(i));
            }
        }
        System.out.println("reload end");
    }

    public boolean checkValid(String str){
        synchronized(lock){
            System.out.println(ve.size());
            return ve.contains(str);
        }
    }
}

说明:在reload和checkValid方法中都增加了synchronized关键字,对lock对象进行加锁。在不同线程中对同一个对象实例分别调用reload和checkValid方法。
在reload方法中,不修改lock对象即注释lock="abc"; ,结果在控制台输出reload end后才输出100。说明是同步调用的。
如果在reload方法中修改lock对象即去掉注释,结果首先输出了一个数字(当前ve的大小),然后输出reload end。说明是异步调用的。

2、单例模式中对多线程的考虑

 public class TestThread {
    private static final class TestThreadHolder {
        private static TestThread theSingleton = new TestThread();
        public static TestThread getSingleton() {
            return theSingleton;
        }
        private TestThreadHolder() {
        }
    }
    private Vector ve =null;
    private Object lock=new Object();
    private TestThread(){
        ve=new Vector();
        initialize();
    }
    public static TestThread getInstance(){
        return TestThreadHolder.getSingleton();
    }
} 

说明:增加了一个内部类,在内部类中申明一个静态的对象,实例化该单例类,初始化的数据都在单例类的构造函数中进行。这样保证了多个实例同时访问的时候,初始化的数据都已经成功初始化了。

总结:
A. 无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁,所以首先应知道需要加锁的对象
B.每个对象只有一个锁(lock)与之相关联。
C.实现同步是要很大的系统开销作为代价的,甚至可能造成死锁,所以尽量避免无谓的同步控制。

synchronized的作用的更多相关文章

  1. JAVA关键词synchronized的作用

    记下来,很重要. Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchron ...

  2. 搜狗一道java题目 关于对象 synchronized 关键字作用在 int, integer

      第一次见到这个题目,我觉得自己没学到java,太浅了,其实这个问题没有考synchronized关键字,只是考什么是对象? 1.在java编程思想的第二章有一句话; 一切都是对象,很可惜int,c ...

  3. Android 虹软2.0人脸识别,注册失败问题 分析synchronized的作用

    人脸识别需要init初始化(FaceServer中),离开时需要unInit销毁:当一个含有人脸识别的界面A跳向另一个含有人脸识别的界面B时,由于初始化和销毁都是对FaceServer类加锁(sync ...

  4. synchronized 的作用?

    在 Java 中,synchronized 关键字是用来控制线程同步的,就是在多线程的环境 下,控制 synchronized 代码段不被多个线程同时执行. synchronized 既可以加在一段代 ...

  5. 从JAVA看C#中volatile和synchronized关键字的作用

    最近一直在想C#中 volatile关键字到底是用来干什么的?查了很多.NET的文章都是说用volatile修饰的变量可以让多线程同时修改,这是什么鬼... 然后查到了下面这篇JAVA中关于volat ...

  6. Java 关键字volatile 与 synchronized 作用与区别

     1,volatile 它所修饰的变量不保留拷贝,直接访问主内存中的.    在Java内存模型中,有main memory,每个线程也有自己的memory (例如寄存器).为了性能,一个线程会在自己 ...

  7. synchronized和lock的作用与对比

    一.synchronized的作用 synchronized是java中的一个关键字,用于线程同步.1. 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象 ...

  8. java中关键字volatile的作用

    用在多线程,同步变量. 线程为了提高效率,将某成员变量(如A)拷贝了一份(如B),线程中对A的访问其实访问的是B.只在某些动作时才进行A和B的同步.因此存在A和B不一致的情况.volatile就是用来 ...

  9. Thread 学习记录 <1> -- volatile和synchronized

    恐怕比较一下volatile和synchronized的不同是最容易解释清楚的.volatile是变量修饰符,而synchronized则作用于一段代码或方法:看如下三句get代码: int i1;  ...

随机推荐

  1. 分布式版本控制git常见问题之gitignore冲突(精简版)

    上次写的的太模糊了,现在简单直接写出个人心得,如下: 原因是有人提交了.gitignore里面的内容,所以和本地的不一样,这样就有问题,那么pull都不可以,所以要这样: git update-ind ...

  2. 弹出框插件layer使用

    layer是一款近年来备受青睐的web弹层组件,她具备全方位的解决方案,致力于服务各水平段的开发人员,您的页面会轻松地拥有丰富友好的操作体验. 插件官方地址:http://layer.layui.co ...

  3. idea的一揽子工程

    我总觉得,世上如果人人都像我一样,那路该多难走.有时候在网上找到一些问题的答案,成功解决之后,就这么过去了,实在罪过. 将idea的几个常见的使用问题综合到一起吧,如果有不会用的,欢迎留言.当然,请支 ...

  4. Range Modular Queries

    Range Modular Queries 题意 给出一个数列,q个查询,问查询区间内有几个数 a[i] % x == y. 分析 其实裸的分块就能过了,跑的还特别快. 这里分块的作用就是排序. 在x ...

  5. PHP版本替换, phpinfo和php -v显示版本信息不一致

    环境:OS X EI Capitan 10.11 & lnmp 背景: 1想将lamp(xampp安装的,php5.2)换成 lnmp(php7.0)   2php5.2卸载(xampp卸载& ...

  6. git与github的使用

    git和github是两个完全不同的概念,就好比如雷锋与雷峰塔的关系. git是一个版本管理工具,用来更好的管理你的程序,比如你原来提交过的内容,以后虽然修改了,但是通过git这个工具, 可以把你原来 ...

  7. maven问题:org.springframewor.web.filter.CharacterEncodingFileter不能强转为javax.servlet.Filter

    使用maven搭建ssm(SpringMVC.Spring .Mybatis)项目,启动时报: java.lang.ClassCastException:org.springframewor.web. ...

  8. 非Controller中调用Service

    1.       新增文件 package com.library.common; import org.springframework.beans.BeansException; import or ...

  9. Caused by: java.lang.RuntimeException: by java.lang.OutOfMemoryError: PermGen space(tomcat 启动时提示内存溢出)

    设置MaxPermSize大小TOMCAT_HOME/bin/catalina.bat 文件头加set JAVA_OPTS='-Xms512m -Xmx1024m -XX:MaxPermSize=51 ...

  10. [转] .NET领域驱动设计—初尝(疑问、模式、原则、工具、过程、框架、实践)

    阅读目录: 1.1.疑问 1.1.1.UML何用 1.1.2.领域建模 1.2.模式 1.3.原则 1.5.过程 1.6.框架 1.7.项目演示 最近在研究DDD颇有收获,所以整理出来跟大家分享,共同 ...