一、什么是多线程
二、为什么使用多线程
•使用多线程的优点
–最大化使用硬件的资源(最大化利用CPU的性能)
–更合理的利用时间
•使用多线程的缺点
–线程同步问题
–死锁问题
三、多线程的程序如何运行
代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/fa05710d8907bfb785a41248403bfdb0.png)
运行结果如下:
![](https://i-blog.csdnimg.cn/blog_migrate/48997ef34368bbf70597f766d26cb51e.png)
四、如何创建线程
•方式一:创建Thread对象,重写run()方法。代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/229ef86be94a55f68a2fec4447b3dc13.png)
•方式二:创建Thread对象,使用参数为Runnable的构造方法,传入Runnable接口实现类的实例。代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/245c210f88d868abaf028ed5fa48a808.png)
五、线程池
•线程池的优点
–避免了频繁的创建和销毁线程对象
–方便管理线程
•线程池的创建,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/1ed2ec186b35731d9699cdec26367b1d.png)
六、线程同步问题
•现在有如下一个静态方法print(),方法将输入的字符串的字符逐一打印出来,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/b1369dc78c4eb34e1c0d3b79fb6debd9.png)
•单线程调用该方法:
![](https://i-blog.csdnimg.cn/blog_migrate/13141779a833fc076c81140d56585ac1.png)
•运行结果:
![](https://i-blog.csdnimg.cn/blog_migrate/1dbb357050c870d96127edabd748afc3.png)
•多线程调用该方法(两个线程,循环5次):
![](https://i-blog.csdnimg.cn/blog_migrate/94f6d14784346a38d33e1dbc2d628983.png)
•运行结果:
![](https://i-blog.csdnimg.cn/blog_migrate/e25ec05bde13ff4e11e1ca5a10609f1e.png)
七、解决线程同步问题(一)
•修改print()方法,加入synchronized关键字修饰,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/23f669879ddab6b935e53ff985c9ab0d.png)
•运行结果:
![](https://i-blog.csdnimg.cn/blog_migrate/d70cb96211976bb0501b20dbdd28b066.png)
八、解决线程同步问题(二)
•修改print()方法,使用Lock,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/88bdafe49bd60788ef99e17a0675504e.png)
•运行结果:
![](https://i-blog.csdnimg.cn/blog_migrate/e2d51f3502e2adb20e7864ca456c8838.png)
九、解决线程同步问题(三)
•修改print()方法,自定义锁,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/b517aad7dcd46457855c89e3113068b2.png)
•运行结果:
![](https://i-blog.csdnimg.cn/blog_migrate/75c1a93d4c3850c382f6987ab70fe91c.png)
十、实例:批量订单创建
•场景:批量创建100笔订单,订单状态全部为已付款未发货
•多线程模型:“生产者-消费者” 模型
–生产者工作:订单创建过程
–消费者工作:订单付款
–过程:生产者创建订单后,将订单放入一个池中,消费者去池中获取订单,然后进行付款操作
生产者和消费者各5个线程,共10个线程
![](https://i-blog.csdnimg.cn/blog_migrate/8e20d70baa51f0a107c0abf7d58afb7f.jpeg)
•代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/1c3bdb148370d7d42301e7d503003436.png)
![](https://i-blog.csdnimg.cn/blog_migrate/a93124a56456b4a2667f518740104ec8.png)
十一、死锁问题
![](https://i-blog.csdnimg.cn/blog_migrate/7819a4d9eb095475c0c2f7b6f33e4644.png)
十二、多线程下的单例模式
•单线程下的单例模式,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/b07594f1ce89f87a89775c1ac316be7d.png)
•多线程下的单例模式,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/a711e2ca526e930b46269858a955fa50.png)
![](https://i-blog.csdnimg.cn/blog_migrate/58d1bce499042449265aee2cb379639d.png)
一个遗留问题
•ArrayList在循环过程中调用remove()方法,代码如下:
![](https://i-blog.csdnimg.cn/blog_migrate/4e3edcc9cf9bede9e859b924239161a3.png)
•出现以下异常:
![](https://i-blog.csdnimg.cn/blog_migrate/869e9ba41467a89f04149781548f55e4.png)
后续分享内容预览
•多线程分享(二)
–多线程间的数据共享(ThreadLocal介绍)
–volatile关键字
–原子操作类介绍(基础类型对应的原子类型,如AtomicInteger等)
–读写锁
–阻塞条件Condition的使用
–异步任务处理(Callable、Future、ExecutorCompletionService)
•多线程分享(三)
–线程同步工具的使用(Semaphere、CyclicBarrier、CountDownLatch、Exchanger)
–堵塞队列(ArrayBlockingQueue等)与同步集合(ConcurrentLinkedQueue等)