本文主要是介绍《java多线程》--利用线程池来模拟选课,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
多线程一直是热门问题,用好了能保证数据的安全性,用差了小则降低程序的性能,大则使程序崩溃,今天小编用一个高校学生选课的实例来模拟一下Java中的生产者与消费者问题。如果把学生选课的逻辑简单化,我们可以将其映射到多线程的生产者和消费者问题上。
需求:管理员设置一定数量的选修课供学生来选择,当选修课的数量降为0时,学生们停止选课,并通知管理员添加课程,当管理员添加课程完毕后,通知学生们可以选课了。这里的学生就是消费者,而老师就是生产者。
图解:
课程代码
public class Courses {private int subjectNum=20;public synchronized void set(){//如果有数据就等待if(subjectNum>=0){try {this.wait();} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}//设置数据subjectNum= 20;System.out.println("又重新加了" + subjectNum + "个课程");this.notify();}public synchronized void get(){//如果有数据就等待if(subjectNum<0){try {System.out.println("课程被选完了,请等待管理员添加课程");this.notify();this.wait();} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}else{System.out.println("课程被剩余" + ":" + subjectNum);subjectNum--;}}}
生产者代码
public class SetThread implements Runnable {private Courses c;public SetThread(Courses c){this.c=c;}@Overridepublic void run() {// TODO Auto-generated method stubwhile(true){c.set();}}}
消费者代码
public class GetThread implements Runnable {private Courses c;public GetThread(Courses c){this.c=c;}@Overridepublic void run() {// TODO Auto-generated method stubwhile(true){c.get();}}}
执行选课代码
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class SelectCoursesDemo {public static void main(String[] args) {// TODO Auto-generated method stubCourses c =new Courses();SetThread st =new SetThread(c);GetThread gt1=new GetThread(c);//选课学生甲GetThread gt2=new GetThread(c);//选课学生乙GetThread gt3=new GetThread(c);//选课学生丙ExecutorService pool=Executors.newFixedThreadPool(4);//声明一个线程池,里面放四个线程pool.submit(st);pool.submit(gt1);pool.submit(gt2);pool.submit(gt3);}}
效果图:
小结
上面就是Java多线程中的生产者与消费者问题,例子挺简单,将选课的逻辑化简了,归根到底,其实选课就是生产者和消费者问题,只不过是逻辑复杂了写。另外,利用线程池来创建线程比单独实例化线程对象耗费的性能要底,因为当系统不再用某个线程时,不是将此线程当垃圾回收,而是放到线程池中,等待下次用的时候再取出来,这就节省了系统重新创建线程和销毁线程的性能。
这篇关于《java多线程》--利用线程池来模拟选课的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!