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

Java线程同步 (synchronized wait notify)

    博客分类:
  • JAVA
阅读更多

Java线程同步 (synchronized wait notify)

同步(阻塞) :是一种防止对共享资源访问导致的数据不一致的一种模式。

在Java中,由于对多线程的支持,对同步的控制主要通过以下几个方法,synchronized,和wait(),notify()和notifyAll(),下面进行一一的讲解:


A关键字synchronized

每个java对象都有一把锁, 当有多个线程同时访问共享资源的时候, 需要Synchronize 来控制安全性, synchronize 分 synchronize 方法 和synchronize块,使用synchronize块时, 一定要显示的获得该对象的锁(如synchronize(object))而方法则不需要。

java的内存模型是对每一个进程有一个主内存, 每个线程有自己的内存, 他们从主内存中取数据, 然后计算, 再存入主内存中。

并发问题如下:如果多个线程同事操作同一数据, A线程从主内存中取的I的值为1, 然后进行加1操作, 这时B线程也取I的值, 进行加2操作, 然后A存入2到主内存中, B也存入, 这样就覆盖了A的值(同数据库中的并发问题一样)。

解决办法是用synchronize, 如用synchronized(I)。被synchronize 修饰的方法(块)把以下三步操作当成一个原子操作:取数据, 操作数据, 存数据。 我们知道原子操作是不可以被打断的, 所以其保证了数据一致性, 这样同一时间只有一个线程再执行, 对性能有一定的影响。这也是synchronize的第二个作用:保证统一时间只有一个线程再运行。 当实现SOCKET连接的时候经常用到.

JAVA中规定对非FLOAT, LONG的原始类型的取和存操作为原子操作。 其实就是对一个字(32位)的取,存位原始操作, 因为FLOAT, LONG为两个字节的长度, 所以其取, 存为非原子操作。 如果想把他们也变为原子操作, 可以用VOLATILE关键字来修饰


使用方法:

作用区域主要有两种:

1.方法

2.代码块

被synchronized声明的方法被称为同步方法,被其修饰的代码块称为同步语句。无论是同步方法还是同步语句,只要声明为同步了,在同一时刻,同一个对象的同步XX是不可以被同时访问的,而不同对象之间的同步方法是互不干扰的。


具体实现(如下代码都在某个类定义中):

同步方法:

Public synchronized void change() {

//

}


同步语句:(因为效率问题,有时考虑使用同步语句块)

    Public void change() {

Synchronized(this) {


}

}

这个同步语句是针对当前对象的,有时,我们就是想让一段代码同步,可能与当前对象并没什么关系,可以自定义同步的锁。如下:

private byte[]  lock= new byte[0];


 

  Public void change() {

Synchronized(lock) {


}

}

自定义锁注意事项:

1必须是private,防止在类外部引用改变。

2如果可能用到,重写get方法,返回对象的clone,而不是本身。

 

 

其他用法:

Synchronized除了可以作用于方法,代码块,还可以作用于静态方法,类,某个实例。但是都存在效率问题,一定要慎用。


Class Foo   {   


public synchronizedstatic void methodAAA()// 同步的static 函数  

{   


//….  


}  

  public void methodBBB()   {   


synchronized(Foo.class)// class literal(类名称字面常量) 


 } }

这样修饰后代表的是:统一时刻,被修饰部分只有一个对象可以运行,因为它的声明是针对类的。


2.wait()/notify()/notifyAll()


注意:

在Java中,每个对象都有个对象锁标志(Object lock flag)与之想关联,当一个线程A调用对象的一段synchronized代码时,

  它首先要获取与这个对象关联的对象锁标志,然后执行相应的代码,执行结束后,把这个对象锁标志返回给对象;因此,在线程A执行

  synchronized代码期间,如果另一个线程B也要执行同一对象的一段synchronized代码时(不一定与线程A执行的相同),它将

  要等到线程A执行完后,才能继续....

 

  如何利用wait() notify() notifyAll()?

 

  在synchronized代码被执行期间,线程可以调用对象的wait()方法,释放对象锁标志,进入等待状态,并且可以调用notify()或者

  notifyAll()方法通知正在等待的其他线程。notify()通知等待队列中的第一个线程,notifyAll()通知的是等待队列中的所有线程

例子程序:

/**

 *PrintNum.java

 * Created on 5:18:04 PM Feb 22, 2009

 *@author Quasar063501

 *@version 0.1

 *

 */

public class PrintNum {


private byte[] lock = new byte[0];  //自定义锁对象,这样代价最小,也可已使用当前对象this


public void demo() {

PrintThread a = new PrintThread("a");

PrintThread b = new PrintThread("b");

a.start();

b.start();

}


class PrintThread extends Thread {


public PrintThread(String name) {

this.setName(name);

}


public void run() {

synchronized(lock) {

for(int i =0; i < 100; i++) {

if(i % 10 == 0 && 0 != i) {

try {

lock.wait();   //暂时释放资源

lock.notify();       //唤醒另外一个进程

} catch (InterruptedException e) {

e.printStackTrace();

}

}

System.out.println(this.getName()+": "+i);

}

}

}

}

}

 

  Java线程同步 (synchronized wait notify)
注:wait notify 都是Object的方法

同步(阻塞) :是一种防止对共享资源访问导致的数据不一致的一种模式。

详细请参看操作系统。

 

 


在Java中,由于对多线程的支持,对同步的控制主要通过以下几个方法,synchronized,和wait(),notify()和notifyAll(),下面进行一一的讲解:


A关键字synchronized

每个java对象都有一把锁, 当有多个线程同时访问共享资源的时候, 需要Synchronize 来控制安全性, synchronize 分 synchronize 方法 和synchronize块,使用synchronize块时, 一定要显示的获得该对象的锁(如synchronize(object))而方法则不需要。

java的内存模型是对每一个进程有一个主内存, 每个线程有自己的内存, 他们从主内存中取数据, 然后计算, 再存入主内存中。

并发问题如下:如果多个线程同事操作同一数据, A线程从主内存中取的I的值为1, 然后进行加1操作, 这时B线程也取I的值, 进行加2操作, 然后A存入2到主内存中, B也存入, 这样就覆盖了A的值(同数据库中的并发问题一样)。

解决办法是用synchronize, 如用synchronized(I)。被synchronize 修饰的方法(块)把以下三步操作当成一个原子操作:取数据, 操作数据, 存数据。 我们知道原子操作是不可以被打断的, 所以其保证了数据一致性, 这样同一时间只有一个线程再执行, 对性能有一定的影响。这也是synchronize的第二个作用:保证统一时间只有一个线程再运行。 当实现SOCKET连接的时候经常用到.

JAVA中规定对非FLOAT, LONG的原始类型的取和存操作为原子操作。 其实就是对一个字(32位)的取,存位原始操作, 因为FLOAT, LONG为两个字节的长度, 所以其取, 存为非原子操作。 如果想把他们也变为原子操作, 可以用VOLATILE关键字来修饰


使用方法:

作用区域主要有两种:

1.方法

2.代码块

被synchronized声明的方法被称为同步方法,被其修饰的代码块称为同步语句。无论是同步方法还是同步语句,只要声明为同步了,在同一时刻,同一个对象的同步XX是不可以被同时访问的,而不同对象之间的同步方法是互不干扰的。


具体实现(如下代码都在某个类定义中):

同步方法:

Public synchronized void change() {

//

}


同步语句:(因为效率问题,有时考虑使用同步语句块)

    Public void change() {

Synchronized(this) {


}

}

这个同步语句是针对当前对象的,有时,我们就是想让一段代码同步,可能与当前对象并没什么关系,可以自定义同步的锁。如下:

private byte[]  lock= new byte[0];


 

  Public void change() {

Synchronized(lock) {


}

}

自定义锁注意事项:

1必须是private,防止在类外部引用改变。

2如果可能用到,重写get方法,返回对象的clone,而不是本身。

 

 

其他用法:

Synchronized除了可以作用于方法,代码块,还可以作用于静态方法,类,某个实例。但是都存在效率问题,一定要慎用。


Class Foo   {   


public synchronizedstatic void methodAAA()// 同步的static 函数  

{   


//….  


}  

  public void methodBBB()   {   


synchronized(Foo.class)// class literal(类名称字面常量) 


 } }

这样修饰后代表的是:统一时刻,被修饰部分只有一个对象可以运行,因为它的声明是针对类的。


2.wait()/notify()/notifyAll()


注意:

在Java中,每个对象都有个对象锁标志(Object lock flag)与之想关联,当一个线程A调用对象的一段synchronized代码时,

  它首先要获取与这个对象关联的对象锁标志,然后执行相应的代码,执行结束后,把这个对象锁标志返回给对象;因此,在线程A执行

  synchronized代码期间,如果另一个线程B也要执行同一对象的一段synchronized代码时(不一定与线程A执行的相同),它将

  要等到线程A执行完后,才能继续....

 

  如何利用wait() notify() notifyAll()?

 

  在synchronized代码被执行期间,线程可以调用对象的wait()方法,释放对象锁标志,进入等待状态,并且可以调用notify()或者

  notifyAll()方法通知正在等待的其他线程。notify()通知等待队列中的第一个线程,notifyAll()通知的是等待队列中的所有线程

例子程序:

/**

 *PrintNum.java

 * Created on 5:18:04 PM Feb 22, 2009

 *@author Quasar063501

 *@version 0.1

 *

 */

public class PrintNum {


private byte[] lock = new byte[0];  //自定义锁对象,这样代价最小,也可已使用当前对象this


public void demo() {

PrintThread a = new PrintThread("a");

PrintThread b = new PrintThread("b");

a.start();

b.start();

}


class PrintThread extends Thread {


public PrintThread(String name) {

this.setName(name);

}


public void run() {

synchronized(lock) {

for(int i =0; i < 100; i++) {

if(i % 10 == 0 && 0 != i) {

try {

lock.wait();   //暂时释放资源

lock.notify();       //唤醒另外一个进程

} catch (InterruptedException e) {

e.printStackTrace();

}

}

System.out.println(this.getName()+": "+i);

}

}

}

}

}

这个程序最终会因为互相唤醒而死锁

 

 

 

 

 

分享到:
评论

相关推荐

    Java 同步锁 wait notify 学习心得

    ### Java同步锁原理 在Java中,`synchronized`关键字用于实现线程同步,即确保同一时刻只有一个线程可以访问特定的代码块或方法。这种机制通过内部维护一个锁来实现,每个对象都有一个内置锁,这个锁可以被任何拥有...

    Java 同步方式 wait和notify/notifyall

    总结一下,`wait()`, `notify()`, 和 `notifyAll()` 是Java多线程编程中的核心工具,它们与`synchronized`关键字一起,为线程间的协作提供了强大的支持。理解并熟练掌握这些概念,对于编写高效、安全的多线程程序至...

    Java多线程wait和notify

    在Java中,`wait()` 和 `notify()` 方法是实现线程间通信和协作的重要工具,它们属于 `java.lang.Object` 类,这意味着所有类都默认继承了这两个方法。本文将详细探讨如何使用 `wait()` 和 `notify()` 来控制子线程...

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

    总之,Java多线程同步通过wait()、notify()和notifyAll()等方法,配合synchronized关键字,实现了对共享资源的访问控制,保证了程序的正确性。同时,原子操作和volatile关键字提供了更细粒度的线程安全保证。理解并...

    java之wait,notify的用法([ 详解+实例 ])

    在Java多线程编程中,wait和notify是两个非常重要的方法,它们都是Object类的方法,用于线程之间的通信和同步。下面我们将详细解释wait和notify的用法。 wait方法 wait方法是Object类的一个方法,用于让当前线程...

    浅谈java多线程wait,notify

    在Java多线程编程中,wait和notify是两个非常重要的机制,用于实现线程之间的通信和同步。在本文中,我们将通过示例代码详细介绍Java多线程wait和notify的使用,帮助读者更好地理解和掌握这两个机制。 wait机制 在...

    基于Java synchronized同步锁实现线程交互.pdf

    Java synchronized同步锁可以保证同一时刻只有一个线程操作同一资源,使用wait()、notify()切换线程状态保证线程操作的前后顺序实现线程交互。 Java线程状态有五种:新建状态、就绪状态、运行状态、休眠状态和死亡...

    Java多线程中wait、notify、notifyAll使用详解

    Java多线程中wait、notify、notifyAll使用详解 Java中多线程编程中,wait、notify、notifyAll三个方法是非常重要的,它们都是Object对象的方法,用于线程之间的通信。下面我们将详细介绍这三个方法的使用和作用。 ...

    Java的sychronized、wait和notify范例

    `synchronized`关键字、`wait()`和`notify()`方法是Java多线程中用于控制并发访问共享资源的重要工具,它们是Java内存模型(JMM)的一部分,主要用于解决线程间的同步问题。 一、`synchronized`关键字 `...

    java线程同步及通信

    Java提供了多种机制来实现线程同步,如`synchronized`关键字、`Lock`接口、`ReentrantLock`类、`Semaphore`信号量等。在`Callme.java`和`Caller.java`的例子中,`synchronized`关键字被用来同步方法`call()`和`f1()`...

    操作系统实验 多线程同步与互斥 java编写 有界面

    在“操作系统实验 多线程同步与互斥 java编写 有界面”的实验中,可能需要设计一个图形用户界面(GUI),通过按钮或事件触发线程的创建和同步操作,直观地展示线程间的交互和同步效果。例如,可以模拟银行账户转账,...

    Java线程同步例子.pdf

    在synchronized方法或代码块内部,可以通过调用wait()方法使当前线程放弃锁,并进入等待状态,直到其他线程调用notify()方法或notifyAll()方法。notify()方法唤醒在此对象监视器上等待的单个线程,而notifyAll()方法...

    java线程同步

    在Java中,线程同步的机制主要包括锁、同步块、同步方法、volatile关键字以及线程通信(如wait()、notify()和notifyAll())等。下面我们将详细探讨这些知识点。 1. **锁**:Java提供了两种基本的锁,即内置锁(也...

    java 多线程同步方法的实例

    Java提供了多种机制来实现线程同步,主要包括synchronized关键字、wait()、notify()和notifyAll()方法以及ReentrantLock等。 1. **synchronized关键字**:这是Java中最基本的线程同步方式。它可以用于修饰方法或...

    JAVA实现线程间同步与互斥生产者消费者问题

    在Java编程中,线程同步和互斥是多线程编程中的重要概念,它们用于解决多个线程同时访问共享资源时可能出现的问题。本项目通过一个生产者消费者问题的实例,展示了如何在Java中实现线程间的同步与互斥。 生产者消费...

    一个理解wait()与notify()的例子

    本文旨在解析一个具体的Java多线程示例代码,以帮助读者更好地理解`wait()`与`notify()`方法的作用及其实现机制。这两个方法是Java中实现线程间通信的重要手段之一,尤其在解决生产者消费者模型、读者写者问题等经典...

    java线程同步的例子[文].pdf

    首先,我们需要理解Java中的两个主要同步机制:`synchronized`关键字和`wait()`, `notify()`, `notifyAll()`方法。`synchronized`关键字可以用于修饰方法或代码块,它确保同一时间只有一个线程能执行特定的代码段。...

    java中线程同步方法

    Java提供了多种机制来实现线程同步,包括synchronized关键字、wait/notify方法、atomic操作、Lock接口以及Volatile变量等。本文将深入探讨这些同步方法及其应用场景。 #### 同步问题的提出 线程同步问题源于多线程...

    java线程.pdf

    Java提供了多种机制来实现线程间的同步,包括`synchronized`关键字、显式锁(`ReentrantLock`)、`volatile`关键字等。 #### synchronized关键字 `synchronized`可以用于修饰方法或代码块,保证在同一时刻只有一个...

Global site tag (gtag.js) - Google Analytics