线程应用实例--简单线程池实现

        对于服务端的程序,经常面对的是客户端传入的短小(执行时间短、工作内容较为单一)任务,需要服务端快速处理并返回结果。如果服务端每次接受到一个任务,创建一个线程,然后进行执行,这在原型阶段是个不错的选择,但是面对成千上万的任务递交进服务器时,如果还是采用一个任务一个线程的方式,那么将会创建数以万记的线程,这不是一个好的选择。因为这会使操作系统频繁的进行线程上下文切换,无故增加系统的负载,而线程的创建和消亡都是需要耗费系统资源的,也无疑浪费了系统资源。

        线程池技术能够很好地解决这个问题,它预先创建了若干数量的线程,并且不能由用户直接对线程的创建进行控制,在这个前提下重复使用固定或较为固定数目的线程来完成任务的执行。这样做的好处是,一方面,消除了频繁创建和消亡线程的系统资源开销,另一方面,面对过量任务的提交能够平缓的劣化。

一、线程池接口定义

public interface ThreadPool <Job extends Runnable>{

        //向线程池提交任务

        void execute(Job job);

        //关闭线程池

        void shutdown();

        //增加工作者线程

        void addWorkers(int num);

        //减少工作者线程

        void removeWorker(int num);

        //获取等待执行的任务数

        int getJobSize();

}

二、线程池实现

public class DefaultThreadPool<Job extends Runnable> implements ThreadPool<Job>{

        //最大工作者线程数量

        private static final int MAX_WORKER_NUMBERS = 10;

        //默认工作者线程数量

        private static final int DEFAULT_WORKER_NUMBERS = 5;

        //最小工作者线程数量

        private static final int MIN_WORKER_NUMBERS = 1;

        //工作任务列表

        private final LinkedList<Job> jobs = new LinkedList<Job>();

        //工作者线程列表

        private final List<Worker> workers = Collections.synchronizedList(new ArrayList<Worker>());

        //工作者线程数量

        private int workerNum = DEFAULT_WORKER_NUMBERS;

        //线程编号生成

        private AtomicLong threadNum = new AtomicLong();

        public DefaultThreadPool() {

                //初始化并启动工作者线程

                initializeWokers(DEFAULT_WORKER_NUMBERS);

        }

        public DefaultThreadPool(int num) {

                workerNum = num > MAX_WORKER_NUMBERS? MAX_WORKER_NUMBERS : num < MIN_WORKER_NUMBERS?                 MIN_WORKER_NUMBERS : num;

                initializeWokers(workerNum);

        }

        //向线程池提交任务֪

        public void execute(Job job) {

                if (job != null) {

                    //提交任务后,向等待的在工作任务列表的线程发送唤醒通知֪

                    synchronized (jobs) {

                            jobs.addLast(job);

                            jobs.notify();

                    }

                }

        }

        public void shutdown() {

                for (Worker worker : workers) {

                    worker.shutdown();

                }

        }

        //增加工作者线程

        public void addWorkers(int num) {

                synchronized (jobs) {

                        if (num + this.workerNum > MAX_WORKER_NUMBERS) {

                                num = MAX_WORKER_NUMBERS - this.workerNum;

                        }

                        initializeWokers(num);

                        this.workerNum += num;

                }

        }

        //减少工作者线程

        public void removeWorker(int num) {

                synchronized (jobs) {

                        if (num >= this.workerNum) {

                                throw new IllegalArgumentException("beyond workNum");

                        }

                        int count = 0;

                        while (count < num) {

                                Worker worker = workers.get(count);

                                if (workers.remove(worker)) {

                                        worker.shutdown();

                                        count++;

                                }

                        }

                        this.workerNum -= count;

                }

        }

        //获取等待执行的任务数

        public int getJobSize() {

                    return jobs.size();

        }

        //初始化工作者线程列表

        private void initializeWokers(int num) {

                for (int i = 0; i < num; i++) {

                    Worker worker = new Worker();

                    workers.add(worker);

                    Thread thread = new Thread(worker, "ThreadPool-Worker-" + threadNum.

                    incrementAndGet());

                    thread.start();

            }

        }

        //工作者线程定义

        class Worker implements Runnable {

               //是否工作

                private volatile boolean running = true;

                public void run() {

                        while (running) {

                            Job job = null;

                            synchronized (jobs) {

                                    //如果任务列表中没有任务则等待

                                    while (jobs.isEmpty()) {

                                            try {

                                                    jobs.wait();

                                             } catch (InterruptedException ex) {

                                                    Thread.currentThread().interrupt();

                                                    return;

                                             }

                                    }

                                    //获取任务执行

                                    job = jobs.removeFirst();

                                    if(job != null) {

                                        try {

                                            job.run();

                                        } catch (Exception ex) {

                                        }

                                    }

                        }

                }

            }

            public void shutdown() {

                    running = false;

                }

        }

}

        当客户端调用execute(Job)方法时,会不断地向任务列表jobs中添加Job,而每个工作者线程当客户端调用execute(Job)方法时,会不断地向任务列表jobs中添加Job,而每个工作者线程会不断地从jobs上取出一个job进行执行,当jobs为空时,工作者线程进入等到状态。

        添加一个job后,对工作队列jobs调用了notify()方法,而不是notifyAll()方法,因为能够确定有工作者线程被唤醒,这时使用notify()方法将会比notifyAll()方法获得更小的开销(避免将等待队列中的线程全部移到阻塞队列中)。

        可以看到线程池的本质是使用了一个线程安全的工作队列连接工作者线程和客户端线程,客户端线程将任务放入工作队列后边返回,而工作者线程则不断地从工作任务队列取出工作并执行。当工作队列为空时,所有的工作者线程均等待在工作队列上,当有客户端提交了一个任务之后会通知任意一个工作者线程,随着大量任务的提交,更多的工作者线程会被唤醒。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,734评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,931评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,133评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,532评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,585评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,462评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,262评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,153评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,587评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,792评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,919评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,635评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,237评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,855评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,983评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,048评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,864评论 2 354

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,656评论 18 139
  • ~家像个垃圾堆那样~~ ~~有点原则,有点规律,有点思想好不好~~ ~~搞得家像垃圾堆那样~~ ~不要太随意,点点...
    烂笔头韦阅读 113评论 0 0
  • 不要轻易去依赖一个人,它会成为你的习惯,当分别来临,你失去的不是某个人,而是你精神的支柱。无论何时何地,都要学会独...
    鹿哈鹿哈阅读 529评论 0 0
  • 参考http://stackoverflow.com/questions/29505622/spannablest...
    小面包屑阅读 1,873评论 0 2
  • 水逆啊诸事不顺。又是丢东西又是生病的。。
    晨曦iuiu阅读 154评论 0 0