进程:系统上面运行的一个程序就是一个进程。一个进程包含一到多个线程。
线程:可以看成是一组指令的集合。通常由操作系统来负责多个线程的调度喝执行
总结:进程是线程的集合体,每一个线程都是进程中的一条执行路径。
多线程的好处是可以有效的提高程序的运行效率。可以比作是有多个人合作完成原来一个人的工作。 多个线程互相不干扰,一个任务的崩溃不会影响到另外一个任务。
如:迅雷下载、后台任务、分布式计算。
public class MyThread extends Thread {
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println("run:i="+i);
}
}
}
创建线程完成之后可以通过start()方法来启动线程:
public class Test {
public static void main(String[] args) {
System.out.println("开始创建线程.........");
MyThread thread = new MyThread();
// 启动线程用start()方法,不能直接调用run()方法,若调用run()方法,线程不起作用
thread.start();
System.out.println("线程正在运行.........");
for (int i = 0; i < 100; i++) {
System.out.println("main:i="+i);
}
}
}
线程执行通过操作系统进行随机分配,多个线程交叉执行。例如该程序执行结果中:
...
main:i=51
run:i=76
main:i=52
run:i=77
run:i=78
main:i=53
run:i=79
run:i=80
main:i=54
main:i=55
main:i=56
...
public class MyThread2 implements Runnable{
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println("run:i="+i);
}
}
}
启动线程的方法与继承Thread类不同,因为实现Runnable的类中没有start()方法,所以直接调用该实例的start()是错误的,调用方法:
public class Test {
public static void main(String[] args) {
System.out.println("开始创建线程.........");
MyThread2 thread2 = new MyThread2();
new Thread(thread2).start();
System.out.println("线程正在运行.........");
for (int i = 0; i < 100; i++) {
System.out.println("main:i="+i);
}
}
}
一般使用中实现Runnable接口优于继承Thread类,因为Java单继承,但可以实现多个接口。
public class Test {
public static void main(String[] args) {
System.out.println("开始创建线程.........");
// 匿名内部类
new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println("run:i="+i);
}
}
}).start();
for (int i = 0; i < 100; i++) {
System.out.println("main:i="+i);
}
}
}
常用的线程api | |
---|---|
start() | 启动线程 |
currentThread() | 获取当前线程对象 |
getID() | 获取当前线程ID ID随机分配 |
getName() | 获取当前线程名称 Thread-编号 该编号从0开始 |
sleep(long second) | 休眠线程,参数为时间(毫秒) |
stop() | 停止线程 |
setDaemon(boolean flag) | 设置线程为守护线程 |
常用线程构造函数 | |
---|---|
Thread() | 分配一个新的Thread对象 |
Thread(String name) | 分配一个具有指定名称的的Thread对象 |
Thread(Runnable r) | 分配一个Thread对象(通过实现Runnable接口创建的) |
Thread(Runnable r,String name) | 分配一个指定名称的Thrad对象(通过实现Runnable接口创建的) |
在run()方法当中不能抛出异常,只能通过try catch来捕获异常
线程进入阻塞状态的原因:
守护线程:当进程不存在或者主线程停止,守护线程也随之停止。 非守护线程:一般指用户线程,与主线程互不影响。
同步:代码自上而下执行,上面的任务执行完成之后才可以执行下面的方法。 异步:多个任务并行执行,任务之间互不影响。
A线程中调用B线程的join()方法,表示A线程等待B线程死亡之后才开始执行。
public class TestJoin {
public static void main(String[] args) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 100; i++) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("子线程"+i);
}
}
});
thread.start();
System.out.println("====================");
try {
// 主线程调用join方法表示主线程释放自己的资源交给子线程,等子线程死亡之后再执行
thread.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
for (int i = 0; i < 100; i++) {
System.out.println("主线程"+i);
}
}
}
原文:https://www.cnblogs.com/kfw5264/p/12183404.html