`
toknowme
  • 浏览: 139877 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

synchronized的使用

    博客分类:
  • J2EE
 
阅读更多

在编写一个类时,如果该类中的代码可能运行于多线程环境下,那么就要考虑同步的问题。在Java中内置了语言级的同步原语--synchronized,这也大大简化了Java中多线程同步的使用。

直接上代码

package com.crm.learn.thread;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class TestThread {

	public static CountDownLatch count = new CountDownLatch(2);

	public static Object o = new Object();
	public static Long numx = new Long(0l);

	public static void main(String[] args) {
		Lock mainLock = new ReentrantLock();
		try {
			VThread v1 = new VThread(1000000, mainLock);
			VThread v2 = new VThread(2000000, mainLock);
			
			v1.start();
			v2.start();
			count.await();
			System.out.println(numx);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}

 

线程类

package com.crm.learn.thread;

import java.util.concurrent.locks.Lock;

public class VThread extends Thread {

	private long addNum = 0;
	private Lock mainLock;

	public VThread(long addNum, Lock mainLock) {
		this.addNum = addNum;
		this.mainLock = mainLock;
	}

	public void run() {
		try {
			for (int i = 0; i < addNum; i++) {
				try {
					//如果锁定numx,数量 != 3000000
//					synchronized (TestThread.numx) {
//						TestThread.numx++;
//					}
					
					//如果锁定o,数量 =3000000
//					synchronized (TestThread.o) {
//						TestThread.numx++;
//					}
					
					//如果锁定TestThread.class,数量 =3000000
//					synchronized (TestThread.class) {
//						TestThread.numx++;
//					}
					
					//使用锁对象,数量 =3000000
//					mainLock.lock();
//					TestThread.numx++;
//					mainLock.unlock();
					
				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			TestThread.count.countDown();
		}
	}
}

 

 

为啥

synchronized (TestThread.numx) {

TestThread.numx++;

 

}

这个值不为3000000

0
3
分享到:
评论
4 楼 toknowme 2015-10-29  
lisaiori 写道
// synchronized (TestThread.numx) { 
//      TestThread.numx++; 
// } 
第一次循环的时候,TestThread.numx 是静态成员变量,多个线程之间访问的是同一把锁
然而每次++运算之后,会 new 一个新的Long对象出来,这样多个线程之间,锁定的对象就不是同一个了。所以,结果不对。


还是昊子大神给力!  
3 楼 lisaiori 2015-10-27  
// synchronized (TestThread.numx) { 
//      TestThread.numx++; 
// } 
第一次循环的时候,TestThread.numx 是静态成员变量,多个线程之间访问的是同一把锁
然而每次++运算之后,会 new 一个新的Long对象出来,这样多个线程之间,锁定的对象就不是同一个了。所以,结果不对。
2 楼 toknowme 2015-09-08  
老兄!你这是两个不同的线程,不同的SynchronizedCode 对象,各跑各的没有问题呀。
synchronized (this)这里的this要是同一个对象才行,需要将对象从类的外面传进去,你在这里写死,它永远不可能时同一个对象。


看来还是这个老兄厉害

package com.crm.learn.thread;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class TestThread {

	public static CountDownLatch count = new CountDownLatch(2);

	public static Long o = new Long(01);
	public static Long numx = new Long(0l);

	public static void main(String[] args) {
		Lock mainLock = new ReentrantLock();
		try {
			VThread v1 = new VThread(1000000, mainLock, numx);
			VThread v2 = new VThread(2000000, mainLock, numx);

			v1.start();
			v2.start();
			count.await();
			System.out.println(numx);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}






package com.crm.learn.thread;

import java.util.concurrent.locks.Lock;

public class VThread extends Thread {

	private long addNum = 0;
	private Lock mainLock;
	private Object self;

	public VThread(long addNum, Lock mainLock, Object self) {
		this.addNum = addNum;
		this.mainLock = mainLock;
		this.self = self;
	}

	public void run() {
		try {
			for (int i = 0; i < addNum; i++) {
				try {
					// 如果锁定numx,数量 != 3000000
					// synchronized (TestThread.numx) {
					// TestThread.numx++;
					// }

					// 如果锁定o,数量 =3000000
					// synchronized (TestThread.o) {
					// TestThread.numx++;
					// }

					// 如果锁定TestThread.class,数量 =3000000
					// synchronized (TestThread.class) {
					// TestThread.numx++;
					// }

					// 使用锁对象,数量 =3000000
					// mainLock.lock();
					// TestThread.numx++;
					// mainLock.unlock();

					// 锁定自己
					synchronized (self) {
						TestThread.numx++;
					}

				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			TestThread.count.countDown();
		}
	}
}


1 楼 toknowme 2015-09-08  
自己补一刀:

synchronized必须在同一个实例中使用。

两次new之后,生成两个实例,各自的synchronized不会对其它实例的thread起作用。


这个应该怎么理解?

相关推荐

    Java synchronized使用案例

    在`Synchronized`压缩包中,可能包含了通过实验来演示`synchronized`关键字使用的一些Java代码。例如,创建一个共享资源类,然后通过多个线程去操作这个资源,使用`synchronized`关键字来保证线程安全。实验可能涉及...

    17 资源有限,请排队等候—Synchronized使用、原理及缺陷.pdf

    本章重点讲述了`synchronized`的使用、工作原理及其潜在的缺陷。 首先,`synchronized`的主要作用在于保证线程同步,即在多线程环境下,对共享资源的访问进行控制,确保同一时刻只有一个线程能够访问特定的代码块或...

    解决Maven 项目报错 java.httpservlet和synchronized使用方法

    解决Maven项目报错java.httpservlet和synchronized使用方法 在Maven项目中,我们经常会遇到一些报错问题,例如java.httpservlet和synchronized使用方法等。这篇文章将为大家带来解决Maven项目报错java.httpservlet...

    synchronized用法大全实例

    本实例大全将全面解析`synchronized`的使用方式,包括同步方法、同步语句块、类锁和对象锁。 ### 1. 同步方法 同步方法是通过在方法声明前加上`synchronized`关键字实现的。这样,同一时间只有一个线程可以执行该...

    java中synchronized的使用

    java中synchronized的使用,java中的锁锁的到底是什么?是括号里的代码块吗?肯定不是的;

    [JAVA][synchronized的使用]

    本篇文章将深入探讨`synchronized`的使用,包括其基本原理、使用方式以及实际应用场景。 1. **synchronized的基本原理** `synchronized`关键字在Java中提供了互斥访问,也就是说,当一个线程进入一个对象的`...

    java锁机制Synchronizedjava锁机制Synchronized

    但是,在某些情况下,可能需要在一定时期内霸占某个对象的钥匙,这时可以使用 Synchronized 代码块。 Java 锁机制 Synchronized 的优点 Java 锁机制 Synchronized 的优点是可以解决多线程并发访问共享资源时可能...

    java里面synchronized用法.doc

    Java 中的 synchronized 用法详解 Java 中的 synchronized 关键字是用于解决多线程并发问题的重要工具之一。...正确地使用 synchronized 关键字可以帮助开发者们编写更加高效、可靠的多线程程序。

    Android synchronized 测试案例

    本测试案例深入探讨了`synchronized`的使用方法,包括同步单个对象、同步多个对象以及成功与失败的场景对比。 一、`synchronized`关键字的基本概念 `synchronized`关键字可以修饰方法或用作代码块,其主要作用是...

    java中synchronized用法

    "Java 中 synchronized 用法详解" Synchronized 是 Java 语言中用于解决多线程共享数据...synchronized 关键字是 Java 语言中解决多线程共享数据同步问题的重要工具,但需要正确地使用它,以免造成死锁和系统开销。

    Java中使用synchronized关键字实现简单同步操作示例

    下面将详细介绍`synchronized`关键字的三种使用方式:修饰函数、修饰代码块以及修饰静态方法。 1. **synchronized 修饰函数** 当`synchronized`修饰非静态方法时,它会锁定当前对象实例。这意味着,对于同一个对象...

    volatile和synchronized的区别

    5. **线程阻塞**:使用`synchronized`时,当一个线程持有锁时,其他试图获取该锁的线程会被阻塞;而使用`volatile`时,不会发生线程阻塞。 #### 四、使用建议 - 对于简单的状态变量(如标志位)的可见性保证,推荐...

    synchronized关键字的用法详解

    `synchronized`关键字有两种主要的使用方式:一种是在方法声明中使用,另一种则是在代码块中使用。 ##### 1. synchronized方法 在方法声明中添加`synchronized`关键字,可以将整个方法体变成同步代码块。例如: `...

    synchronized并发讲解源码.zip

    本文将深入解析`synchronized`的工作原理、使用方式以及它在并发编程中的重要性。通过分析提供的源码,我们可以更直观地理解这个关键字的实际应用。 `synchronized`关键字主要有两种用法:修饰实例方法和同步代码块...

    synchronized的几种示例

    本文将深入探讨`synchronized`的几种使用示例,包括方法加锁、代码块加锁(针对`this`和对象)以及静态方法加锁。 1. **方法加锁** 当在实例方法前使用`synchronized`关键字时,它会锁定当前对象的监视器,只有...

    synchronized关键字的实质及用法

    1. **synchronized的两种使用方式** - **方法同步**:在方法声明前加上`synchronized`关键字,这将使得整个方法成为同步方法,每次只有一个线程可以执行该方法。 ```java public synchronized void someMethod...

    Synchronized关键字的用法

    - **非阻塞算法**:为了减少锁的竞争,可以考虑使用非阻塞算法或工具类如`java.util.concurrent`包下的类来替代`synchronized`。 #### 结论 `synchronized`关键字是Java中实现线程安全的基础工具之一,通过对共享...

    正确使用多线程同步锁@synchronized()1

    本文将深入探讨`@synchronized`的工作原理、特点以及使用时需要注意的事项。 首先,`@synchronized`的原理是基于递归互斥锁(Recursive Mutex)。在Objective-C中,当你使用`@synchronized`块包围一段代码时,会为...

    JAVA synchronized详解

    通过使用`synchronized`,可以确保同一时刻只有一个线程能够访问被标记为同步的方法或代码块,从而避免多线程环境下的数据不一致问题。 #### 一、基本概念 `synchronized`关键字主要具有以下特性: 1. **互斥性**...

    java_synchronized详解

    通过在方法或代码块上使用`synchronized`,可以确保同一时间只有一个线程能访问这些代码区域,从而有效避免了多线程环境下的数据竞争和不一致性问题。 #### 二、synchronized的作用机制 `synchronized`主要通过对象...

Global site tag (gtag.js) - Google Analytics