Java小白进阶笔记(7)-多线程
Java的多线程基础学完了!
还是看的偏头痛杨大哥的博客:
10.偏头痛杨的Java入门教学系列之初级多线程篇
记一下学习笔记,最后是偏头痛杨留下的作业,我的一种解答。
线程与进程
线程与进程是包含关系。一个进程至少包含一个线程,至多可以包含n个线程,一个线程必须从属于一个进程。
线程(Thread)
- 单线程是一条逻辑执行流,从上到下执行,遇到阻塞就会停止。
- 多线程是以资源(内存,CPU)换时间,如果服务端的内存和CPU使用率低,那使用多线程可以提高效率。
- 线程是进程的基本单位
- 当进程被初始化后,主线程就被创建了。
- 线程可以拥有自己的堆栈、计数器、局部变量,但不用有系统资源,同一进程下的多个线程共享该进程所拥有的全部资源。
- 线程的执行是抢占式的,相同进程下的多个线程可以并发执行并相互通信方便,线程之间共享内存。
- 多线程的使用有效提高了CPU的利用率从而提升了运行效率。
进程(Process)
- 进程是操作系统中独立存在的实体,拥有独立的系统资源(内存,私有地址空间)
- 在没有允许的情况下,两个进程不可以直接通讯。
- 进程之间不共享内存,通信较困难。
- 进程是程序的一种动态形式,是cpu,内存等资源占用的基本单位,而线程不能独立的占有这些资源。
线程的概念
线程的创建与启动
- 有三种方式创建&&启动线程:Runnable,Callable,Thread
- 在执行main()的时候,main()本身会启动一个main线程,也叫主线程,这与我们自己新建的线程不同(如没有run()方法)。
线程的状态和生命周期
- 在JDK1.5之后Thread类有6种状态:NEW,RUNNABLE,BLOCKED,WAITING,TIMED_WAITING,TERMINATED.
线程的控制
join加入
- “插队”,在Thread类种提供了join()方法来实现。当在某个线程中(如在某个线程的run()方法中或者main的方法体中)调用其他线程的join()方法时,调用的线程将被阻塞,直到被join()方法加入的线程执行完之后它才会继续执行。
- 我们可以将大问题分解程许多个小问题,再为每个小问题分配一个线程,当所有的小问题都完成之后,再调用主线程接着往下走。
sleep休眠
- Thread.sleep()让执行的线程暂停若干毫秒,进入等待状态,时间到了自动恢复。
- 在休眠时间范围内即使当前没有任何可执行线程,休眠中的线程也不会被执行。
- sleep()不释放对象锁,如果当前线程持有synchronized锁并sleep(),则其他线程仍不能访问。
yield让步
- Thread.yield()使当前线程放弃CPU调度,但不是使线程阻塞&等待,线程仍处于可执行状态,随时可能再次获得CPU调度。
- 相当于当前线程已经执行了足够的时间从而转到另一个线程。
- 也有可能出现刚调用完Thread.yield()放弃CPU调度,当前线程立刻又获得CPU调度。
Priority优先级
- 优先级高的具有较多的执行机会,优先级低的线程具有较少的执行机会。这只是概率的大小,并不严格的先后执行
- 每个线程的默认优先级与创建它的父线程的优先级相同。优先级范围:1-10。
- 一般用Thread.MAX_PRIORITY等。
daemon后台&守护线程
- 后台线程是指在后台运行的线程,为其他线程提供服务,比如JVM的GC。
- 如果前台线程全部死亡,那后台线程自动死亡。
- 前台线程创建的子线程默认为前台线程,后台线程创建的子线程默认为后台线程。
- 把某线程设置为后台线程的操作必须在线程启动之前,否则会报错。
线程的同步
线程安全问题
- 由多个线程同时处理共享资源所导致的
- 比如买票卖出-1张的情况
同步代码块
- 为了解决线程安全问题,Java提供了同步机制,当多个线程使用同一种共享资源时,可以将处理的共享资源的代码放在synchronized代码块中,这个代码块就是同步代码块。
- 同步代码块的锁对象可以是任意类型的对象,但多个线程共享的锁对象必须是唯一的。
- 锁对象的创建不能放到run()方法中,否则每次都会产生新的对象,每个线程都有一个不同的锁,不能达到同步效果。
- synchronized锁和Lock锁:详细看偏头痛杨原文。
线程的通信
线程组&&线程池
略(以后学明白了再补充)
作业
使用3个线程,要求三个线程顺序执行,不允许使用sleep()强制让线程有顺序。
线程A输出1、2、3,
线程B输出4、5、6,
线程C输出7、8、9,
线程A输出10、11、12,
线程B输出13、14、15,
以此类推,一直输出到1000为止。
我的解答:
package thread_package; public class ThreadHomeWork2 { public static void main(String [] args) { Num num = new Num(); SynchronizedThread r = new SynchronizedThread(num); while(num.cnt<20) { new Thread(r).start(); new Thread(r).start(); new Thread(r).start(); } } } class Num{ String flag = "A"; int cnt = 1; } class SynchronizedThread implements Runnable{ Num num; public SynchronizedThread(Num num) { this.num = num; } public void run() { synchronized(num) { for(int i=0;i<3;i++) { if(num.cnt <=20) { System.out.println("线程:"+num.flag+" "+num.cnt++); } else { break; } } if("A".equals(num.flag)) { num.flag = "B"; } else if("B".equals(num.flag)) { num.flag = "C"; } else { num.flag = "A"; } } } }
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
阿里云Kubernetes SpringCloud 实践进行时(4): 分布式链路追踪
简介 为了更好地支撑日益增长的庞大业务量,我们常常需要把服务进行整合、拆分,使我们的服务不仅能通过集群部署抵挡流量的冲击,又能根据业务在其上进行灵活的扩展。随着分布式的普及、服务的快速增长与云计算技术的进步,微服务架构也因其特有的优势而备受关注。微服务架构的本质,是把整体的业务拆分成很多有特定明确功能的服务,通过很多分散的小服务之间的配合,去解决更大,更复杂的问题。对被拆分后的服务进行分类和管理,彼此之间使用统一的接口来进行交互。 本系列讲述了在阿里云Kubernetes容器服务基础之上,如何快速搭建基于Spring Cloud的微服务架构中的基础设施: 第一篇:分布式服务注册与发现系统 第二篇:分布式配置管理系统 第三篇:API网关服务Zuul 系统 第四篇:分布式追踪系统 第五篇:分布式弹性服务与容错处理框架Hystrix及其监控仪表板 第六
- 下一篇
深度学习中的正则化技术(附Python代码)
数据科学家面临的常见问题之一是如何避免过拟合。你是否碰到过这样一种情况:你的模型在训练集上表现异常好,却无法预测测试数据。或者在一个竞赛中你排在public leaderboard的顶端,但是在最终排名中却落后了几百名?那么这篇文章就是为你而准备的! (译者注: 在kaggle这样的数据竞赛中, public leaderboard排名是根据一部分测试集来计算的,用于给选手提供及时的反馈和动态展示比赛的进行情况;而private leaderboard是根据测试集的剩余部分计算而来,用于计算选手的最终得分和排名。 通常我们可以把public LB理解为在验证集上的得分,private LB为真正未知数据集上的得分,这样做的目的是提醒参赛者,我们建模的目标是获取一个泛化能力好的模型) 避免过拟合可以提高我们模型的性能。 目录 1. 什么是正
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS7,CentOS8安装Elasticsearch6.8.6
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- Linux系统CentOS6、CentOS7手动修改IP地址
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- SpringBoot2全家桶,快速入门学习开发网站教程
- CentOS7设置SWAP分区,小内存服务器的救世主