CGB2108day16

一、线程

1.什么是进程?什么是程序?有什么区别?

程序:数据与指令的集合,程序是静态的

进程:给程序加入了时间的概念,不同的时间进程有不同的状态

进程是动态的,就代表OS中正在运行的程序

特点是独立性,动态性和并发性

2.什么是并行?什么是串行?什么是并发?

CPU:电脑的核心处理器,类似于“大脑”

串行:是指同一时刻一个CPU只能处理一件事,类似于单车道

并行:相对来说资源比较充足,多个CPU可以同时处理不同的多件事,类似于多车道

并发:相对来说资源比较紧缺,多个进程同时抢占公共资源,比如多个进程抢占一个CPU

3.什么是线程?线程与进程有什么关系?

线程是OS能够进行运算调度的最小单位

一个进程可以拥有多个线程,当然,也可以只拥有一个线程,只有一个线程的进程被称作单线程程序

注意:每个线程也有自己独立的内存空间,当然也有一部分公共的空间用于保存共享的数据

在宏观上:一个CPU看似可以同时处理多件事

在微观上:一个CPU同一时刻只能处理一件事

结论:线程的执行具有随机性,我们控制不了,是由OS底层的算法来决定的

4.线程有几种状态?它们是怎么转换的?

新建状态:new—申请PCB,进行资源的分配

就绪/可运行状态:万事俱备,只欠CPU,其实是将创建好的线程对象加入到就绪队列中,等待OS选中,这个选择我们是控制不了的

执行/运行状态:就绪队列中的线程被OS选中,正在执行

注意:只有就绪状态才能切换成执行状态

阻塞状态:线程在执行中遇到了问题:

锁阻塞、休眠阻塞、等待阻塞...问题解决后再加入到就绪队列中

终止状态:线程成功执行完毕,释放资源

 二、多线程实现方案总结

1.多线程实现的方案一:继承

1.自定义一个类extends Thread

2.重写run()里面写业务

3.创建多个线程对象

4.线程对象调用star(),以多线程的方式启动

注意:可以通过调用父类Thread的含参构造Thread(String name)

给自定义线程对象起名字,调用方式:super(name);

构造方法摘要:

Thread()创建一个新的线程对象,名字是系统自定义的

Thread(String name)与上面功能一致,还可以自定义线程名

多线程实现的方案二:实现

1.自定义一个类实现Runnable接口implements Runnable

2.添加接口中未实现的抽象方法run(),其中是我们的业务

3.打印线程名称:Thread.currentThread().getName()

4.创建Runnable接口的实现类【也就是自定义类】对象,作为目标业务对象

5.创建线程对象—Thread t1 = new Thread(target);

目的:为了把实现类与Thread建立关系,原因是想用Thread的start()

6.通过线程对象调用start(),把线程对象加入就绪队列

构造方法摘要:

Thread(Runnable target)创建一个线程对象,参数为Runnable实现类的对象

Thread(Runnable target,String name)与上面功能一致,还可以自定义线程名

7.虽然方案二写法较为复杂,但是方案二的有点如下:

1)耦合性不强,没有继承,后续仍然可以实现其他接口

2)采用实现接口的方式,后续仍然可以实现其他接口

3)可以给所有线程对象统一业务,业务保持一致

4)面向接口编程,代码更高级

多线程实现的方案三:线程池ExecutorSrevice

1.Executors是用来辅助创建线程池的工具类对象

2.常用方法是:newFixedThreadPool(int)

这个方法可以创建指定线程数目的线程池对象

3.创建出来的线程池对象是ExecutorService:用来存储线程的池子

负责:新建/启动/关闭线程execute()让线程池中的线程来执行业务,每次调用都会讲一个线程加入到就绪队列

4.注意1:线程池负责将线程加入到就绪的队列中,但并不代表所有的线程都会运行,线程能否执行还是取决于OS的调用,如果没有被分配时间片,是转换不了运行状态的

注意2:线程池是不关闭的,只要就是想实现线程的随取随用,这样就避免了频繁的创建与销毁线程,浪费大量资源

三、多线程数据安全隐患解决方案

1.出现数据安全问题的原因

1)多线程程序

2)多个线程拥有共享数据

3)多条语句操作共享数据

2.解决方案:加锁synchronized

1)同步代码块【常用】,格式:

synchronized(唯一的锁对象){

可能出现数据安全问题的所有代码块

}

2)同步方法【不常用】,格式:

在方法定义上加synchronized

3.使用同步时的注意事项

1)锁对象必须唯一!

比如1:如果是实现接口的方式,只创建一个目标业务类对象(接口实现类对象),那么也只有一个锁对象

比如2:如果是继承Thread类的方式,可能要创建多个子类的对象,那这个时候需要给锁对象加static,保证锁对象唯一被所有对象共享

2.所以:继承的方式,常用的锁对象是类名.calss字节码对象

锁对象的类型不做限制,只要能保证唯一即可

3.加锁的范围需要认真考虑,不能太大,也不能太小,太大浪费效率,太小锁不住

4.同步异步

异步:是多个线程抢占资源的效果,不排队,效率高,但是数据不安全

同步:每次只有一个线程独占资源,排队,效率低但是安全,synchronized也被称作同步关键字

5.多线程售票案例中问题的解决方案

1)创建4个线程对象,售票400张

解决方案:将票数设置为静态,被全局所有对象共享

2)票数出现了重卖(一张票卖给了多个人)的现象:

解决方案:使用同步代码块,确保一次只要一个线程卖票

3)票数出现了超卖(卖出了超过范围的票0 -1 -2)的现象

解决方案:优化代码逻辑,双重校验,有票的时候再卖票,没票的时候就停止。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值