`
longgangbai
  • 浏览: 7328915 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

java synchronized静态同步方法与非静态同步方法,同步语句块

阅读更多

        进行多线程编程,同步控制是非常重要的,而同步控制就涉及到了锁。

       对代码进行同步控制我们可以选择同步方法,也可以选择同步块,这两种方式各有优缺点,至于具体选择什么方式,就见仁见智了,同步块不仅可以更加精确的控制对象锁,也就是控制锁的作用域,何谓锁的作用域?锁的作用域就是从锁被获取到其被释放的时间。而且可以选择要获取哪个对象的对象锁。但是如果在使用同步块机制时,如果使用过多的锁也会容易引起死锁问题,同时获取和释放所也有代价,而同步方法,它们所拥有的锁就是该方法所属的类的对象锁,换句话说,也就是this对象,而且锁的作用域也是整个方法,这可能导致其锁的作用域可能太大,也有可能引起死锁,同时因为可能包含了不需要进行同步的代码块在内,也会降低程序的运行效率。而不管是同步方法还是同步块,我们都不应该在他们的代码块内包含无限循环,如果代码内部要是有了无限循环,那么这个同步方法或者同步块在获取锁以后因为代码会一直不停的循环着运行下去,也就没有机会释放它所获取的锁,而其它等待这把锁的线程就永远无法获取这把锁,这就造成了一种死锁现象。


 
 详细解说一下同步方法的锁,同步方法分为静态同步方法与非静态同步方法。

       所有的非静态同步方法用的都是同一把锁——实例对象本身,也就是说如果一个实例对象的非静态同步方法获取锁后,该实例对象的其他非静态同步方法必须等待获取锁的方法释放锁后才能获取锁,可是别的实例对象的非静态同步方法因为跟该实例对象的非静态同步方法用的是不同的锁,所以毋须等待该实例对象已获取锁的非静态同步方法释放锁就可以获取他们自己的锁。

        而所有的静态同步方法用的也是同一把锁——类对象本身,这两把锁是两个不同的对象,所以静态同步方法与非静态同步方法之间是不会有竞态条件的。但是一旦一个静态同步方法获取锁后,其他的静态同步方法都必须等待该方法释放锁后才能获取锁,而不管是同一个实例对象的静态同步方法之间,还是不同的实例对象的静态同步方法之间,只要它们同一个类的实例对象!

    而对于同步块,由于其锁是可以选择的,所以只有使用同一把锁的同步块之间才有着竞态条件,这就得具体情况具体分析了,但这里有个需要注意的地方,同步块的锁是可以选择的,但是不是可以任意选择的!!!!这里必须要注意一个物理对象和一个引用对象的实例变量之间的区别!使用一个引用对象的实例变量作为锁并不是一个好的选择,因为同步块在执行过程中可能会改变它的值,其中就包括将其设置为null,而对一个null对象加锁会产生异常,并且对不同的对象加锁也违背了同步的初衷!这看起来是很清楚的,但是一个经常发生的错误就是选用了错误的锁对象,因此必须注意:同步是基于实际对象而不是对象引用的!多个变量可以引用同一个对象,变量也可以改变其值从而指向其他的对象,因此,当选择一个对象锁时,我们要根据实际对象而不是其引用来考虑!作为一个原则,不要选择一个可能会在锁的作用域中改变值的实例变量作为锁对象!!!!

 

Java线程:线程的同步与锁

 
 
 
一、同步问题提出
 
线程的同步是为了防止多个线程访问一个数据对象时,对数据造成的破坏。
例如:两个线程ThreadA、ThreadB都操作同一个对象Foo对象,并修改Foo对象上的数据。
 
public class Foo {
    private int x = 100;

    public int getX() {
        return x;
    }

    public int fix(int y) {
        x = x - y;
        return x;
    }
}
 
public class MyRunnable implements Runnable {
    private Foo foo = new Foo();

    public static void main(String[] args) {
        MyRunnable r = new MyRunnable();
        Thread ta = new Thread(r, "Thread-A");
        Thread tb = new Thread(r, "Thread-B");
        ta.start();
        tb.start();
    }

    public void run() {
        for (int i = 0; i < 3; i++) {
            this.fix(30);
            try {
                Thread.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + " : 当前foo对象的x值= " + foo.getX());
        }
    }

    public int fix(int y) {
        return foo.fix(y);
    }
}
 
运行结果:
Thread-A : 当前foo对象的x值= 40
Thread-B : 当前foo对象的x值= 40
Thread-B : 当前foo对象的x值= -20
Thread-A : 当前foo对象的x值= -50
Thread-A : 当前foo对象的x值= -80
Thread-B : 当前foo对象的x值= -80

Process finished with exit code 0
 
          从结果发现,这样的输出值明显是不合理的。原因是两个线程不加控制的访问Foo对象并修改其数据所致。
 
        如果要保持结果的合理性,只需要达到一个目的,就是将对Foo的访问加以限制,每次只能有一个线程在访问。这样就能保证Foo对象中数据的合理性了。
 
在具体的Java代码中需要完成一下两个操作:
把竞争访问的资源类Foo变量x标识为private;
同步哪些修改变量的代码,使用synchronized关键字同步方法或代码。
 
二、同步和锁定
 
1、锁的原理
 
Java中每个对象都有一个内置锁
 
         当程序运行到非静态的synchronized同步方法上时,自动获得与正在执行代码类的当前实例(this实例)有关的锁。获得一个对象的锁也称为获取锁、锁定对象、在对象上锁定或在对象上同步。
 
当程序运行到synchronized同步方法或代码块时才该对象锁才起作用。
 
            一个对象只有一个锁。所以,如果一个线程获得该锁,就没有其他线程可以获得锁,直到第一个线程释放(或返回)锁。这也意味着任何其他线程都不能进入该对象上的synchronized方法或代码块,直到该锁被释放。
 
释放锁是指持锁线程退出了synchronized同步方法或代码块。
 
关于锁和同步,有一下几个要点:
1)、只能同步方法,而不能同步变量和类;
2)、每个对象只有一个锁;当提到同步时,应该清楚在什么上同步?也就是说,在哪个对象上同步?
3)、不必同步类中所有的方法,类可以同时拥有同步和非同步方法。
4)、如果两个线程要执行一个类中的synchronized方法,并且两个线程使用相同的实例来调用方法,那么一次只能有一个线程能够执行方法,另一个需要等待,直到锁被释放。也就是说:如果一个线程在对象上获得一个锁,就没有任何其他线程可以进入(该对象的)类中的任何一个同步方法
5)、如果线程拥有同步和非同步方法,则非同步方法可以被多个线程自由访问而不受锁的限制。

6)、线程睡眠时,它所持的任何锁都不会释放。

7)、线程可以获得多个重进入(synchronized )锁。比如,在一个对象的同步方法里面调用另外一个对象的同步方法,则获取了两个对象的同步锁。

8)、同步损害并发性,应该尽可能缩小同步范围。同步不但可以同步整个方法,还可以同步方法中一部分代码块。
9)、在使用同步代码块时候,应该指定在哪个对象上同步,也就是说要获取哪个对象的锁。例如:
    public int fix(int y) {
        synchronized (this) {
            x = x - y;
        }
        return x;
    }
 
当然,同步方法也可以改写为非同步方法,但功能完全一样的,例如:
    public synchronized int getX() {
        return x++;
    }
    public int getX() {
        synchronized (this) {
            return x;
        }
    }
效果是完全一样的。
 
三、静态方法同步
 
要同步静态方法,需要一个用于整个类对象的锁,这个对象是就是这个类(XXX.class)。
例如:
public static synchronized int setName(String name){
      Xxx.name = name;
}
等价于
public static int setName(String name){
      synchronized(Xxx.class){
            Xxx.name = name;
      }
}

 
四、如果线程不能不能获得锁会怎么样
 
       如果线程试图进入同步方法,而其锁已经被占用,则线程在该对象上被阻塞。实质上,线程进入该对象的的一种池中,必须在哪里等待,直到其锁被释放,该线程再次变为可运行或运行为止。
 
当考虑阻塞时,一定要注意哪个对象正被用于锁定:
1、调用同一个对象中非静态同步方法的线程将彼此阻塞。如果是不同对象,则每个线程有自己的对象的锁,线程间彼此互不干预。
 
2、调用同一个类中的静态同步方法的线程将彼此阻塞,它们都是锁定在相同的Class对象上。
 
3、静态同步方法和非静态同步方法将永远不会彼此阻塞,因为静态方法锁定在Class对象上,非静态方法锁定在该类的对象上。
 
4、对于同步代码块,要看清楚什么对象已经用于锁定(synchronized后面括号的内容)。在同一个对象上进行同步的线程将彼此阻塞,在不同对象上锁定的线程将永远不会彼此阻塞。
 
五、何时需要同步
 
      在多个线程同时访问互斥(可交换)数据时,应该同步以保护数据,确保两个线程不会同时修改更改它。
 
对于非静态字段中可更改的数据,通常使用非静态方法访问。
对于静态字段中可更改的数据,通常使用静态方法访问
 
如果需要在非静态方法中使用静态字段,或者在静态字段中调用非静态方法,问题将变得非常复杂。已经超出SJCP考试范围了。
 
六、线程安全类
 
      当一个类已经很好的同步以保护它的数据时,这个类就称为“线程安全的”。
 即使是线程安全类,也应该特别小心,因为操作的线程是间仍然不一定安全。
 
举个形象的例子,比如一个集合是线程安全的,有两个线程在操作同一个集合对象,当第一个线程查询集合非空后,删除集合中所有元素的时候。第二个线程也来执行与第一个线程相同的操作,也许在第一个线程查询后,第二个线程也查询出集合非空,但是当第一个执行清除后,第二个再执行删除显然是不对的,因为此时集合已经为空了。
看个代码:
 
public class NameList {
    private List nameList = Collections.synchronizedList(new LinkedList());

    public void add(String name) {
        nameList.add(name);
    }

    public String removeFirst() {
        if (nameList.size() > 0) {
            return (String) nameList.remove(0);
        } else {
            return null;
        }
    }
}
 
public class Test {
    public static void main(String[] args) {
        final NameList nl = new NameList();
        nl.add("aaa");
        class NameDropper extends Thread{
            public void run(){
                String name = nl.removeFirst();
                System.out.println(name);
            }
        }

        Thread t1 = new NameDropper();
        Thread t2 = new NameDropper();
        t1.start();
        t2.start();
    }
}
 
虽然集合对象
    private List nameList = Collections.synchronizedList(new LinkedList());是同步的,但是程序还不是线程安全的。出现这种事件的原因是,上例中一个线程操作列表过程中无法阻止另外一个线程对列表的其他操作。
 
解决上面问题的办法是,在操作集合对象的NameList上面做一个同步。改写后的代码如下:
public class NameList {
    private List nameList = Collections.synchronizedList(new LinkedList());

    public synchronized void add(String name) {
        nameList.add(name);
    }

    public synchronized String removeFirst() {
        if (nameList.size() > 0) {
            return (String) nameList.remove(0);
        } else {
            return null;
        }
    }
}
 
这样,当一个线程访问其中一个同步方法时,其他线程只有等待。
 
七、线程死锁
 
       死锁对Java程序来说,是很复杂的,也很难发现问题。当两个线程被阻塞,每个线程在等待另一个线程时就发生死锁。
 
还是看一个比较直观的死锁例子:
 
public class DeadlockRisk {
    private static class Resource {
        public int value;
    }

    private Resource resourceA = new Resource();
    private Resource resourceB = new Resource();

    public int read() {
        synchronized (resourceA) {
            synchronized (resourceB) {
                return resourceB.value + resourceA.value;
            }
        }
    }

    public void write(int a, int b) {
        synchronized (resourceB) {
            synchronized (resourceA) {
                resourceA.value = a;
                resourceB.value = b;
            }
        }
    }
}
 
        假设read()方法由一个线程启动,write()方法由另外一个线程启动。读线程将拥有resourceA锁,写线程将拥有resourceB锁,两者都坚持等待的话就出现死锁。
 
        实际上,上面这个例子发生死锁的概率很小。因为在代码内的某个点,CPU必须从读线程切换到写线程,所以,死锁基本上不能发生。
 
         但是,无论代码中发生死锁的概率有多小,一旦发生死锁,程序就死掉。有一些设计方法能帮助避免死锁,包括始终按照预定义的顺序获取锁这一策略。已经超出SCJP的考试范围。
 
八、线程同步小结
 
1、线程同步的目的是为了保护多个线程反问一个资源时对资源的破坏。
2、线程同步方法是通过锁来实现,每个对象都有切仅有一个锁,这个锁与一个特定的对象关联,线程一旦获取了对象锁,其他访问该对象的线程就无法再访问该对象的其他同步方法。
3、对于静态同步方法,锁是针对这个类的,锁对象是该类的Class对象。静态和非静态方法的锁互不干预。一个线程获得锁,当在一个同步方法中访问另外对象上的同步方法时,会获取这两个对象锁。
4、对于同步,要时刻清醒在哪个对象上同步,这是关键。
5、编写线程安全的类,需要时刻注意对多个线程竞争访问资源的逻辑和安全做出正确的判断,对“原子”操作做出分析,并保证原子操作期间别的线程无法访问竞争资源。
6、当多个线程等待一个对象锁时,没有获取到锁的线程将发生阻塞。
7、死锁是线程间相互等待锁锁造成的,在实际中发生的概率非常的小。真让你写个死锁程序,不一定好使,呵呵。但是,一旦程序发生死锁,程序将死掉。

 

 

 

 

package com.etrip.concurrent.executor;

import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
/**
 * 非静态同步方法,静态同步方法,同步语句块的使用
 * 
 * 
 * 进行多线程编程,同步控制是非常重要的,而同步控制就涉及到了锁。 

       对代码进行同步控制我们可以选择同步方法,也可以选择同步块,这两种方式各有优缺点,至于具体选择什么方式,就见仁见智了,同步块不仅可以更加精确的控制对象锁,也就是控制锁的作用域,何谓锁的作用域?锁的作用域就是从锁被获取到其被释放的时间。而且可以选择要获取哪个对象的对象锁。但是如果在使用同步块机制时,如果使用过多的锁也会容易引起死锁问题,同时获取和释放所也有代价,而同步方法,它们所拥有的锁就是该方法所属的类的对象锁,换句话说,也就是this对象,而且锁的作用域也是整个方法,这可能导致其锁的作用域可能太大,也有可能引起死锁,同时因为可能包含了不需要进行同步的代码块在内,也会降低程序的运行效率。而不管是同步方法还是同步块,我们都不应该在他们的代码块内包含无限循环,如果代码内部要是有了无限循环,那么这个同步方法或者同步块在获取锁以后因为代码会一直不停的循环着运行下去,也就没有机会释放它所获取的锁,而其它等待这把锁的线程就永远无法获取这把锁,这就造成了一种死锁现象。 
 * 
 * @author longgangbai
 */
public class StaticInstanceLock {
	
	
	private   int count;
	private  static  StaticInstanceLock  instance=null;
	private StaticInstanceLock(){
	}
	/**
	 * 静态方法的锁
	 * 
	 * @return
	 */
	public static synchronized StaticInstanceLock getInstance(){
		if(instance==null){
			instance=new  StaticInstanceLock();
		}
		return instance;
	}

	/**
	 * 非静态方法的锁
	 * @return
	 */
	public synchronized int getCount(){
		return count;
	}
	
	public synchronized  void setCount(int count){
		this.count=count;
	}
	/**
	 * 同步语句块的使用
	 * 
	 */
	public void synmethod(){
		      //HashMap为非安全性Map
	          HashMap<String,String> hashmap = new HashMap<String,String>();
	          hashmap.put("ZH","中国");
	          hashmap.put("EN","英国");
	          hashmap.put("AM","美国");
	          hashmap.put("FR","法国");
	          
	          //创建一个同步的对象Map
			  Map<String,String> m = Collections.synchronizedMap(hashmap);
	          Set<String> s = m.keySet();  // Needn't be in synchronized block
	          //这里同步的对象均为需要使用同步的对象如Map而非Set
		      synchronized(m) {  // Synchronizing on m, not s!
		          Iterator<String> i = s.iterator(); // Must be in synchronized block
		          while (i.hasNext()){
		              foo(i.next());
		          }
		      }
	}
	
	public void foo(String entry){
		System.out.println("StaticInstanceLock ="+entry);
	}
	
	public static void main(String[] args) {
	
		StaticInstanceLock instance=StaticInstanceLock.getInstance();
		instance.setCount(7);
		int count = instance.getCount();
		instance.synmethod();
	}
	
}

 

分享到:
评论
1 楼 sunshine_bean 2016-10-18  
看了很多有关锁的文章,读完您的这篇,算是大彻大悟了

相关推荐

    synchronized用法大全实例

    - **对象锁**:如果`synchronized`修饰非静态方法或者同步语句块中的对象引用是实例引用,那么锁定的是该实例对象。例如: ```java public class MyObject { public synchronized void method() { // ... } }...

    Java synchronized关键字使用方式及特性解析

    在Java中,使用synchronized关键字可以作用在方法上,从而保证了访问同一个对象的同一个方法的线程同步。例如: public synchronized void testFun(String str){ for(int i=0;i;i++){ System.out.println(str+",...

    10个Java经典的Main方法面试题

    10. **Main方法的终结**:在Java中,main方法可以正常结束,当程序执行完main方法内的所有语句,或者遇到return语句时,main方法会结束,程序也随之终止。如果需要在main方法结束后执行某些清理工作,可以使用...

    Java多线程同步(wait()notify()notifyAll())[文].pdf

    - 使用wait()时,必须在同步代码块(由synchronized修饰的方法或代码块)中调用,因为只有持有锁的线程才能调用wait()。 - 恢复执行的线程会从wait()调用后的下一条语句开始,因此通常在循环中调用wait(),以确保...

    10个Java经典的Main方法面试题!.pdf

    因此,不能将main方法改为非静态。 5. **main方法是否可重载**:可以,一个Java类可以有多个main方法,只要它们的参数列表不同即可。但是,通常只会在主类中定义一个用于程序执行的main方法。 6. **main方法能否被...

    如何在JAVA中使用Synchronized

    当`synchronized`关键字用于修饰一个代码块时,它创建了一个同步语句块。例如: ```java synchronized(this) { // 代码块 } ``` 在这个例子中,`this`引用当前对象。这意味着,当一个线程进入这个同步代码块...

    Java面试宝典2018版

    2. 使用synchronized关键字进行同步,包括同步方法和同步代码块。 3. sleep()方法属于Thread类,调用后线程进入阻塞状态,而wait()方法属于Object类,调用后线程释放对象锁并进入等待状态。 字符串处理: 1. String...

    java程序员面试宝典

    12. Java中的静态成员:涉及静态变量和实例变量的区别,以及静态方法和非静态方法的调用规则。 对于笔试和面试中遇到的Java相关问题,求职者需要具备扎实的Java基础知识,并能灵活运用这些知识解决实际问题。另外,...

    java题库,超赞

    46. Java线程可以通过继承Thread类或实现Runnable接口来创建,synchronized关键字用于同步方法或代码块。 47. sleep是Thread类的方法,使当前线程暂停执行指定的时间;wait是Object类的方法,使当前线程进入等待状态...

    corejava_面试题

    不可以,因为非static方法依赖于特定对象实例,而static方法不依赖于任何对象,所以静态方法无法直接调用非静态方法。 13. Integer与int的区别? int是基本数据类型,而Integer是int的包装类。Integer可以存储null...

    JAVA水平测试考试题目

    * JAVA中的同步机制,synchronized关键字、wait、notify等方法。 7. JAVA数据库编程知识点: * JAVA中的JDBC(Java Database Connectivity)编程,Connection、Statement、ResultSet等接口。 * JAVA中的SQL语句...

    Java面试宝典2013_PDF版.pdf

    但是,如果在静态上下文中引用非静态成员,需要先创建外部类的实例。 28. AnonymousInnerClass(匿名内部类)是否可以extends(继承)其它类,是否可以implements(实现)interface(接口)? 匿名内部类既可以继承一个类也...

    java期末考试试卷

    2. **静态与非静态成员**:在静态方法中不能直接访问类的非静态数据成员,因为静态方法属于类,而非实例对象。 3. **参数传递**:Java中方法调用时参数传递总是按值传递,这意味着在方法内部修改参数不会影响到调用...

    java 程序设计笔记

    - synchronized:同步方法,修饰类方法(即 static 方法)调用前,将把系统类 Class中对应当前类的对象加锁。修饰对象方法(即非 static 方法)调用前,将把当前对象加锁。 7. 接口: - 编程者可以把用于完成特定...

    java实验报告(免费下载)

    - 构造代码块(非静态代码块)在创建对象时执行,每个实例化过程都会执行一次。 - 局部代码块(方法内的代码块)在进入该代码块时执行。 - 构造方法用于初始化新创建的对象。 2. **继承与类定义**: - 子类可以...

    java基础总结资料

    Java支持静态方法和非静态方法,前者与类关联,后者与对象关联。 6. **数组**:在Java中,数组是一种存储固定数量相同类型元素的数据结构。可以是一维、二维或多维数组。 7. **字符串处理**:Java中的`String`类是...

    java常见英语笔试题

    为何使用同步块(synchronized block)而不是同步方法? 同步块比同步方法更精确地控制锁的范围。同步块只锁定特定的代码块,因此其他线程可以访问类的其他方法,而不会阻塞。这样可以减少死锁的风险,并提高多线程...

    JAVA面试宝典2013版

    同步实现方法包括使用synchronized关键字,或者使用java.util.concurrent包中的工具,如ReentrantLock等。 51. 启动线程应该使用start()方法而不是run()方法。start()方法会启动一个新的线程,然后调用run()方法。...

    java面试资料汇总

    线程同步的常见方法有synchronized关键字、java.util.concurrent.locks包下的Lock和Condition接口等。 应用服务器如Tomcat、WebLogic和WebSphere等。 接口可以继承接口,抽象类可以实现接口,但抽象类不能继承实现...

Global site tag (gtag.js) - Google Analytics