在Web微信应用中使用博客园RSS以及Quartz.NET实现博客文章内容的
本篇随笔介绍在Web微信应用中使用博客园RSS以及Quartz.NET实现博客文章内容的定期推送功能,首先对Quartz.NET进行一个简单的介绍和代码分析,掌握对作业调度的处理,然后对博客园RSS内容的处理如何获取,并结合微信消息的群发接口进行内容的发送,从而构建了一个在Web应用中利用作业调度来进行消息发送的业务模型。 Quartz.NET是一个开源的作业调度框架,非常适合在平时的工作中,定时轮询数据库同步,定时邮件通知,定时处理数据等。?Quartz.NET允许开发人员根据时间间隔(或天)来调度作业。它实现了作业和触发器的多对多关系,还能把多个作业与不同的触发器关联。整合了 Quartz.NET的应用程序可以重用来自不同事件的作业,还可以为一个事件组合多个作业。 Quartz框架的一些基础概念解释: ? Scheduler ? ? 作业调度器。 ? IJob ? ? ? ? ? ? 作业接口,继承并实现Execute,?编写执行的具体作业逻辑。 JobBuilder ? ? ? 根据设置,生成一个详细作业信息(JobDetail)。 TriggerBuilder ? 根据规则,生产对应的Trigger 官方的使用案例代码如下所示 button1_Click(= Common.Logging.Simple.ConsoleOutLoggerFactoryAdapter { Level =
</span><span style="color: #008000;">//</span><span style="color: #008000;"> Grab the Scheduler instance from the Factory </span>
IScheduler scheduler =<span style="color: #000000;"> StdSchedulerFactory.GetDefaultScheduler();
</span><span style="color: #008000;">//</span><span style="color: #008000;"> and start it off</span>
<span style="color: #000000;"> scheduler.Start();
<span style="color: #000000;"> scheduler.ScheduleJob(job,trigger);
<span style="color: #000000;"> scheduler.Shutdown(); { Console.WriteLine(se); }
?启动定义一个HelloJOb的对象,如下代码所示 <h3 id="autoid-4-1-2">2、Quartz的cron表达式 cron expressions 整体上还是非常容易理解的,只有一点需要注意:"?"号的用法,看下文可以知道“?”可以用在 day of month 和 day of week中,他主要是为了解决如下场景,如:每月的1号的每小时的31分钟,正确的表达式是:* 31 * 1 * ?,而不能是:* 31 * 1 * *,因为这样代表每周的任意一天。 由7段构成:秒 分 时 日 月 星期 年(可选)"-" :表示范围? MON-WED表示星期一到星期三"," :表示列举 MON,WEB表示星期一和星期三"*" :表是“每”,每月,每天,每周,每年等"/" :表示增量:0/15(处于分钟段里面) 每15分钟,在0分以后开始,3/20 每20分钟,从3分钟以后开始"?" :只能出现在日,星期段里面,表示不指定具体的值"L" :只能出现在日,星期段里面,是Last的缩写,一个月的最后一天,一个星期的最后一天(星期六)"W" :表示工作日,距离给定值最近的工作日"#" :表示一个月的第几个星期几,例如:"6#3"表示每个月的第三个星期五(1=SUN...6=FRI,7=SAT) 官方cron表达式实例 <table style="height: 455px; width: 1050px;"> 代表意义 |
|
我曾经在统一接口的Web API后台,使用了这个Quartz.NET来实现站场信息的同步处理,这样可以把其他供应商提供的接口数据,同步到本地,可以加快数据的检索和处理效率。
具体代码如下所示。
首先是在Global.asax的后台代码里面进行同步代码处理。
</span><span style="color: #0000ff;">protected</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> Application_Start()
{
GlobalConfiguration.Configuration.EnableCors();
GlobalConfiguration.Configure(WebApiConfig.Register);
</span><span style="color: #008000;">//</span><span style="color: #008000;">创建执行同步的处理</span>
ISchedulerFactory sf = <span style="color: #0000ff;">new</span><span style="color: #000000;"> StdSchedulerFactory();
scheduler </span>=<span style="color: #000000;"> sf.GetScheduler();
CalendarTask();
CreateOnceJob();
</span><span style="color: #008000;">//</span><span style="color: #008000;">启动所有的任务</span>
<span style="color: #000000;"> scheduler.Start();
}
{
<span style="color: #0000ff;">if(scheduler != <span style="color: #0000ff;">null<span style="color: #000000;">)
{
scheduler.Shutdown(<span style="color: #0000ff;">true<span style="color: #000000;">);
}
}
</span><span style="color: #808080;">///</span> <span style="color: #808080;"><summary></span>
<span style="color: #808080;">///</span><span style="color: #008000;"> 创建同步任务
</span><span style="color: #808080;">///</span> <span style="color: #808080;"></summary></span>
<span style="color: #0000ff;">private</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> CalendarTask()
{
IJobDetail job </span>= JobBuilder.Create<StationSyncJob><span style="color: #000000;">()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">StationSyncJob</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span><span style="color: #000000;">)
.Build();
</span><span style="color: #008000;">//</span><span style="color: #008000;">每天凌晨1点执行一次:0 0 1 * * ?</span>
ICronTrigger trigger =<span style="color: #000000;"> (ICronTrigger)TriggerBuilder.Create()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">trigger1</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span>) <span style="color: #008000;">//</span><span style="color: #008000;">"0 34,36,38,40 * * * ?"</span>
.WithCronSchedule(<span style="color: #800000;">"</span><span style="color: #800000;">0 0 1 * * ?</span><span style="color: #800000;">"</span>)<span style="color: #008000;">//</span><span style="color: #008000;">"0 0 1 * * ?"</span>
<span style="color: #000000;"> .Build();
DateTimeOffset ft </span>=<span style="color: #000000;"> scheduler.ScheduleJob(job,trigger);
LogTextHelper.Info(</span><span style="color: #0000ff;">string</span>.Format(<span style="color: #800000;">"</span><span style="color: #800000;">您在 {0} 时候创建了Quartz任务</span><span style="color: #800000;">"</span><span style="color: #000000;">,DateTime.Now));
}
</span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> CreateOnceJob()
{
IJobDetail onceJob </span>= JobBuilder.Create<StationSyncJob><span style="color: #000000;">()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">onceJob</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span><span style="color: #000000;">)
.Build();
</span><span style="color: #008000;">//</span><span style="color: #008000;">启动的时候运行一次</span>
DateTimeOffset startTime = DateBuilder.NextGivenSecondDate(<span style="color: #0000ff;">null</span>,<span style="color: #800080;">30</span><span style="color: #000000;">);
ISimpleTrigger simpleTrigger </span>=<span style="color: #000000;"> (ISimpleTrigger)TriggerBuilder.Create()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">simpleOnce</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span><span style="color: #000000;">)
.StartAt(startTime)
.Build();
DateTimeOffset ft </span>=<span style="color: #000000;"> scheduler.ScheduleJob(onceJob,simpleTrigger);
}
}</span></pre>
其中同步站场信息的Job实现如下所示(这里是通过调用第三方接口获取数据,然后把它们保存到本地,这个定时服务设定在每天的一个是时间点上执行,如凌晨1点时刻)。
StationDetailResult result </span>= <span style="color: #0000ff;">new</span><span style="color: #000000;"> StationDetailResult();
</span><span style="color: #0000ff;">try</span><span style="color: #000000;">
{
QueryStationJson json </span>= <span style="color: #0000ff;">new</span> QueryStationJson();<span style="color: #008000;">//</span><span style="color: #008000;">空查询,一次性查询所有</span>
<span style="color: #000000;">
BaseDataAgent agent
result =<span style="color: #000000;"> agent.QueryStationDetail(json);
<span style="color: #0000ff;">if(result != <span style="color: #0000ff;">null &&<span style="color: #000000;"> result.success)
{
<span style="color: #0000ff;">foreach(StationDetailJson detail <span style="color: #0000ff;">in<span style="color: #000000;"> result.data)
{
StationInfo info =<span style="color: #000000;"> detail.ConvertInfo();
<span style="color: #0000ff;">try<span style="color: #000000;">
{
BLLFactory
}
{
LogTextHelper.Error(ex);
LogTextHelper.Info(info.ToJson());
}
}
}
}
<span style="color: #0000ff;">catch<span style="color: #000000;"> (Exception ex)
{
result.errmsg =<span style="color: #000000;"> ex.Message;
result.success = <span style="color: #0000ff;">false<span style="color: #000000;">;
LogTextHelper.Error(ex);
}
}
}</span></pre>
4、博客的RSS
原则上我们可以利用任何RSS来源来获取响应的博客内容,这里我以自己博客园的RSS源进行介绍使用,我们每个博客园的账号都有一个如下的连接,提供我们最新的博客列表信息。
打开连接,可以看到它的内容就是最新显示的博客内容,如下所示
?
处理RSS的内容,我们使用内置的SyndicationFeed对象来处理即可,非常方便。
上面代码就是获取到对应的RSS内容,然后把它们转换为XMLReader进行解析即可。
然后可以通过一个遍历的处理就可以获取到其中各个的XML节点内容了,非常方便。
5、在微信应用中发送博客内容
通过上面的RSS读取操作,我们可以获得对应的博客内容,如果我们需要每周给客户发送一些内容,那么这些就可以通过上面RSS源进行处理发送了。
关于发送文本消息的处理,可以参考我的随笔文章《》
这里我就直接应用上面的接口对内容进行处理发送,具体接口的逻辑就不再罗列。
ICustomerApi api </span>= <span style="color: #0000ff;">new</span><span style="color: #000000;"> CustomerApi();
</span><span style="color: #0000ff;">foreach</span> (SyndicationItem item <span style="color: #0000ff;">in</span><span style="color: #000000;"> feed.Items)
{
Console.WriteLine(item.ToJson());
</span><span style="color: #0000ff;">var</span> id =<span style="color: #000000;"> item.Id;
</span><span style="color: #0000ff;">string</span> subject =<span style="color: #000000;"> item.Title.Text;
</span><span style="color: #0000ff;">string</span> summary =<span style="color: #000000;"> item.Summary.Text;
</span><span style="color: #0000ff;">var</span> content = <span style="color: #0000ff;">string</span>.Format(<span style="color: #800000;">"</span><span style="color: #800000;"><a href='{0}'>{1}</a></span><span style="color: #800000;">"</span><span style="color: #000000;">,id,subject);
CommonResult result </span>=<span style="color: #000000;"> api.SendText(token,openId,content);
Console.WriteLine(</span><span style="color: #800000;">"</span><span style="color: #800000;">发送内容:</span><span style="color: #800000;">"</span> + (result.Success ? <span style="color: #800000;">"</span><span style="color: #800000;">成功</span><span style="color: #800000;">"</span> : <span style="color: #800000;">"</span><span style="color: #800000;">失败:</span><span style="color: #800000;">"</span> +<span style="color: #000000;"> result.ErrorMessage));
}
}</span></pre>
得到的界面效果如下所示。
但是这样的效果还是有点差强人意,我们知道微信里面有图文消息的接口,可以利用图文消息的接口进行发送,则更加美观一些。
调整后的代码如下所示。
</span><span style="color: #0000ff;">string</span> url = <span style="color: #800000;">"</span><span style="color: #800000;">http://feed.cnblogs.com/blog/u/12391/rss</span><span style="color: #800000;">"</span><span style="color: #000000;">;
XmlReader reader </span>=<span style="color: #000000;"> XmlReader.Create(url);
SyndicationFeed feed </span>=<span style="color: #000000;"> SyndicationFeed.Load(reader);
reader.Close();
</span><span style="color: #0000ff;">int</span> i = <span style="color: #800080;">0</span><span style="color: #000000;">;
</span><span style="color: #0000ff;">foreach</span> (SyndicationItem item <span style="color: #0000ff;">in</span><span style="color: #000000;"> feed.Items)
{
list.Add(
</span><span style="color: #0000ff;">new</span><span style="color: #000000;"> ArticleEntity
{
Title </span>=<span style="color: #000000;"> item.Title.Text,Description </span>=<span style="color: #000000;"> item.Summary.Text,PicUrl </span>= i == <span style="color: #800080;">0</span> ? <span style="color: #800000;">"</span><span style="color: #800000;">http://www.iqidi.com/Content/Images/cnblogs_whc.png</span><span style="color: #800000;">"</span> : <span style="color: #800000;">"</span><span style="color: #800000;">http://www.iqidi.com/Content/Images/frame_web.png</span><span style="color: #800000;">"</span><span style="color: #000000;">,Url </span>=<span style="color: #000000;"> item.Id
});
</span><span style="color: #0000ff;">if</span>(i >= <span style="color: #800080;">8</span><span style="color: #000000;">)
{
</span><span style="color: #0000ff;">break</span><span style="color: #000000;">;
}
i</span>++<span style="color: #000000;">;
}
ICustomerApi customerApi </span>= <span style="color: #0000ff;">new</span><span style="color: #000000;"> CustomerApi();
</span><span style="color: #0000ff;">var</span> result =<span style="color: #000000;"> customerApi.SendNews(token,list);
}</span></pre>
这样就是发送图文消息的代码,需要重新构建一个实体类集合进行发送,得到发送的效果如下所示。
整体的界面效果就是我们需要的效果了,不过如果我们需要使用批量发送给订阅用户的话,那么我们需要使用消息的群发接口,群发的消息接口封装如需了解,可以参考文章《》。
整个群发消息的逻辑代码如下所示,主要逻辑就是获取博客文章,并上传文章的图片,接着上传需要群发的图文消息资源,最后调用群发接口进行消息的发送即可。
</span><span style="color: #0000ff;">string</span> url = <span style="color: #800000;">"</span><span style="color: #800000;">http://feed.cnblogs.com/blog/u/12391/rss</span><span style="color: #800000;">"</span><span style="color: #000000;">;
XmlReader reader </span>=<span style="color: #000000;"> XmlReader.Create(url);
SyndicationFeed feed </span>=<span style="color: #000000;"> SyndicationFeed.Load(reader);
reader.Close();
</span><span style="color: #008000;">//</span><span style="color: #008000;">上传图片获取MediaId</span>
IMediaApi mediaApi = <span style="color: #0000ff;">new</span><span style="color: #000000;"> MediaApi();
</span><span style="color: #0000ff;">var</span> result1 = mediaApi.UploadTempMedia(token,UploadMediaFileType.image,<span style="color: #800000;">@"</span><span style="color: #800000;">E:我的网站资料iqidiSoftwarecontentimagescnblogs_whc.png</span><span style="color: #800000;">"</span>);<span style="color: #008000;">//</span><span style="color: #008000;">"</span><span style="color: #008000; text-decoration: underline;">http://www.iqidi.com/Content/Images/cnblogs_whc.png</span><span style="color: #008000;">");</span>
<span style="color: #0000ff;">var</span> result2 = mediaApi.UploadTempMedia(token,<span style="color: #800000;">@"</span><span style="color: #800000;">E:我的网站资料iqidiSoftwarecontentimagesframe_web.png</span><span style="color: #800000;">"</span>);<span style="color: #008000;">//</span><span style="color: #008000;">"</span><span style="color: #008000; text-decoration: underline;">http://www.iqidi.com/Content/Images/frame_web.png</span><span style="color: #008000;">");</span>
<span style="color: #0000ff;">if</span> (result1 != <span style="color: #0000ff;">null</span> && result2 != <span style="color: #0000ff;">null</span><span style="color: #000000;">)
{
</span><span style="color: #0000ff;">int</span> i = <span style="color: #800080;">0</span><span style="color: #000000;">;
</span><span style="color: #0000ff;">foreach</span> (SyndicationItem item <span style="color: #0000ff;">in</span><span style="color: #000000;"> feed.Items)
{
list.Add(
</span><span style="color: #0000ff;">new</span><span style="color: #000000;"> NewsUploadJson
{
author </span>= <span style="color: #800000;">"</span><span style="color: #800000;">伍华聪</span><span style="color: #800000;">"</span><span style="color: #000000;">,title </span>=<span style="color: #000000;"> item.Title.Text,content </span>=<span style="color: #000000;"> item.Summary.Text,</span><span style="color: #008000;">//</span><span style="color: #008000;">digest = item.Summary.Text,</span>
thumb_media_id = i == <span style="color: #800080;">0</span> ?<span style="color: #000000;"> result1.media_id : result2.media_id,content_source_url </span>=<span style="color: #000000;"> item.Id,});
</span><span style="color: #0000ff;">if</span> (i >= <span style="color: #800080;">8</span><span style="color: #000000;">)
{
</span><span style="color: #0000ff;">break</span><span style="color: #000000;">;
}
i</span>++<span style="color: #000000;">;
}
}
</span><span style="color: #0000ff;">if</span> (list.Count > <span style="color: #800080;">0</span><span style="color: #000000;">)
{
UploadJsonResult resultNews </span>=<span style="color: #000000;"> mediaApi.UploadNews(token,list);
</span><span style="color: #0000ff;">if</span> (resultNews != <span style="color: #0000ff;">null</span><span style="color: #000000;">)
{
IMassSendApi massApi </span>= <span style="color: #0000ff;">new</span><span style="color: #000000;"> MassSendApi();
</span><span style="color: #0000ff;">var</span> result = massApi.SendByGroup(token,MassMessageType.mpnews,resultNews.media_id,<span style="color: #800000;">"</span><span style="color: #800000;">0</span><span style="color: #800000;">"</span>,<span style="color: #0000ff;">true</span><span style="color: #000000;">);
}
</span><span style="color: #0000ff;">else</span><span style="color: #000000;">
{
Console.WriteLine(</span><span style="color: #800000;">"</span><span style="color: #800000;">上传图文消息失败</span><span style="color: #800000;">"</span><span style="color: #000000;">);
}
}
}</span></pre>
群发的消息在微信上看到内容和前面的差不多,不过点击并不会直接跳转链接,而是进去到一个详细内容的页面里面,只有单击阅读原文才进行跳转URL,如下所示。
6.结合Quartz.NET实现博客文章内容的定期推送功能
在Web微信应用中使用博客RSS以及Quartz.NET实现文章内容的定期推送功能,我们需要结合Quartz.NET的作业调度处理、微信接口的内容发送,以及博文RSS内容的获取处理,三者整合进行实现整个功能。
首先我们根据上面的代码,设计好调度的Job内容,如下所示。
然后,在Web应用的Global.asa的后台代码里面,编写代码启动作业调度即可。
而根据前面Corn表达式的说明,我们要每周定时发送一次的的规则,如下所示。
每周星期天凌晨1点实行一次:?0 0 1 ? * L
这样我们最终的Globa.asa后台代码如下所示。
</span><span style="color: #0000ff;">void</span> Application_Start(<span style="color: #0000ff;">object</span><span style="color: #000000;"> sender,EventArgs e)
{
</span><span style="color: #008000;">//</span><span style="color: #008000;"> 在应用程序启动时运行的代码</span>
<span style="color: #000000;"> AreaRegistration.RegisterAllAreas();
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
</span><span style="color: #008000;">//</span><span style="color: #008000;">构造调度对象,并创建对应的调度任务</span>
<span style="color: #ff0000;"> scheduler = StdSchedulerFactory.GetDefaultScheduler();
CalendarTask();
</span><span style="color: #008000;">//</span><span style="color: #008000;">启动所有的任务</span>
<span style="color: #000000;"><span style="color: #ff0000;"> scheduler.Start();
}
</span><span style="color: #0000ff;">protected</span> <span style="color: #0000ff;">void</span> Application_End(<span style="color: #0000ff;">object</span><span style="color: #000000;"> sender,EventArgs e)
{
</span><span style="color: #0000ff;">if</span> (scheduler != <span style="color: #0000ff;">null</span><span style="color: #000000;">)
{
scheduler.Shutdown(</span><span style="color: #0000ff;">true</span><span style="color: #000000;">);
}
}
</span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> CalendarTask()
{
IJobDetail job </span>= JobBuilder.Create<BlogArticleSendJob><span style="color: #000000;">()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">BlogArticleSendJob</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span><span style="color: #000000;">)
.Build();
</span><span style="color: #008000;">//</span><span style="color: #008000;">每周星期天凌晨1点实行一次:0 0 1 ? * L</span>
ICronTrigger trigger =<span style="color: #000000;"> (ICronTrigger)TriggerBuilder.Create()
.WithIdentity(</span><span style="color: #800000;">"</span><span style="color: #800000;">trigger1</span><span style="color: #800000;">"</span>,<span style="color: #800000;">"</span><span style="color: #800000;">group1</span><span style="color: #800000;">"</span><span style="color: #000000;">)
.WithCronSchedule(</span><span style="color: #800000;">"</span><span style="color: #ff0000;">0 0 1 ? * L</span><span style="color: #800000;">"</span>)<span style="color: #008000;">//</span><span style="color: #008000;">0 0 1 ? * L</span>
<span style="color: #000000;"> .Build();
DateTimeOffset ft </span>=<span style="color: #000000;"> scheduler.ScheduleJob(job,DateTime.Now));
}</span></pre>
综合上面的思路,我们可以利用Quartz.NET做成更多的数据同步任务调度,另外在微信应用中,我们也可以整合很多组件或者控件,来实现更加弹性化的业务支持,如消息群发、访客汇总,内容同步等处理。
以上就是我的一些组件代码的应用思路,其实我们只要涉猎更广一些,很多东西可以使用拿来主义,经过自己的整合优化,可以为我们服务的更好。
(编辑:李大同)
【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!