1.多线程数据安全隐患解决方案
1. 出现数据安全问题的原因:
- 多线程程序
- 多个线程拥有共享数据
- 多条语句操作共享数据
2. 同步与异步
异步:是多个线程抢占资源的效果,不排队,所以效率高,但是数据不安全
同步:每次只有一个线程独占资源,排队,所以效率低,但是安全,为了安全必要应该牺牲一部分资源
synchronized也被称作同步关键字
3. 解决方案:加锁synchronized
同步代码块【常用】,格式:
synchronized(唯一的锁对象){
可能出现数据安全问题的所有代码
}
4.使用同步时的注意事项:
- 锁对象必须唯一!!!
比如:如果是实现接口的方式,只创建了一个目标业务类对象(接口实现类对象),那么也只有一个锁对象
比如2:如果是继承Thread类的方式,你可能要创建多个子类的对象,那这个时候需要给锁对象加static,保证锁对象唯一被所有对象共享
注意:类名.class字节码对象作为锁对象,不管哪种实现多线程的方式都可以用哦,不过一般在继承Thread实现方式使用较多 - 锁对象的类型不做限制,只要能保证唯一即可
- 加锁的范围需要认真考虑
不能太大,也不能太小,太大浪费效率,太小锁不住
5.多线程售票案例中问题的解决方案:
1.创建4个线程对象,售卖400张票:
解决方案:将票数设置为静态,被全局所有对象共享
2.票数出现了重卖(一张票卖给了多个人)的现象:
解决方案:使用同步代码块,确保一次只有一个线程卖票
3.票数出现了超卖(卖出了超出范围的票0 -1 -2)的现象:
解决方案:优化代码逻辑,有票的时候再卖票,没票的时候就停止,有多种方案,以测试结果为准即可。
详细代码与笔记请参照:
多线程售票案例与同步锁解决方案详细笔记 点这里
2 线程的创建方式3
多线程编程实现方案三:Executors 创建线程池的方式
- 创建线程池的工具类:
Executors.newFixedThreadPool(int n);可以创建包含最多n个线程的线程池对象 - 创建好的线程池对象:ExecutorService
使用pool.excute()来讲线程池中的线程以多线程的方式启动,每次调用都会将一个线程对象加入到就绪队列之中
这个线程池对象负责: 新建/启动/关闭线程,而我们主要负责的是自定义的业务
注意:线程池是不关闭的,实现的效果就是线程池中线程对象的随取随用,这样就避免了频繁的创建与销毁线程,不会造成资源浪费 - 合理利用线程池可以拥有的优势:
1. 降低系统的资源消耗:减少系统创建与销毁线程对象的次数,每个线程都可以重复利用,执行多次任务
2. 提高响应速度:当任务到达时,任务可以不用等待线程创建就能立即执行
3. 提高线程的可管理性:可以根据系统的承受能力,调整线程池中线程的数目
防止因为创建多个线程消耗过多的内存导致服务器的崩溃
【每个线程大约需要1MB的内存,线程开的越多,消耗的内存也就越大,最后死机】