java多线程简介
我们目前所写代码都是顺序往下执行,多个业务代码累加起来可能会非常耗时,并且拆分不易,比如同时读取数据库10个表的数据,我们假设每个表需要10s钟,那么一共需要100s钟;假如我们可以同时读取这十个表的数据,
在最理想情况下,一共也只需要花10s钟。
多线程编程就是利用多个线程,每个线程执行对应的业务代码,来达到分拆任务,加快总体执行速度的编程方法。
但是,多线程也存在其问题,那就是线程太多,线程之间来回切换也需要耗时;在对公共资源进行操作时,会经常遇到各种并发问题。
Java 本身具有多线程机制,即使不涉及到具体的多线程程序编写,也会无时不刻在用它 (比如在调用main方法时已经启动了一个主线程 )。Java 的线程机制是抢占式的 ,也就是说jvm 的调度机制会周期性的(时间极短)中断当
前某个线程,切换到另外一个线程去执行。
1 Thread和Runnable
在早期我们可以使用两种方式定义并启动线程:
继承Thread ,实现其 run方法,然后调用start方法启动线程,比如:
public class MyThread extends Thread {
public void run(){
System.out.println("执行线程 ");
}
}
public static void main(String[] args) {
MyThread mt=new MyThread();
//启动线程
mt.start();
}
注意:run方法体是实现具体业务代码的地方。
也可以让Runnable 和Thread配合使用, Runnable是一个描述任务的一个接口,而 Thread则用于启动这个任务,下面我们看看怎样使用它们编写一个线程程序,
首先必须创建一个类来实现 Runnable接口
public class MyRunnable implements Runnable {
@Override
public void run() {
System.out.println("runnable");
}
}
然后使用 Thread对象来启动:
Thread thread=new Thread(new MyRunnable());
thread.start();
2 线程常用API
9.3.1 Thread.sleep(2000)
该方法表示当前线程进入休眠状态,里面传入的参数表示将要休眠的时间,这个参数的单位是毫秒,也就是说此处应该休眠 2s,同理:Thread.sleep(3000) 表示休眠 3s
9.3.2 Thread.currentThread()
该方法得到当前线程对象,在调试的时候我们往往需要打印出当前的线程信息,就可以使用这个 API:
Thread thread=Thread.currentThread();
System.out.println(thread.getName());
getName() 方法可以打印出某个线程的名称
9.4 线程同步问题
我们通常会把某个具体的业务分拆成多个业务并行执行,而这些业务往往会涉及到某些共享资源,处理好这些资源信息的同步,是我们编写多线程并发程序的一大阻碍和难点。
那么,资源同步问题是怎样产生的?
1 线程执行顺序的不确定性
2 多个线程共享资源,并且在任务中对资源有更新和读取操作。
下面我们通过实现一个多线程累加数值的功能来讲解线程同步问题。
3.1 线程同步示例
首先我们编写一个累加数值的类,这个类有一个 count属性:
public class Const {
private int count;
public int getCount() {
return count;
}
public void setCount(int count) {
this.count = count;
}
public void increament(){
System.out.println("加之前 "+this.count);
count++;
System.out.println("加之后 "+this.count);
}
}
然后新建一个 Runnable实现类,来调用递增方法:
public class IncreamentRunnable implements Runnable {
private Const cnt;
public IncreamentRunnable(Const cnt){
this.cnt=cnt;
}
@Override
public void run() {
cnt.increament();
}
}
这个类需要传入上面定义过的 Const类的对象,然后在run方法中调用递增方法。
下面我们启动多个线程来执行它:
Const cnt=new Const();
for(int i=0;i<10;i++){
IncreamentRunnable ir=new IncreamentRunnable(cnt);
Thread thread2=new Thread(ir);
thread2.start();
}
注:这十个线程目前是公用的同一个 Const对象(共享资源)
我们会发现方法在执行的时候又被其他线程侵入并执行了,也就是说这个对象方法不是执行完成之后其他再执行的,而是同时都在执行,这就是所谓的线程不同步问题。
为了解决这个问题,我们可以使用 synchronized关键字使线程同步, synchronized既可以用在方法上,也可以作为方法体内的某个语句块。
方法体上:
public synchronized void increament()
代码块:
synchronized(this){}
使用代码块级别的锁的好处是,可以自己控制锁的粒度。
在方法上加上同步锁关键字 synchronized后,运行一下结果如下:
现在输出就正常了,每个方法都是执行完之后才会执行下一次。
3.2 wait(),notifyAll()方法
当线程得到某个对象锁时,可以调用此对象的 wait()来使当前线程进入等待状态;当另外的线程得到某个对象锁后,此对象假如调用 notifyAll()时,就会唤醒等待在此对象上的所有线程。
synchronized(obj){
obj.wait();
}
synchronized(obj){
obj.notifyAll();
}
注意:wait 和notifyAll都是 Object的方法,并且只能在锁内调用。
4 java5并发包的使用
Java5/6提供了并发包,可以简化很多多线程代码,并且提供更强劲,更细致的功能,并发包在 java.util.concurrent下面,它提供了非常简单的 API来实现线程池和细粒度的并发限制等功能。
在之前的多线程程序中,我们会给每一个任务分配一个线程,当线程的任务执行完毕后,这个线程就被销毁,这种做法在技术上没有什么问题,但是有如下缺点:
1 线程的创建和销毁是需要很大的开销的,有可能这个开销比执行任务的开销还要大
2 线程个数不固定,会随着任务的增多而增多,而每个线程本身也是需要消耗系统资源的,假如任务过多,服务器需要创建或维护大量的线程,可能会导致系统内存空间不足
而线程池就可以解决这些问题,线程池会预先创建多个工作线程,然后让他们不断的执行某些任务,当某个线程执行完一个任务时,就会继续执行下一个任务,这样的话,线
程池中的线程就可以被重复利用,减少了创建和销毁的开销,并且线程池可以设定线程个数,不至于造成内存空间不足。
在java并发包中,创建线程池常用的大致有三种方式:
一,根据执行的任务数自动设定线程池的大小
Executors.newCachedThreadPool();
二,根据传入的参数来设定线程大小
Executors.newFixedThreadPool(5);
三,创建的线程池中只有一个线程
Executors. SingleThreadExecutor();
Executors创建线程池时都会返回 ExecutorService 对象,我们就是通过这个执行对象来执行我们的任务。这里所说的任务,其实就是一个 Runnable实现类的对象,执行某个线程任务:
exec.execute(Runnable 的实现类对象 )
所以上面启动多线程程序的代码可以改为:
Const cnt=new Const();
ExecutorService es=Executors.newCachedThreadPool();
for(int i=0;i<10;i++){
IncreamentRunnable ir=new IncreamentRunnable(cnt);
es.execute(ir);
}
运行一下我们会发觉应用程序自己不会停掉,此时可以调用 shutdown()方法来停掉线程池:es.shutdown();
那么java 并发包是怎么处理线程同步问题的呢?
除了之前所说的synchronized之外,可以使用并发包自带的 ReentrantLock对象,达到细粒度的锁控制
ReentrantLock lock = new ReentrantLock(true);
使用ReentrantLock的 lock()和unlock() 来包含需要加锁的代码块。
一般来说,会把加锁 (lock)和解锁(unlock) 的代码放在 try{}finally{}里面:
try{
// 加锁
lock.lock();
System.out.println("加之前 "+this.count);
count++;
System.out.println("加之后 "+this.count);
}finally{
// 解锁
lock.unlock();
}
这是最佳实践!
5 总结
在java5 并发包之前,我们使用 Thread和Runnable 来定义并且启动线程。
使用synchronized 关键字来使线程同步,它既可以放在方法上,也可以放在代码块上。
并发包中,使用Executors来创建线程池,通过 ExecutorService对象来启动任务。
并发包中,可以使用ReentrantLock来创建一个锁,实现更细粒度的锁控制。
相关推荐
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
《基于YOLOv8的智慧社区独居老人生命体征监测系统》(包含源码、可视化界面、完整数据集、部署教程)简单部署即可运行。功能完善、操作简单,适合毕设或课程设计
Android Studio Meerkat 2024.3.1 Patch 1(android-studio-2024.3.1.14-mac.dmg)适用于macOS Intel系统,文件使用360压缩软件分割成两个压缩包,必须一起下载使用: part1: https://download.csdn.net/download/weixin_43800734/90557060 part2: https://download.csdn.net/download/weixin_43800734/90557056
侧轴承杯加工工艺编制及夹具设计.zip
NASA数据集锂电池容量特征提取(Matlab完整源码和数据) 作者介绍:机器学习之心,博客专家认证,机器学习领域创作者,2023博客之星TOP50,主做机器学习和深度学习时序、回归、分类、聚类和降维等程序设计和案例分析,文章底部有博主联系方式。从事Matlab、Python算法仿真工作8年,更多仿真源码、数据集定制私信。
板料折弯机液压系统设计.zip
C6150车床的设计.zip
机器学习之KNN实现手写数字
python爬虫;智能切换策略,反爬检测机制
mpls-vpn-optionA-all
56tgyhujikolp[
GB 6442-86企业职工伤亡事故调查分析规则.pdf
汽车液压式主动悬架系统的设计().zip
2000-2024年各省专利侵权案件结案数数据 1、时间:2000-2024年 2、来源:国家知识产权J 3、指标:专利侵权案件结案数 4、范围:31省 5、用途:可用于衡量知识产权保护水平
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
内容概要:本文档详细复现了金融数学课程作业,涵盖欧式看涨期权定价和投资组合优化两大部分。对于欧式看涨期权定价,分别采用Black-Scholes模型和蒙特卡洛方法进行了计算,并对彩虹期权进行了基于最大值的看涨期权定价。投资组合优化部分则探讨了最小方差组合、给定收益的最小方差组合、最大效用组合以及给定风险的最大收益组合四种情形,还对比了拉格朗日乘数法和二次规划求解器两种方法。文中不仅提供了详细的MATLAB代码,还有详尽的中文解释,确保每一步骤清晰明了。 适合人群:金融工程专业学生、量化分析师、金融数学爱好者。 使用场景及目标:①帮助学生理解和掌握金融衍生品定价的基本原理和方法;②为从事量化分析的专业人士提供实用工具和技术支持;③作为教学材料辅助高校教师讲授相关内容。 其他说明:文档还包括了完整的论文结构建议,从封面页到结论,再到附录,涵盖了所有必要元素,确保提交的作业符合学术规范。此外,还特别强调了数据预处理步骤,确保代码可以顺利运行。
脉冲电解射流加工喷射装置设计(1)
ThinkPad S1 (2nd Generation) 和ThinkPad Yoga 260 用户指南V3.0,包含如何拆机更换硬件
charles描述文件下载
python代码-使用人类对话数据集lora微调deepseek