java的多线程之入门
一.java多线程基本概念
调用run():在主线程调用子线程的run()方法会中断主线程等到子线程执行完毕之后再执行主线程。
调用start():在主线程中执行子线程的start()后会与主线程同步执行。
二.线程创建的三种方法
Thread、Runnable、Callable
1.Thread入门
步骤:
- 继承Thread类
- 重写run方法
- 调用start开启线程
//创建线程方式一步骤:继承Thread类,重写run方法,调用start开启线程
public class test01 extends Thread{ @Override
public void run() {
for (int i = 0; i < 10;i++){
System.out.println("run方法执行..." + i);
}
} public static void main(String[] args) { //创建线程对象
test01 t1 = new test01(); //开启线程
t1.start(); for (int i = 0; i < 1000; i++){
System.out.println("main方法执行..." + i);
}
}
}
执行结果:交替执行
练习:使用Thread实现多线程同步下载图片
(1).导入依赖的jar包:
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.6</version>
</dependency>
(2).编写test02.java:
/**
* 使用多线程下载图片
* @author USTC_WZH
* @create 2019-12-05 11:26
*/
public class test02 extends Thread { //图片url地址
private String url;
//保存的文件名
private String name; public test02(String url, String name) {
this.url = url;
this.name = name;
} @Override
public void run() {
WebDownloader webDownloader = new WebDownloader();
webDownloader.downloader(url, name);
System.out.println("下载文件名:" + name);
} public static void main(String[] args) {
test02 t1 = new test02("http://lswhw.ustc.edu.cn/upload/20150427/20150427102635.jpg", "images/图片1.jpg");
test02 t2 = new test02("http://lswhw.ustc.edu.cn/upload/20150427/20150427103657.jpg", "images/图片2.jpg");
test02 t3 = new test02("http://lswhw.ustc.edu.cn/upload/20150427/20150427103010.jpg", "images/图片3.jpg"); t1.start();
t2.start();
t3.start();
}
} class WebDownloader { public void downloader(String url, String name) {
try {
FileUtils.copyURLToFile(new URL(url), new File(name));
} catch (IOException e) {
e.printStackTrace();
System.out.println("downloader方法出现异常");
}
}
}
2.实现Runable接口(重点使用)
三个步骤:
- 实现Runnable接口,重写run方法
- 执行线程需要丢入Runnable接口的实现类
- 调用start方法
/**
* 创建线程方法二步骤:
* 1.实现Runnable接口,重写run方法。
* 2.执行线程需要丢入runnable接口的实现类。
* 3.调用start方法。
* @author USTC_WZH
* @create 2019-12-05 14:14
*/
public class test03 implements Runnable{ @Override
public void run() {
for (int i = 0; i < 10;i++){
System.out.println("run方法执行..." + i);
}
} public static void main(String[] args) { //创建runnable接口的实现类对象
test01 t1 = new test01(); //创建线程对象,通过线程对象来开启我们的线程,代理模式
new Thread(t1).start(); for (int i = 0; i < 1000; i++){
System.out.println("main方法执行..." + i);
}
}
}
补充:模拟并发问题
/**
* 模拟买火车票
*
* @author USTC_WZH
* @create 2019-12-05 14:32
*/
public class testerror implements Runnable { //火车票个数
private int ticketNums = 10; @Override
public void run() { while (true) { if (ticketNums <= 0) {
break;
} //模拟延时
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println(Thread.currentThread().getName() + "-->拿了第" + ticketNums-- + "票");
}
} public static void main(String[] args) { testerror t = new testerror(); new Thread(t, "小明").start();
new Thread(t, "小红").start();
new Thread(t, "花花").start();
}
}
当同一个对象被多个线程访问时出现的并发错误的问题!
3.实现Callable接口(了解)
步骤:7步
- 实现Callable接口,需要返回值类型
- 重写call方法,需要抛出异常
- 创建目标对象
- 创建执行服务
- 提交执行
- 获取结果
- 关闭服务
package ustc.wzh.callable; import org.apache.commons.io.FileUtils; import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.concurrent.*; /**
* @author USTC_WZH
* @create 2019-12-05 14:44
* <p>
* 创建线程方式三步骤:
* 1.实现Callable接口,需要返回值类型
* 2.重写call方法,需要抛出异常
* 3.创建目标对象
* 4.创建执行服务
* 5.提交执行
* 6.获取结果
* 7.关闭服务
*/
//1.实现Callable接口
public class testCallable implements Callable<Boolean> { //图片url地址
private String url;
//保存的文件名
private String name; public testCallable(String url, String name) {
this.url = url;
this.name = name;
} //2.重写call方法
@Override
public Boolean call() throws Exception {
WebDownloader webDownloader = new WebDownloader();
webDownloader.downloader(url, name);
System.out.println("下载文件名:" + name);
return true;
} public static void main(String[] args) throws ExecutionException, InterruptedException { //创建目标对象
testCallable t1 = new testCallable("http://lswhw.ustc.edu.cn/upload/20150427/20150427102635.jpg", "images/图片1.jpg");
testCallable t2 = new testCallable("http://lswhw.ustc.edu.cn/upload/20150427/20150427103657.jpg", "images/图片2.jpg");
testCallable t3 = new testCallable("http://lswhw.ustc.edu.cn/upload/20150427/20150427103010.jpg", "images/图片3.jpg"); //4.创建执行服务
ExecutorService ser = Executors.newFixedThreadPool(3); //5.提交执行
Future<Boolean> r1 = ser.submit(t1);
Future<Boolean> r2 = ser.submit(t2);
Future<Boolean> r3 = ser.submit(t3); //6.获取结果
boolean rs1 = r1.get();
boolean rs2 = r2.get();
boolean rs3 = r3.get(); System.out.println("rs1=" + rs1 + ",rs2=" + rs2 + ",rs3=" + rs3); //7.关闭服务
ser.shutdownNow();
} } class WebDownloader { public void downloader(String url, String name) {
try {
FileUtils.copyURLToFile(new URL(url), new File(name));
} catch (IOException e) {
e.printStackTrace();
System.out.println("downloader方法出现异常");
}
}
}
java的多线程之入门的更多相关文章
- Java的多线程 简单入门
Java的多线程 简单入门 首先能够先搞清楚什么是程序.进程.线程,以及它们之间的关系: 定义: 一 程序仅仅是一组指令的有序集合.它是静态的 二 进程是具有一定独立功能的程序关于某个数据集合上的一次 ...
- Java 高级 --- 多线程快速入门
这世上有三样东西是别人抢不走的:一是吃进胃里的食物,二是藏在心中的梦想,三是读进大脑的书 多线程快速入门 1.线程与进程区别 每个正在系统上运行的程序都是一个进程.每个进程包含一到多个线程.线程是一组 ...
- 【java】-- 多线程快速入门
1.什么是进程?什么是线程?两者区别? 1.每个正在系统上运行的程序都是一个进程,每个进程包含一到多个线程,多线程处理就是允许一个进程中在同一时刻执行多个任务. 2.线程是一组指令的集合,或者是程序的 ...
- 微博,and java 多线程编程 入门到精通 将cpu 的那个 张振华
http://down.51cto.com/data/2263476 java 多线程编程 入门到精通 将cpu 的那个 张振华 多个用户可以同时用一个 vhost,但是vhost之间是隔离的. ...
- java 多线程 快速入门
------------恢复内容开始------------ java 多线程 快速入门 1. 进程和线程 什么是进程? 进程是正在运行的程序它是线程的集合 进程中一定有一个主线程 一个操作系统可以有 ...
- 【Java】多线程入门
Java多线程学习(入门) 前言 目前对于线程的了解仅仅停留在学习python的threading库,很多线程的概念没有真正弄清楚,所以选择来系统性的学习多线程.那么这次选择的是Java的多线程学习, ...
- Java程序员快速入门Go语言
这篇文章帮助Java程序员快速入门Go语言. 转载至 开源中国社区. http://www.oschina.net 本文将以一个有代表性的例子为开始,以此让Java程序员对Go语言有个初步认识,随后将 ...
- Java工程师学习指南 入门篇
Java工程师学习指南 入门篇 最近有很多小伙伴来问我,Java小白如何入门,如何安排学习路线,每一步应该怎么走比较好.原本我以为之前的几篇文章已经可以解决大家的问题了,其实不然,因为我之前写的文章都 ...
- Java的多线程机制系列:不得不提的volatile及指令重排序(happen-before)
一.不得不提的volatile volatile是个很老的关键字,几乎伴随着JDK的诞生而诞生,我们都知道这个关键字,但又不太清楚什么时候会使用它:我们在JDK及开源框架中随处可见这个关键字,但并发专 ...
随机推荐
- 客观->感官->意识->语言->思维->世界观、科学->思想
客观->感官->意识->语言->思维->世界观.科学->思想
- Flink基本的API
Flink使用 DataSet 和 DataStream 代表数据集.DateSet 用于批处理,代表数据是有限的:而 DataStream 用于流数据,代表数据是无界的.数据集中的数据是不可以变的, ...
- 关于 返回数据类型 后 加& 的作用
关于 返回数据类型 后 加& 的作用 请看以下图片 我的测试 //纯粹就是 为了 可作为 左值而存在的 方法 前 的返回值 int& ,如 int& Method(); // ...
- c#结束练习题
1.输入一个秒数,输出对应的小时.分钟.秒. 例:输入“4000“(秒),输出“1小时6分40秒”. 2.计算1-1/2+1/3-1/4+...-1/100的值. 3.写一个函数,对一个一维数组排序. ...
- a属性+DOM创建回流+动画运动+
超链接a的属性 href分析: < a href = " " > 点击刷新页面,相当于向后台发送了一次请求 < a href = " # &quo ...
- Myeclipse中JSP镶嵌的html报错
Window > perferences > General > Editors > File Associations > 在File types 中选择 *.jsp ...
- vue-cli项目中使用vw——相比flexible更原生的移动端解决方案
安装命令行输入: yarn add postcss-px-to-viewport 或 npm i postcss-px-to-viewport -save -dev 配置package.json中,在 ...
- 鼠标按下改变RelativeLayout背景颜色,松开变回
在drawable下创建bg.xml文件 <?xml version="1.0" encoding="utf-8"?> <selector x ...
- vue数组更新界面无变化
1. vue数组更新界面无变化 1.1. 说明 对数组进行更新或者添加,一定要注意方式,我的情况是数组套数组,双重循环,在造数据的时候,不断从尾部添加数据,所以写成了如下形式,每次下拉都会去加载一批相 ...
- Python 序列、列表(List)、元组(Tuple)
序列 序列是Python中最基本的数据结构,包括字符串.列表.元组. 序列,顾名思义,是有序的,序列都有索引,都能进行索引.切片(截取).加(连接).乘(倍增).检查成员的操作. 因为序列有序,可通过 ...