android synchronized的使用


今天,简单讲讲


synchronized 的使用。


synchronized 是Java语言关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。synchronized 关键字,它包括两种用法:synchronized 方法和 synchronized 块。  

本文直接以代码的形式来展示 synchronized 关键字的使用:

一.synchronized 的简单示例

【1】synchronized  Demo1:

package com.andyidea.demo;

/**
 * 当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,
 * 一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码
 * 块以后才能执行该代码块。
 * @author Andy.Chen
 *
 */
public class Thread01 implements Runnable {

	@Override
	public void run() {

		synchronized (this) {
			for(int i=0;i<3;i++){
				System.out.println(Thread.currentThread().getName()+" synchronized loop "+i);
			}
		}
	}
	
	public static void main(String[] args) {
		Thread01 t01 = new Thread01();
		
		System.out.println("Welcome to Andy.Chen Blog! \n"
		          +"synchronized 关键字使用 \n"
		          +"--------------------------");
		
		Thread ta = new Thread(t01,"A");
		Thread tb = new Thread(t01,"B");
		
		ta.start();
		tb.start();
	}

}

运行结果如下:

Welcome to Andy.Chen Blog! 
synchronized 关键字使用 
--------------------------
B synchronized loop 0
B synchronized loop 1
B synchronized loop 2
A synchronized loop 0
A synchronized loop 1
A synchronized loop 2


【2】synchronized  Demo2:

package com.andyidea.demo;

/**
 * 当一个线程访问object的一个synchronized(this)同步代码块时,
 * 另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。
 * @author Andy.Chen
 *
 */
public class Thread02 {
	
	public void method01(){
		synchronized (this) {
			int i=0;
			while(i++ < 3){
				System.out.println(Thread.currentThread().getName() +":"+ i);
				try {
					Thread.sleep(1000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	public void method02(){
		
		//第1种方式:当一个线程访问object的一个synchronized(this)同步代码块时,
		//另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。
//		int j=0;
//		while(j++ < 3){
//			System.out.println(Thread.currentThread().getName() +":"+ j);
//			try {
//				Thread.sleep(1000);
//			} catch (InterruptedException e) {
//				e.printStackTrace();
//			}
//		}
		
		//第2种方式:当一个线程访问object的一个synchronized(this)同步代码块时,
		//其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。
		synchronized (this) {
			int j=0;
			while(j++ < 3){
				System.out.println(Thread.currentThread().getName() +":"+ j);
				try {
					Thread.sleep(1000);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	/**
	 * 当一个线程访问object的一个synchronized(this)同步代码块时,
	 * 它就获得了这个object的对象锁。
	 * 结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。
	 */
	public synchronized void method3(){
		int k=0;
		while(k++ < 3){
			System.out.println(Thread.currentThread().getName() +":"+ k);
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}
	
	public static void main(String[] args) {
		final Thread02 t02 = new Thread02();
		
		System.out.println("Welcome to Andy.Chen Blog! \n"
		          +"synchronized 关键字使用 \n"
		          +"--------------------------");
		
		Thread t02A = new Thread(new Runnable() {
			
			@Override
			public void run() {
				t02.method01();
			}
		},"A");
		
		Thread t02B = new Thread(new Runnable() {
			
			@Override
			public void run() {
				t02.method02();
			}
		},"B");
		
		Thread t02C = new Thread(new Runnable() {
			
			@Override
			public void run() {
				t02.method3();
			}
		},"C");
		
		t02A.start();
		t02B.start();
		t02C.start();
	}

}


运行结果如下:

Welcome to Andy.Chen Blog! 
synchronized 关键字使用 
--------------------------
B:1
B:2
B:3
C:1
C:2
C:3
A:1
A:2
A:3



二.synchronized与static synchronized 的区别

1.synchronized与static synchronized 的区别
      synchronized是对类的当前实例进行加锁,防止其他线程同时访问该类的该实例的所有synchronized块,注意这里是“类的当前实例”, 类的两个不同实例就没有这种约束了。那么static synchronized恰好就是要控制类的所有实例的访问了,static synchronized是限制线程同时访问jvm中该类的所有实例同时访问对应的代码快。实际上,在类中某方法或某代码块中有 synchronized,那么在生成一个该类实例后,改类也就有一个监视快,放置线程并发访问改实例synchronized保护快,而static synchronized则是所有该类的实例公用一个监视快了,也也就是两个的区别了,也就是synchronized相当于 this.synchronized,而
static synchronized相当于Something.synchronized.
      一个日本作者-结成浩的《java多线程设计模式》有这样的一个列子:
      pulbic class Something(){
         public synchronized void isSyncA(){}
         public synchronized void isSyncB(){}
         public static synchronized void cSyncA(){}
         public static synchronized void cSyncB(){}
     }
   那么,加入有Something类的两个实例a与b,那么下列组方法何以被1个以上线程同时访问呢
   a.   x.isSyncA()与x.isSyncB()
   b.   x.isSyncA()与y.isSyncA()
   c.   x.cSyncA()与y.cSyncB()
   d.   x.isSyncA()与Something.cSyncA()
    这里,很清楚的可以判断:
   a,都是对同一个实例的synchronized域访问,因此不能被同时访问
   b,是针对不同实例的,因此可以同时被访问
   c,因为是static synchronized,所以不同实例之间仍然会被限制,相当于Something.isSyncA()与   Something.isSyncB()了,因此不能被同时访问。
   那么,第d呢?,书上的 答案是可以被同时访问的,答案理由是synchronzied的是实例方法与synchronzied的类方法由于锁定(lock)不同的原因。
   个人分析也就是synchronized 与static synchronized 相当于两帮派,各自管各自,相互之间就无约束了,可以被同时访问。目前还不是分清楚java内部设计synchronzied是怎么样实现的。


    结论:A: synchronized static是某个类的范围,synchronized static cSync{}防止多个线程同时访问这个    类中的synchronized static 方法。它可以对类的所有对象实例起作用。
  
               B: synchronized 是某实例的范围,synchronized isSync(){}防止多个线程同时访问这个实例中的synchronized 方法。



     2.synchronized方法与synchronized代码快的区别
      synchronized methods(){} 与synchronized(this){}之间没有什么区别,只是 synchronized methods(){} 便于阅读理解,而synchronized(this){}可以更精确的控制冲突限制访问区域,有时候表现更高效率。


     3.synchronized关键字是不能继承的
     这个在http://www.learndiary.com/archives/diaries/2910.htm一文中看到的,我想这一点也是很值得注意的,继承时子类的覆盖方法必须显示定义成synchronized.


android synchronized的使用就讲完了。


就这么简单。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值