Java定时发布文章简单方案

没有需求,就没有折腾。

首先,阐述一下背景。

早上迷迷糊糊地开始了春节后的一天上班日程,脑袋还在噼里啪啦放烟花,项目管理部和SEO小伙伴就提了一桶凉水过来,往我头上一浇,瞬间烟花都湮灭了。

“给官网加个定时发布文章的功能吧。”

“啥?”

“我们的官网,每次新增文章都是立即执行静态化并进行发布,现在周末也需要发布文章,SEO周末是不上班哒,所以,请给我们开发一个定时发布文章的功能吧。”

“啊?”

“评估一下时间,越快越好。”

“...”

“需求了解了吧,那就这样,尽快产出哦。”

“...”

脑袋还在宕机中。

喂喂喂,那你们浇灭了我的烟花,都不用赔一下吗,真不厚道。

虽然我不喜欢频繁需求变动,但是我爱折腾。

不过这么简单的功能,貌似也算不上折腾,但是记录下来也许能帮助到别人呢,Hard to say。

环境说明

1、centOS 服务器一台

2、基于SSM + 一些没必要在这里提到的第三方控件

3、Bootstrap前端框架

4、最最重要的是:帅比码农一枚

其实,上面前三点都没必要提及,主要是基于Java环境来实现定时任务。所以最重要的,请记住第四点,强调,是第四点。

思路

SEO通过管理后台新增文章,但是并不是立即发布,而是可以手动选择发布方式,包括立即发布定时发布,定时发布可以指定一个时间,交由系统自动实现发布功能。

说了跟没说似的,原谅我,帅比码农说话都比较高(zhuang)深(shen)莫(nong)测(gui)。

1、前端通过bootstrap-datepicker插件,在文章表单中新增一个发布时间的选择控件。具体使用方式请参考官网API或留言。

<!-- 页面元素 -->
<div class="input-append date form_datetime">
    <input id="pubTime" name="pubTime" size="16" type="text" value="" readonly>
    <span class="add-on"><i class="icon-th"></i></span>
</div>

<!-- Javascript -->
<script type="text/javascript">
    $(".form_datetime").datetimepicker({
            language:"zh-CN",
            showMeridian: true,
            todayBtn:true,
            startDate:new Date(),
            format: "yyyy-mm-dd hh:ii:ss"
        }).on('changeDate', function(ev){
            $('#pubTiming').attr('checked',true);//通过事件,实现[定时发布]单选按钮的联动选择
        });
</script> 
界面

2、后台新增文章的方法,新增入参[发布方式-pubType]和[发布时间-pubTime]来接收表单传递过来的值,当用户选择发布方式为定时发布时,要求发布时间必须选择。

由于这里是以实体的方式来接收表单的,只需要在Article实体中新增pubType和pubTime两个属性,并生成getter和setter即可接收表单值。

部分代码如下

    /**
     * 新增文章
     *
     * @param article 文章实体
     * @param request 请求
     * @return ResponseBean 响应实体
     */
    @RequestMapping("/add")
    @ResponseBody
    public ResponseBean add(Article article, HttpServletRequest request) {
        boolean success = articleService.add(article, request);
        ...
    }
    
    /**
     * 文章
     *
     * @author zoro
     * @version 1.0
     * @since 2018/02/23
     */
    public class Article implements Serializable {
        ...
        private Integer pubType;//发布方式,1立即,2定时
        @DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")
        private Date pubTime;//定时发布的时间
        
        public Integer getPubType() {
            return pubType;
        }
    
        public void setPubType(Integer pubType) {
            this.pubType = pubType;
        }
    
        public Date getPubTime() {
            return pubTime;
        }
    
        public void setPubTime(Date pubTime) {
            this.pubTime = pubTime;
        }
        ...
    
    }

3、将文章内容和发布状态保存到数据库,如果是立即发布,则执行文章渲染,通过模板渲染成html文件,以供访问。

articleRender.rendering();

4、如果是定时发布的话,就需要建立定时任务。
这里有几种情况需要说明:

  • 新增文章
    直接保存文章,并建立定时任务。
  • 修改文章
    修改文章会存在不同时间点重复发布任务的可能性,所以需要特殊处理。

针对修改文章,每次新建定时任务的时候,先判断是否存在同一篇文章的定时任务,如果有,则标识该任务为取消状态(取消状态下的任务,任务体不会执行任何操作),并从id映射和缓存中移除

文章任务部分代码如下

package com.andatech.admin.service;

...
import com.andatech.tools.StringUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicBoolean;

/**
 * 文章发布任务
 *
 * @author Zoro
 * @date 2018/2/23
 * @since 1.0
 */
public class ArticlePublishJob implements Runnable {

    /**logger*/
    private static final Logger LOGGER = LoggerFactory.getLogger(ArticlePublishJob.class);

    /**id映射*/
    private static volatile ConcurrentHashMap<String, String> idMapping = new ConcurrentHashMap<>();
    /**任务缓存*/
    private static volatile ConcurrentHashMap<String, ArticlePublishJob> cache = new ConcurrentHashMap<>();
    
    private volatile AtomicBoolean canceled = new AtomicBoolean(false);//任务取消状态
    private String jobId;//任务id
    private String articleUuid;//文章id
    private ArticleRender articleRender;//文章渲染器

    
    public ArticlePublishJob(ArticleRender articleRender) {
        this.jobId = IdGenerator.plainJdkUUID();
        this.articleRender = articleRender;
        this.articleUuid = articleRender.getArticle().getUuid();
        //取消并清除上一次任务
        cancelAndClearLastJobIfExist();
        //缓存本次任务
        cacheThisJob();
    }

    /**
     * 取消并清除上一次任务
     */
    private void cancelAndClearLastJobIfExist(){
        if (StringUtil.isNotEmpty(idMapping.get(articleUuid))) {
            ArticlePublishJob lastJob = cache.get(idMapping.get(articleUuid));
            if (null != lastJob) {
                lastJob.cancelJob();
                cache.remove(idMapping.get(articleUuid));
                idMapping.remove(articleUuid);
            }
        }
    }

    /**
     * 缓存本次任务
     */
    private void cacheThisJob(){
        //id映射
        idMapping.put(this.articleUuid, this.jobId);
        //文章发布任务缓存
        cache.put(this.jobId, this);
    }

    @Override
    public void run() {
        //判断任务是否被取消
        if (!canceled.get()) {
            ArticleService articleService = (ArticleService) SpringContextHolder.getBean("articleService");
            //渲染
            try {
                articleRender.rendering();
            } catch (IOException e) {
                LOGGER.error("render log err:" + e.getMessage(), e);
            }
            //更新文章状态
            Article updArticle = new Article();
            updArticle.setUuid(articleRender.getArticle().getUuid());
            updArticle.setStatus(Article.STATUS_NORMAL);
            articleService.edit(updArticle);

            //从缓存中清理本任务
            clear();
        }
    }

    /**
     * 取消任务
     */
    public void cancelJob() {
        this.canceled.set(true);
    }

    /**
     * 清理缓存
     */
    public void clear() {
        idMapping.remove(articleUuid);
        cache.remove(this.jobId);
    }

}

5、具体定时任务方式,包括以下几种

  • Thread方式:线程等待,不安全。
  • timer方式:线程资源没有复用。
  • 任务调度框架,比如Quartz等:需要继承框架。
  • ScheduledExecutorService方式:被相中了。

综上分析,选择了最后一种,也是较好的选择之一,下面给出最简单的用法,如有深入需要,建议查看JavaAPI。

部分代码如下

/**创建线程池*/
public static ScheduledExecutorService service = Executors.newScheduledThreadPool(50);


/**新建任务,并设定执行时间*/
ArticlePublishJob job = new ArticlePublishJob(articleRender);
long delay = article.getPubTime().getTime() - System.currentTimeMillis();
service.schedule(job, delay, TimeUnit.MILLISECONDS);

测试,大功告成。

总结
不结合业务来说,定时任务的创建无非就"第5点"中的几种方式,熟悉API并熟练使用即可。
结合业务情况下,需要考虑任务是否会重复,重复了怎么处理等问题。

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

推荐阅读更多精彩内容