目录:
前言
1.进程与线程的区别?
2.进程是操作系统进行资源分配的基本单位,而操作系统是以线程为单位进行调度的。
3. Java操作多线程,依赖最核心的类Thread。
4.关于start和run的区别?
5.使用JDK自带的工具jconsole来查看当前java进程中的所有线程。
Java中创建线程的五种写法
1.继承Thread,重写run
2.实现Runnable接口
3.使用匿名内部类,继承Thread
4.使用匿名内部类,实现Runable
5.使用Lambda表达式
前言
进程与线程的区别?
因为CPU现在进入了多核心的时代,要想进一步提高程序的执行速度,就需要充分的利用CPU的多核资源。引入进程这个概念,最主要的目的就是为了解决并发编程这样的问题。其实多进程编程已经可以解决并发编程的问题了,但是进程太重了,在资源分配和回收上的消耗资源多且速度比较慢,创建销毁调度进程的开销比较大。这个时候就引入了线程的概念,线程也可以叫做“轻量级进程”,在解决并发编程的前提下,让创建销毁调度的速度更快一些。因为线程是把资源的申请和释放的操作给省略了,所以更轻更快。
进程和线程的关系:进程包含线程,一个进程可以包含一个线程,也可以包含多个线程。只有第一个线程启动的时候,开销与创建进程相当,但是后续线程就省事了。同一个进程里面的多个线程之间是共用了进程的同一份资源,比如内存(线程1 new的对象在线程234中都可以直接使用)和文件描述符表(线程1打开的文件在线程234中都可以直接使用)。
进程是操作系统进行资源分配的基本单位,而操作系统是以线程为单位进行调度的。
关于进程的调度,是每个进程里面只有一个线程这样的情况,如果每个进程里面有多个线程了,每个线程是独立在CPU上调度的。其中一个线程也是通过一个PCB来描述的,一个进程里面可能是对应一个PCB,也可能是对应多个PCB。PCB的状态、上下文、优先级和记账信息都是每个线程有自己的,各自记录各自的。但是同一个进程里面的PCB之间的pid是一样的,内存指针和文件描述符表也是一样的(在Linux中不区分TCP和PCB,都用一个表示)。
关于多线程问题,当线程足够多,而系统资源有限的情况下,可能会引发线程安全问题;还有就是其中一个线程出现异常,那么很可能把整个进程都带走,其他线程也就跟着凉凉了(我们看到谷歌浏览器就是每个页面都是做的一个进程,所以谷歌浏览器占用的系统资源挺多的)。
在Java中进行多线程编程,要依赖于操作系统提供的API。关于Java能够跨平台,其实是靠无数个不同版本的JVM支持的,比如windows系统实现了一个windows版本的JVM,Linux系统实现了一个Linux版本的JVM,Mac系统实现了Mac版本的JVM,这些不同的JVM内部封装了不同系统的API。
Java操作多线程,依赖最核心的类Thread。
关于t1.start;是线程中的特殊方法,启动一个线程。Start的工作就是创建一个新的线程,新的线程负责执行run()方法。Start就是调用了操作系统的API,通过操作系统的内核创建新线程的PCB,并且把要执行的指令交给到这个PCB,当PCB被调度到CPU上执行的时候,也就执行到了线程run方法中的代码了。
关于操作系统调度线程,是“抢占式执行”,具体哪一个线程先上哪一个线程后上是不确定的,取决于操作系统调度器具体的实现策略。
关于start和run的区别?
Start是真正在系统上创建一个线程,线程是一个独立的执行流。而run只是描述了线程要干的活是啥。如果直接在main方法中调用run,那么此时是没有创建新的线程的,全是main所在线程一个人干活。
我们可以使用JDK自带的工具jconsole来查看当前java进程中的所有线程。
Java中创建线程的五种写法
1.继承Thread,重写run
-
package threadtest;
-
class
MyThread1
extends
Thread{
-
@Override
-
public
void
run
() {
-
while (
true) {
-
System.out.println(
"hello thread");
-
try {
-
Thread.sleep(
1000);
-
}
catch (InterruptedException e) {
-
e.printStackTrace();
-
}
-
}
-
}
-
}
-
public
class
ThreadDemo2 {
-
public
static
void
main
(String[] args)
throws InterruptedException {
-
Thread
t1
=
new
MyThread1();
-
t1.start();
//注意:这里start并没有调用run方法,而是创建了一个新的线程,由新的线程来执行run方法。
-
while (
true) {
-
System.out.println(
"hello main方法");
-
Thread.sleep(
1000);
-
}
-
}
-
}
2.实现Runnable接口
-
//Runnable的作用是描述一个“要执行的任务”,run方法就是任务的执行细节
-
class
MyRunnable
implements
Runnable{
-
-
@Override
-
public
void
run
() {
-
System.out.println(
"hello thread");
-
}
-
}
-
public
class
ThreadDemo3 {
-
public
static
void
main
(String[] args) {
-
Runnable
runnable
=
new
MyRunnable();
-
Thread
t1
=
new
Thread(runnable);
-
t1.start();
-
}
-
}
这样的写法可以解耦合,目的就是让线程和线程需要干的活分开。
3.使用匿名内部类,继承Thread
-
public
class
ThreadDemo4 {
-
public
static
void
main
(String[] args) {
-
Thread
t
=
new
Thread(){
-
@Override
-
public
void
run
() {
-
System.out.println(
"hello thread");
-
}
-
};
-
t.start();
-
}
-
}
new Thread(){}创建了一个Thread子类,子类没有名字所以叫做匿名,创建了子类的实例,并且让t引用指向该实例。
4.使用匿名内部类,实现Runable
-
public
class
ThreadDemo5 {
-
public
static
void
main
(String[] args) {
-
Thread
t
=
new
Thread(
new
Runnable() {
-
@Override
-
public
void
run
() {
-
System.out.println(
"hello runnable");
-
}
-
});
-
t.start();
-
}
-
}
这个写法与第二个本质是相同的,只不过是把实现Runnable任务了交给匿名内部类。此处是创建了一个类,实现了Runnable,同时创建了类的实例,并且传给了Thread的构造方法。
5.使用Lambda表达式
-
public
class
ThreadDemo6 {
-
public
static
void
main
(String[] args) {
-
Thread
t
=
new
Thread(() -> {
-
System.out.println(
"hello lambda");
-
});
-
t.start();
-
}
-
}
把任务用Lambda来描述,直接把Lambda传给Thread方法。
转载:https://blog.csdn.net/crazy_xieyi/article/details/128378342