美文网首页
7.进程、线程、同步和锁

7.进程、线程、同步和锁

作者: 芝麻酱的简书 | 来源:发表于2018-12-19 17:26 被阅读4次

进程Process

创建方式1:使用Runtime

    Runtime runtime = Runtime.getRuntime();
    runtime.exec("notepad");

创建方式2:使用ProcessBuilder

    ProcessBuilder builder = new ProcessBuilder("notepad");
    builder.start();


线程Thread

创建方式1:继承自Thread类

class PlayGames extends Thread {
    
}

// 开启
    PlayGames play = new PlayGames();
    play.start();  // 不要用run

创建方式2:实现Runable接口

class PlayGames implements Runnable{
    
    public void run(){
        
        // ...
    };
}

// 调用
    PlayGames imp = new  PlayGames();
    Thread t = new Thread(imp);
    t.start();

创建方式3:使用匿名内部类

    new Thread(new Runnable(){
        public void run(){
        }
        
    }).start();


同步代码块

同步锁(同步监听对象、同步锁、互斥锁)
语法: synchronized (同步锁){ 需要同步的代码 }

在同一时间只有一个线程可以拿到锁,其他线程需要等待



双重检查加锁volatile

使用关键字volatile,被该关键字修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能够正确处理改变量

final class AnimalTool {
    
    private AnimalTool(){};
    
    private static volatile AnimalTool instance = null;
    
    public static AnimalTool getInstance(){
        
        if (instance == null) {
            synchronized (instance) {
                if (instance == null) {
                    instance = new AnimalTool();
                }
            }
        }
        return instance;
    }
}


Lock同步锁

import java.util.concurrent.locks.*;

import static java.util.Arrays.*;

class AnimalTool {
    
    // 初始化Lock子类 重用锁对象
    private final ReentrantLock lock = new ReentrantLock();
    
    public void synchMethod (){
        lock.lock();
        
        try {
            // ...
        } catch (Exception e) {
            // ...
        } finally {
            lock.unlock();
        }
    }
}


线程通信

线程通信方式一:

wait和notify方法介绍:
java.lang.Object类提供类两类用于操作线程通信的方法.

wait():执行该方法的线程对象释放同步锁,JVM把该线程存放到等待池中,等待其他的线程唤醒该线程.

notify:执行该方法的线程唤醒在等待池中等待的任意一个线程,把线程转到锁池中等待.

notifyAll():执行该方法的线程唤醒在等待池中等待的所有的线程,把线程转到锁池中等待.

注意:上述方法只能被同步监听锁对象来调用,否则报错IllegalMonitorStateException..

假设A线程和B线程共同操作一个X对象(同步锁),A,B线程可以通过X对象的waitnotify方法来进行通信,流程如下:
  1. 当A线程执行X对象的同步方法时,A线程持有X对象的锁,B线程没有执行机会,B线程在X对象的锁池中等待.
  2. A线程在同步方法中执行X.wait()方法时,A线程释放X对象的锁,进入A线程进入X对象的等待池中.
  3. 在X对象的锁池中等待锁的B线程获取X对象的锁,执行X的另一个同步方法.
  4. B线程在同步方法中执行X.notify()方法时,JVM把A线程从X对象的等待池中移动到X对象的锁池中,等待获取锁.
  5. B线程执行完同步方法,释放锁.A线程获得锁,继续执行同步方法.
代码样例

线程通信-使用Lock和Condition接口:
wait和notify方法,只能被同步监听锁对象来调用,否则报错IllegalMonitorStateException.
那么现在问题来了,Lock机制根本就没有同步锁了,也就没有自动获取锁和自动释放锁的概念.
因为没有同步锁,所以Lock机制不能调用wait和notify方法.
解决方案:Java5中提供了Lock机制的同时提供了处理Lock机制的通信控制的Condition接口.

线程通信方式二:

从Java5开始,可以:

  1. 使用Lock机制取代synchronized 代码块和synchronized 方法.
  2. 使用Condition接口对象的await,signal,signalAll方法取代Object类中的wait,notify,notifyAll方法.
样例代码

线程的生命周期

线程的状态 另一种模型
  1. 新建状态(new):使用new创建一个线程对象,仅仅在堆中分配内存空间,在调用start方法之前.
    新建状态下,线程压根就没有启动,仅仅只是存在一个线程对象而已.
    Thread t = new Thread();//此时t就属于新建状态
    当新建状态下的线程对象调用了start方法,此时从新建状态进入可运行状态.
    线程对象的start方法只能调用一次,否则报错:IllegalThreadStateException.

  2. 可运行状态(runnable):分成两种状态,ready和running。分别表示就绪状态和运行状态。

  • 就绪状态:线程对象调用start方法之后,等待JVM的调度(此时该线程并没有运行).
  • 运行状态:线程对象获得JVM调度,如果存在多个CPU,那么允许多个线程并行运行.

3:阻塞状态(blocked):正在运行的线程因为某些原因放弃CPU,暂时停止运行,就会进入阻塞状态.
此时JVM不会给线程分配CPU,直到线程重新进入就绪状态,才有机会转到运行状态.
阻塞状态只能先进入就绪状态,不能直接进入运行状态.
阻塞状态的两种情况:

  • 当A线程处于运行过程时,试图获取同步锁时,却被B线程获取.此时JVM把当前A线程存到对象的锁池中,A线程进入阻塞状态.
  • 当线程处于运行过程时,发出了IO请求时,此时进入阻塞状态.

4:等待状态(waiting)(等待状态只能被其他线程唤醒):此时使用的无参数的wait方法,

  • 当线程处于运行过程时,调用了wait()方法,此时JVM把当前线程存在对象等待池中.

5:计时等待状态(timed waiting)(使用了带参数的wait方法或者sleep方法)

  • 当线程处于运行过程时,调用了wait(long time)方法,此时JVM把当前线程存在对象等待池中.
  • 当前线程执行了sleep(long time)方法.
  1. 终止状态(terminated):通常称为死亡状态,表示线程终止.
  • 正常执行完run方法而退出(正常死亡).
  • 遇到异常而退出(出现异常之后,程序就会中断)(意外死亡).

线程一旦终止,就不能再重启启动,否则报错(IllegalThreadStateException).
在Thread类中过时的方法(因为存在线程安全问题,所以弃用了:

// 如下弃用
 void suspend() :暂停当前线程
 void resume()  :恢复当前线程
 void stop()  :结束当前线程


线程控制

1.线程睡眠sleep

线程休眠:让执行的线程暂停一段时间,进入计时等待状态。
方法:static void sleep(long millis)
调用sleep后,当前线程放弃CPU,在指定时间段之内,sleep所在线程不会获得执行的机会。
此状态下的线程不会释放同步锁/同步监听器.
该方法更多的用于模拟网络延迟,让多线程并发访问同一个资源的错误效果更明显.
在开发中也会故意使用该方法,如:

    Thread.sleep(1000);
2.联合线程join

线程的join方法表示一个线程等待另一个线程完成后才执行。join方法被调用之后,线程对象处于阻塞状态。
有人也把这种方式称为联合线程,就是说把当前线程和当前线程所在的线程联合成一个线程。

联合线程 执行分析
3.后台线程(守护线程)

后台线程:在后台运行的线程,其目的是为其他线程提供服务,也称为“守护线程"。JVM的垃圾回收线程就是典型的后台线程。

特点:若所有的前台线程都死亡,后台线程自动死亡,前台线程没有结束,后台线程是不会结束的。

测试线程对象是否为后台线程:使用thread.isDaemon()

前台线程创建的线程默认是前台线程,可以通过setDaenon(true)方法设置为后台线程,并且当且仅当后台线程创建的新线程时,新线程是后台线程。
设置后台线程:thread.setDaemon(true),该方法必须在start方法调用前,否则出现IllegalThreadStateException异常。

后台线程
4.线程优先级

线程优先级:
每个线程都有优先级,优先级的高低只和线程获得执行机会的次数多少有关,并非线程优先级越高的就一定先执行,哪个线程的先运行取决于CPU的调度。
MAX_PRIORITY=10,最高优先级
MIN_PRIORITY=1,最低优先级
NORM_PRIORITY=5,默认优先级

 int getPriority() :返回线程的优先级。
 void setPriority(int newPriority) : 更改线程的优先级。

每个线程都有默认优先级,主线程默认优先级为5,如果A线程创建了B线程,那么B线程和A线程具有相同优先级.
注意:不同的操作系统支持的线程优先级不同的,建议使用上述三个优先级,不要自定义.

5.线程礼让

线程礼让:
yield方法:表示当前线程对象提示调度器自己愿意让出CPU资源,但是调度器可以自由的忽略该提示。
调用该方法之后,线程对象进入就绪状态,所以完全有可能:某个线程调用了yield()之后,线程调度器又把它调度出来重新执行。
从Java7提供的文档上可以清楚的看出,开发中很少会使用到该方法,该方法主要用于调试或测试,它可能有助于因多线程竞争条件下的错误重现现象。

sleep方法和yield方法的区别:

  • 都能使当前处于运行状态的线程放弃CPU,把运行的机会给其他线程.
  • sleep方法会给其他线程运行机会,但是不考虑其他线程的优先级,yield方法只会给相同优先级或者更高优先级的线程运行的机会.
  • 调用sleep方法后,线程进入计时等待状态,调用yield方法后,线程进入就绪状态.


定时器和线程组

在JDK的java.util包中提供了Timer类,可以定时执行特定的任务.
TimerTask类表示定时器执行的某一项任务.
常用方法:

schedule(TimerTask task,long delay,long period):
schedule(TimerTask task,long delay):

ThreadGroup类表示线程组,可以对一组线程进行集中管理.
用户在创建线程对象时,可以通过构造器指定其所属的线程组.
Thread(ThreadGroup group,String name);
如果A线程创建了B线程,如果没有设置B线程的分组,那么B线程加入到A线程的线程组.
一旦线程加入某个线程组,该线程就一直存在于该线程组中直到线程死亡,不能在中途修改线程的分组.

当Java程序运行时,JVM会创建名为main的线程组,在默认情况下,所有的线程都该改线程组下.

线程相关的知识点:
  1. 线程安全的集合对象
  2. ThreadLocal:本地线程变量,ThreadLocal模式,
  3. 线程池,Java8堆线程池新增的功能.
  4. Java5新增的线程并发操作.

相关文章

  • 7.进程、线程、同步和锁

    进程Process 创建方式1:使用Runtime 创建方式2:使用ProcessBuilder 线程Thread...

  • iOS多线程

    一.概念 1.进程 2.线程 3.进程和线程的关系 4.多线程 5.主线程 6.同步 7.异步 同步和异步关注的是...

  • python之多线程与多进程入门

    python之多线程与多进程 关键词: GIL锁,IO繁忙,线程安全,线程同步,进程池,进程通信,队列 GIL锁;...

  • 安卓编程技巧总结(3) 进程与线程处理

    多进程 线程的管理 线程使用 线程锁 进程使用 进程间通信时注意URI传递 7.带返回值的异步任务Callable...

  • 计算机基础-操作系统提升篇

    一,线程,进程同步 为什么需要同步 1.线程同步 以下都是对资源加锁 互斥量(互斥锁)逻辑开始加个锁,结束再解锁,...

  • multiprocessing sync

    同步 多进程使用的同步原语和线程的是类似的,可以通过锁来限制只能有一个进程在某时访问资源。

  • Java多线程——学习笔记一:进程与线程以及线程安全

    进程和线程 进程 资源分配的最小单位。 线程 程序执行的最小单位 进程和线程的关系 同步和异步 同步:对于方法调用...

  • 字节(抖音)后台开发提前批面经

    一面(7.18, 45min) 自我介绍线程和进程区别和联系线程同步方法自旋锁怎么实现堆区和栈区linux键入to...

  • 多线程入门

    重点: 线程的实现 线程的同步 Process(进程)和 Thread(线程) 进程中有线程,真正执行的是线程 ...

  • 线程 一

    目录 1.多线程 2.线程对比进程 3.线程间资源共享 4.守护线程 5.线程中常用属性 6.线程互斥锁 7.信号...

网友评论

      本文标题:7.进程、线程、同步和锁

      本文链接:https://www.haomeiwen.com/subject/dgujkqtx.html