多线程详解
-
普通方法调用和多线程
-
Process与Thread
- 说起进程,就不得不说下程序。程序是指令和数据的有序集合,其本身没有任何运行的含义,是一个静态的概念。
- 而进程则是执行程序的一次执行过程,它是一个动态的概念。是系统资源分配的单位。
- 通常在一个进程中可以包含若干个线程,当然一个进程中至少有一个线程,不然没有存在的意义。线程是CPU调度和执行的的单位。
注意:很多多线程是模拟出来的,真正的多线程是指有多个CPU,即多核,如服务器。如果是模拟出来的多线程,即在一个CPU的情况下,在同一个时间点,CPU只能执行一个代码,因为切换的很快,所以就有同事执行的错局。
核心概念
- 线程就是独立的执行路径
- 在程序运行时,即使没有自己创建线程,后台也会有多个线程,如主线程,gc线程
- main()称之为主线程,为系统的入口,用于执行整个程序
- 在一个进程中,如果开辟了多个线程,线程的运行由调度器安排调度,调度器是与操作系统紧密相关的,先后顺序是不能认为的干预的。
- 对同一份资源操作时,会存在资源抢夺的问题,需要加入并发控制
- 线程会带来额外的开销,如cpu调度时间,并发控制开销
- 每个线程在自己的工作内存交互,内存控制不当会造成数据不一致
线程实现(重点)
线程创建
- Thread class:继承Thread类(重点)
- Runnable接口:实现Runnable接口(重点)
- Callable接口:实现Callable接口(了解)
Thread
- 自定义线程类继承Thread类
- 重写run()方法,编写线程执行体
- 创建线程对象,调用start()方法启动线程
// 创建线程方式一:继承Thread类,重写run()方法,调用start开启线程
// 总结:注意,线程开启不一定立即执行,由CPU调度执行
public class ThreadDemo extends Thread{
@Override
public void run() {
// run方法线程体
for (int i = 0; i < 200; i++) {
System.out.println("我在学习"+i);
}
}
public static void main(String[] args) {
// main线程,主线程
// 创建一个线程对象
ThreadDemo t = new ThreadDemo();
// 调用start方法开启线程
t.start();
for (int i = 0; i < 1000; i++) {
System.out.println("我在打代码---"+i);
}
}
}
练习(下载图片)——需要先下载commons-io放到项目路径下
// 实现多线程下载图片
public class TestThread2 extends Thread{
private String url;// 网络图片地址
private String name;// 保存的文件名
public TestThread2(String url,String name){
this.url = url;
this.name = name;
}
@Override
public void run() {
WebDownloader wd = new WebDownloader();
wd.downloader(url,name);
System.out.println("下载了文件名为:"+name);
}
public static void main(String[] args) {
// 这里url随便三张图的网址路径
TestThread2 t1 = new TestThread2("https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=1239696591,2099251333&fm=26&gp=0.jpg","1.jpg");
TestThread2 t2 = new TestThread2("https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=1271607131,1035620186&fm=26&gp=0.jpg","2.jpg");
TestThread2 t3 = new TestThread2("https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=112978412,2183174438&fm=26&gp=0.jpg","3.jpg");
t1.start();
t2.start();
t3.start();
}
}
// 下载器
class WebDownloader{
// 下载方法
public void downloader(String url,String file){
try {
FileUtils.copyURLToFile(new URL(url),new File(file));
} catch (IOException e) {
e.printStackTrace();
System.out.println("IO异常,downloader方法出现问题");
}
}
}
以上,可以看出,多线程是同步执行的,那个图片小先下载完就输出。并不是按照从上到下来执行。