`

多线程常用方法以及对锁的控制

阅读更多
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytp16

1.sleep()
    使当前线程(即调用该方法的线程)暂停执行一段时间,让其他线程有机会继续执行,但它并不释放对象锁。也就是如果有Synchronized同步块,其他线程仍然不同访问共享数据。注意该方法要捕获异常
    比如有两个线程同时执行(没有Synchronized),一个线程优先级为MAX_PRIORITY,另一个为MIN_PRIORITY,如果没有Sleep()方法,只有高优先级的线程执行完成后,低优先级的线程才能执行;但当高优先级的线程sleep(5000)后,低优先级就有机会执行了。
    总之,sleep()可以使低优先级的线程得到执行的机会,当然也可以让同优先级、高优先级的线程有执行的机会。

2.join()
    join()方法使调用该方法的线程在此之前执行完毕,也就是等待调用该方法的线程执行完毕后再往下继续执行。注意该方法也要捕获异常。

3.yield()
    它与sleep()类似,只是不能由用户指定暂停多长时间,并且yield()方法只能让同优先级的线程有执行的机会。

4.wait()和notify()、notifyAll()
    这三个方法用于协调多个线程对共享数据的存取,所以必须在Synchronized语句块内使用这三个方法。前面说过Synchronized这个关键字用于保护共享数据,阻止其他线程对共享数据的存取。但是这样程序的流程就很不灵活了,如何才能在当前线程还没退出Synchronized数据块时让其他线程也有机会访问共享数据呢?此时就用这三个方法来灵活控制。

    wait()方法使当前线程暂停执行并释放对象锁标志,让其他线程可以进入Synchronized数据块,当前线程被放入对象等待池中。当调用 notify()方法后,将从对象的等待池中移走一个任意的线程并放到锁标志等待池中,只有锁标志等待池中的线程能够获取锁标志;如果锁标志等待池中没有线程,则notify()不起作用。

    notifyAll()则从对象等待池中移走所有等待那个对象的线程并放到锁标志等待池中。
    注意 这三个方法都是java.lang.Ojbect的方法!

2.run()和start()
    这两个方法应该都比较熟悉,把需要并行处理的代码放在run()方法中,start()方法启动线程将自动调用 run()方法,这是由Java的内存机制规定的。并且run()方法必须是public访问权限,返回值类型为void。

3.关键字Synchronized
    这个关键字用于保护共享数据,当然前提是要分清哪些数据是共享数据。每个对象都有一个锁标志,当一个线程访问该对象时,被Synchronized修饰的数据将被“上锁”,阻止其他线程访问。当前线程访问完这部分数据后释放锁标志,其他线程就可以访问了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public ThreadTest implements Runnable{
    public synchronized void run(){
        for(int i=0;i<10;i++){
            System.out.println(" " + i);
        }
    }
    public static void main(String[] args) {
        Runnable r1 = new ThreadTest();
        Runnable r2 = new ThreadTest();
        Thread t1 = new Thread(r1);
        Thread t2 = new Thread(r2);
        t1.start();
        t2.start();
    }
}
 
    //以上这段程序中的 i 变量并不是共享数据,也就是这里的Synchronized关键字并未起作用。因为t1,t2两个线程是两个对象(r1,r2)的线程。//不同的对象其数据是不同的,所以r1和r2两个对象的i变量是并不是共享数据。
    当把代码改成如下:Synchronized关键字才会起作用
    Runnable r = new ThreadTest();
    Thread t1 = new Thread(r);
    Thread t2 = new Thread(r);
    t1.start();
    t2.start();
wait()和notify(),notifyAll()是Object类的方法,sleep()和yield()是Thread类的方法。
(1).常用的wait方法有wait()和wait(long timeout):
    void wait() 在其他线程调用此对象的 notify() 方法或 notifyAll() 方法前,导致当前线程等待。
    void wait(long timeout) 在其他线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量前,导致当前线程等待。
    wait()后,线程会释放掉它所占有的“锁标志”,从而使线程所在对象中的其它synchronized数据可被别的线程使用。
    wait()和notify()因为会对对象的“锁标志”进行操作,所以它们必须在synchronized函数或synchronized  block中进行调用。如果在non-synchronized函数或non-synchronized block中进行调用,虽然能编译通过,但在运 行时会发生IllegalMonitorStateException的异常。
(2).Thread.sleep(long millis),必须带有一个时间参数。
    sleep(long)使当前线程进入停滞状态,所以执行sleep()的线程在指定的时间内肯定不会被执行;
    sleep(long)可使优先级低的线程得到执行的机会,当然也可以让同优先级和高优先级的线程有执行的机会;
    sleep(long)是不会释放锁标志的。
(3).yield()没有参数。
    sleep 方法使当前运行中的线程睡眼一段时间,进入不可运行状态,这段时间的长短是由程序设定的,yield 方法使当前线程让出CPU占有权,但让出的时间是不可设定的。
    yield()也不会释放锁标志。
    实际上,yield()方法对应了如下操作: 先检测当前是否有相同优先级的线程处于同可运行状态,如有,则把 CPU 的占有权交给此线程,否则继续运行原来的线程。所以yield()方法称为“退让”,它把运行机会让给了同等优先级的其他线程。
    sleep方法允许较低优先级的线程获得运行机会,但yield()方法执行时,当前线程仍处在可运行状态,所以不可能让出较低优先级的线程些时获得CPU占有权。 在一个运行系统中,如果较高优先级的线程没有调用 sleep 方法,又没有受到 I/O阻塞,那么较低优先级线程只能等待所有较高优先级的线程运行结束,才有机会运行。
    yield()只是使当前线程重新回到可执行状态,所以执行yield()的线程有可能在进入到可执行状态后马上又被执行。所以yield()只能使同优先级的线程有执行的机会。
分享到:
评论

相关推荐

    关于多线程常用方法以及对锁的控制(详解)

    多线程常用方法以及对锁的控制详解 多线程编程是Java编程中的一种重要技术,能够提高程序的执行效率和响应速度。但是,多线程编程也带来了许多挑战,如线程安全、锁的控制等。本文将详细介绍多线程常用方法以及对锁...

    java 多线程 队列工厂

    Java并发库(`java.util.concurrent`包)提供了一系列优化的线程安全集合,如`ConcurrentLinkedQueue`、`ConcurrentHashMap`等,它们在多线程环境下性能优越,减少了对锁的依赖。 在`queueFactory`压缩包中,可能...

    java+多线程+同步详解源代码学习

    - 分析源码,了解不同线程创建和控制方法的实现。 - 查看同步机制的实际应用,如synchronized和Lock接口的使用。 - 学习线程通信技巧,如wait/notify机制和Condition接口的应用。 通过实际操作这些代码,你将更...

    Java多线程同步问题的探究.pdf

    在探究Java多线程同步问题时,需要关注的关键知识点包括synchronized关键字的使用、JDK 5引入的java.util.concurrent.locks包下的锁机制、以及JDK 1.2中ThreadLocal类的使用。此外,了解JVM(Java虚拟机)在处理多...

    高并发多线程面试专题及答案(上).pdf

    以上内容涵盖了Java中高并发和多线程面试可能遇到的问题,包括Synchronized关键字的使用、原理、锁定对象、可重入性、JVM对锁的优化、Synchronized的公平性和非公平性、以及锁消除和锁粗化的概念。掌握这些知识点...

    多线程编程中的数据竞争检测.pptx

    ### 多线程编程中的数据竞争检测 #### 线程安全概念与数据竞争定义 在多线程编程中,**线程安全**是指代码在并发执行时能够正确处理资源共享,确保不会出现数据竞争或其他不可预测的行为。实现线程安全的方法通常...

    一线大厂Java多线程面试120题.pdf

    Java多线程是Java开发中的核心技能之一,尤其在面试中,对于一线大厂的面试者来说,深入理解和掌握多线程的相关知识点至关重要。以下是一些关键的Java多线程面试知识点: 1. **自旋锁**:自旋锁是一种等待机制,当...

    Java Core Sprout:基础、并发、算法

    多线程中的常见问题 同步关键字原理 多线程的三大核心 对锁的一些认知 ReentrantLock实现原理 ConcurrentHashMap 的实现原理 如何优雅地使用和理解线程池 深入理解线程通信 一个线程召集的诡异事件 线程池中你不可...

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

    在iOS开发中,多线程同步是保证代码安全的关键技术之一。`@synchronized`关键字是Apple提供的一种简便的同步机制,它可以帮助开发者确保在多线程环境下对共享资源进行原子性操作,防止数据竞争问题。本文将深入探讨`...

    Python多线程中阻塞(join)与锁(Lock)使用误区解析

    在Python中,多线程编程是一种提升程序执行效率的常用方式,尤其是在I/O密集型任务中。然而,在使用Python的多线程编程中,开发者可能会遇到一些常见的误区。本文重点解析了两个常见概念:线程阻塞(join)和线程锁...

    Java高并发实战_java高并发_高并发_

    第1章主要介绍了并行计算中相关的 些基本概念, 树立读者对并行计算的基本认识;...第8章介绍了使用Eclipse进行多线程调试的方法, 并演示了通过Eclipse进行多线程调试重现ArrayList的线程不安全问题。

    线程的监控任务demo

    4. **并发控制**:在多线程环境中,数据同步和竞态条件是常见的问题。监控任务可能包含对锁、信号量、条件变量等同步机制的检查。 5. **日志与调试**:为了分析线程的行为,开发者通常会记录线程相关的日志信息,...

    线程的使用心得

    它主要有两种用法:作为方法修饰符以及作为代码块的锁。 - **作为方法修饰符**:当一个对象的方法被`synchronized`修饰时,该方法在同一时刻只能被一个线程访问。如果多个线程同时请求访问,则其他线程必须等待当前...

    java 多线程死锁详解及简单实例

    Java多线程死锁是并发编程中一个严重的问题,它发生在两个或更多个线程相互等待对方持有的资源而...因此,在编写多线程程序时,应当对锁的使用格外谨慎,尽可能减少死锁的可能性,并且在必要时使用死锁检测和恢复机制。

    基于改进锁相环和谐振控制的分布式电源逆变器谐波抑制方法.pdf

    仿真结果表明,相比于传统控制方法,采用改进锁相环和谐振控制方法后,q轴电压在谐波背景下有所减小,频率精度得到提高,电流总谐波失真也减少了近20%。这表明所提出的控制算法能够有效抑制谐波,改善逆变器输出电能...

    实战JAVA虚拟机++JVM故障诊断与性能优化.pdf

    第8章介绍了Java虚拟机对多线程,尤其是对锁的支持。第9~10章介绍了Java虚拟机的核心——Class文件结构,以及Java虚拟机中类的装载系统。第11章介绍了Java虚拟机的执行系统和字节码,并给出了通过ASM框架进行字节码...

    实战JAVA虚拟机

    第8章介绍了Java虚拟机对多线程,尤其是对锁的支持。第9~10章介绍了Java虚拟机的核心——Class文件结构,以及Java虚拟机中类的装载系统。第11章介绍了Java虚拟机的执行系统和字节码,并给出了通过ASM框架进行字节码...

    实战java虚拟机

    第8章介绍了Java虚拟机对多线程,尤其是对锁的支持。第9~10章介绍了Java虚拟机的核心——Class文件结构,以及Java虚拟机中类的装载系统。第11章介绍了Java虚拟机的执行系统和字节码,并给出了通过ASM框架进行字节码...

    实战Java虚拟机——JVM故障诊断与性能优化

    第8章介绍了Java虚拟机对多线程,尤其是对锁的支持。第9~10章介绍了Java虚拟机的核心——Class文件结构,以及Java虚拟机中类的装载系统。第11章介绍了Java虚拟机的执行系统和字节码,并给出了通过ASM框架进行字节码...

    免费开源!!Java Core Sprout:基础、并发、算法

    常用集合 数组列表/向量 链表 哈希映射 哈希集 链接哈希映射 Java多线程 多线程中的常见问题 同步关键字原理 多线程的三大核心 对锁的一些认知 ReentrantLock实现原理 ConcurrentHashMap 的实现原理 如何优雅地使用...

Global site tag (gtag.js) - Google Analytics