进程和线程(和协程)

通俗的讲, 进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单位(比如QQ是个进程、微信是个进程)

进程和线程的主要差别在于它们是不同的操作系统资源管理方式。

进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单位。每个进程都有自己的独立内存空间,不同进程通过进程间通信来通信。由于进程比较重量,占据独立的内存,所以上下文进程间的切换开销(栈、寄存器、虚拟内存、文件句柄等)比较大,但相对比较稳定安全。

线程是进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.线程自己基本上不拥有系统资源,只拥有一点在运行中必不可少的资源(如程序计数器,一组寄存器和栈),但是它可与同属一个进程的其他的线程共享进程所拥有的全部资源。线程间通信主要通过共享内存,上下文切换很快,资源开销较少,但相比进程不够稳定容易丢失数据。

**协程是一种用户态的轻量级线程**,协程的调度完全由用户控制。协程拥有自己的寄存器上下文和栈。协程调度切换时,将寄存器上下文和栈保存到其他地方,在切回来的时候,恢复先前保存的寄存器上下文和栈,直接操作栈则基本没有内核切换的开销,可以不加锁的访问全局变量,所以上下文的切换非常快。

在Java中使用协程,一般会用到kilim( https://github.com/kilim/kilim )这个框架。因此本文不做讨论

主线程与子线程之间的关系

1、最常见的情况,主线程中开启了一个子线程,开启之后,主线程与子线程互不影响各自的生命周期,即主线程结束,子线程还可以继续执行;子线程结束,主线程也能继续执行。

/**
* @author fangshixiang@vipkid.com.cn
* @description
* @date 2018-11-05 14:32
*/
public class Main {
public static void main(String[] args) {
System.out.println("主线程启动。。。。");
Thread thread = new Thread(new ChildThread());
thread.start();
System.out.println("主线程结束。。。。");
}
}
class ChildThread implements Runnable {
@Override
public void run() {
try {
System.out.println("子线程启动。。。。");
Thread.sleep(5000);
System.out.println("子线程结束。。。。");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出:

主线程启动。。。。

主线程结束。。。。

子线程启动。。。。

子线程结束。。。。

2、主线程开启了子线程,但是主线程结束,子线程也随之结束

thiredThread.setDaemon(true);

只需要把子线程设置为守护线程即可

3、主线程开启了一个子线程,主线程必须要等子线程运行完之后,才能结束主线程

thread.join();

只需要把子线程自动后,join一下就行了

这里使用了join()方法,让主线程等待子线程结束,然后主线程继续执行。这里join()方法必须要在子线程启动之后,再调用。

通信

子线程与主线程的通信,一般在Android中使用,因此本文也不作为重点进行讨论

总结

进程是资源分配的基本单位,线程是cpu调度的基本单位。

但是上面通过调用JVM提供的接口,例如setDaemon与join改变了主线程与子线程的关系,这些应该是JVM接口代码做了处理干扰了线程的生命周期。

对于cpu来说,其实不存在主线程和子线程之分,都是线程(JVM提供的API可能会改变线程的生命周期)。

守护线程与非守护线程本质上没什么区别,但是如果虚拟机中存活的线程都是守护线程的时候,虚拟机就会退出,只要虚拟机中还有一个非守护线程,虚拟机就不会退出。