Java之进程与线程
- 说道进程与线程,就涉及到一个面试必备的问题:进程与线程的区别。下面先举个易懂的例子帮助大家理解。
开个QQ,开了一个进程;
在QQ的这个进程里,传输文字开一个线程、传输语音开了一个线程、弹出对话框又开了一个线程。
所以运行某个软件,相当于开了一个进程。在这个软件运行的过程里(在这个进程里),多个工作支撑的完成QQ的运行,那么这“多个工作”分别有一个线程。所以一个进程管着多个线程。通俗的讲:“进程是爹妈,管着众多的线程儿子”
作者:南理汉子
链接:https://www.zhihu.com/question/25532384/answer/131906596
来源:知乎
1.定义
进程:具有一定独立功能的程序关于某个数据集合上的一次运行活动,进程是系统进行资源分配和调度的一个独立单位.
线程:进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.线程自己基本上不拥有系统资源,只拥有一点在运行中必不可少的资源(如程序计数器,一组寄存器和栈),但是它可与同属一个进程的其他的线程共享进程所拥有的全部资源.
2.关系
一个线程可以创建和撤销另一个线程;同一个进程中的多个线程之间可以并发执行.
相对进程而言,线程是一个更加接近于执行体的概念,它可以与同进程中的其他线程共享数据,但拥有自己的栈空间,拥有独立的执行序列。
3.区别
进程和线程的主要差别在于它们是不同的操作系统资源管理方式。进程有独立的地址空间,一个进程崩溃后,在保护模式下不会对其它进程产生影响,而线程只是一个进程中的不同执行路径。线程有自己的堆栈和局部变量,但线程之间没有单独的地址空间,一个线程死掉就等于整个进程死掉,所以多进程的程序要比多线程的程序健壮,但在进程切换时,耗费资源较大,效率要差一些。但对于一些要求同时进行并且又要共享某些变量的并发操作,只能用线程,不能用进程。
1) 简而言之,一个程序至少有一个进程,一个进程至少有一个线程.
2) 线程的划分尺度小于进程,使得多线程程序的并发性高。
3) 另外,进程在执行过程中拥有独立的内存单元,而多个线程共享内存,从而极大地提高了程序的运行效率。
4) 线程在执行过程中与进程还是有区别的。每个独立的线程有一个程序运行的入口、顺序执行序列和程序的出口。但是线程不能够独立执行,必须依存在应用程序中,由应用程序提供多个线程执行控制。
5) 从逻辑角度来看,多线程的意义在于一个应用程序中,有多个执行部分可以同时执行。但操作系统并没有将多个线程看做多个独立的应用,来实现进程的调度和管理以及资源分配。这就是进程和线程的重要区别。
进程类Thread
- 实现进程类Thread有两种方法:
(1)子类继承父类Thread(不推荐此方法,因为Java是单继承)
(2)子类实现Runnable接口
方法一代码
package cn.com.Gyn;
public class TreadC {
public static void main(String[] args) {
/**
* 用匿名对象调用
*/
new MyThread().start();
// System.out.println("这里是主线程");
// MyThread myThread=new MyThread();
// myThread.start(); //start开始执行子线程方法
}
}
class MyThread extends Thread {
@Override
public void run() {
super.run();
for (int i = 0; i < 10; i++) {
System.out.println("子线程中i=" + i);
if (i == 5) {
try {
Thread.sleep(1000 * 5); // 睡眠5s(单位毫秒)
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
- 注:此代码中,主函数用了两种调用方法(1.用匿名对象调用,简单方便 2.注释掉的代码,为常规调用方法)
- 注:sleep()为线程类的一个方法,表示 在指定的毫秒数内让当前正在执行的线程休眠(暂停执行)
结果截图
- 休眠5秒后
方法二代码
package cn.com.Gyn;
/**
* 子进程之间会交叉执行
*/
public class TreadC2 {
public static void main(String[] args) {
System.out.println("这里是主线程");
RunnableImp1 r1 = new RunnableImp1();
Thread t1 = new Thread(r1);
t1.setName("线程一的i="); //设置名字
t1.start();
RunnableImp2 r2 = new RunnableImp2();
Thread t2 = new Thread(r2);
t2.setName("线程二的i=");
t2.start();
System.out.println("继续主线程");
}
}
class RunnableImp1 implements Runnable {
@Override
public void run() {
for (int i = 0; i < 50; i++) {
System.out.println(Thread.currentThread().getName() + i);
//获得名字
}
}
}
class RunnableImp2 implements Runnable {
@Override
public void run() {
for (int i = 0; i < 50; i++) {
System.out.println(Thread.currentThread().getName() + i);
}
}
}
- Thread.currentThread().getName() 为获得当前线程的名字
- 由结果可以发现线程进行变得无序,一会是线程一,一会是线程二,这是因为:线程的执行是无序的,受到CPU分配的资源决定。