【转】SharePoint定时器 (SPJobDefinition)

需要写一个自定义的sharepoint timer job, 目的是要定时到Site Collection Images这个List里检查图片的过期日期,如果即将过期的话,需要发送email到相关的人员。

 

GOOGLE了一下,发现资料很少。不过幸运的是我发现了 Andrew Connell的博客,上面有一个很详细的示例代码。http://www.andrewconnell.com/blog/articles /CreatingCustomSharePointTimerJobs.aspx

 

代码的基本思路是:我们的自定义job类需要从SPJobDefinition继承过来,并且要改写几个构造体和一个Execute方法。其中不带参数的默认构造体是必须的。

那么在Execute方法中怎么得到这个SPWebApplication对象呢?AC用了如下的代码:

SPWebApplication webApplication = this.Parent as SPWebApplication;

用Parent属性并且CAST成WebApplication对象, 这段我比较困扰,因为根据SDK里的说明,本来就有一个WebApplication属性可以用,为什么还要这么麻烦呢。我试了如下代码

SPWebApplication webApplication = this.WebApplication;

测试结果发现完全等效。

 

 

好,这个类写完以后,我们还要写另外一个SPFeatureReceiver类。因为我们需要一个Feature来把我们的Timer部署到服务器上去,通过Feature Activated/deactivated来触发/关闭这个TimerJob。

 

代码
public override void FeatureActivated(SPFeatureReceiverProperties properties)
        {
            SPSite site 
= properties.Feature.Parent as SPSite;
            SPWeb web 
= site.RootWeb;

          
            
// make sure the job isn't already registered
            
foreach (SPJobDefinition job in site.WebApplication.JobDefinitions)
            {
                
if (job.Name == "ImageValidationCheckerJob")
                {
                    job.Delete();
                }
            }

            
// install the job
            ImageValidationCheckerJob imageValidationCheckerJob 
= new ImageValidationCheckerJob("ImageValidationCheckerJob", site.WebApplication);
            SPDailySchedule schedule 
= new SPMonthlySchedule();

            schedule.BeginHour 
= 23;
            schedule.BeginMinute 
= 40;
            schedule.BeginSecond 
= 1;          
            schedule.EndHour 
= 23;
            schedule.EndMinute 
= 59;
            schedule.EndSecond 
= 1;

            imageValidationCheckerJob.Schedule 
= schedule;
            imageValidationCheckerJob.Update();
        }

public override void FeatureDeactivating(SPFeatureReceiverProperties properties)
        {
            SPSite site 
= properties.Feature.Parent as SPSite;

            
// delete the job
            
foreach (SPJobDefinition job in site.WebApplication.JobDefinitions)
            {
                
if (job.Name == "ImageValidationCheckerJob")
                {
                    job.Delete();
                }
            }
        }


这个SPDailySchedule对象让我有些困扰,我现在理解是你要给定开始时间和结束时间,然后系统会随机算出一个在两者中间的时刻来启动这 个工作。比如上面例子,那么每天运行时间应该是介于23:40:1 到23:59:1 之间, 不知道是不是这样,不过我猜测应该是如此。

 

代码写完了以后,还要写一个Feature,这个Feature比较简单,就一个XML文件

Feature.xml:


代码
<?xml version="1.0" encoding="utf-8" ?>
<Feature xmlns="http://schemas.microsoft.com/sharepoint/"
         Id
="1F481C17-4FDA-4919-A64A-EAE5C1301B4B"
         Title
="Image Validation Checker"
         Description
="If any images in the top level site colleciton images are expiring soon,email relative person."
         Scope
="Site"
         Hidden
="TRUE"
         Version
="1.0.0.0"
         ReceiverAssembly
="TimerJobControl, Version=1.0.0.0, Culture=neutral, PublicKeyToken=f2aef6a9088f714f"
ReceiverClass
="TimerJobControl.ImageValidationCheckerJobInstaller">
</Feature>


剩下的就是把feature装好就好了,假如代码没有出错的话。

至于怎么Debug这个程序,走以下步骤:(有些步骤有时候可能不用,但是全用可以保证没有问题,这个是我花了1天的时间得出的血的教训)

(为了调试方面,把Schedule设置为SPMinuteSchedule,并且设置为每2分钟运行一次)

把Assembly DLL放到 GAC
命令行:iisreset
先Deactivate feature, 然后activate feature.
命令行:net stop SPTimerV3
命令行:net start SPTimerV3
Visual Studio: Attach to process: OWSTIMER.EXE
完。

还有一点要注意的是,如果你想在TimerJob类里面要从web.config文件得到一些值得话,我觉得是不可能的,至少我还没有发现该怎么做,所以,假如代码需要从外部取得一些信息的话,

解决办法是 :在 c:/program files/common files/microsoft shared/…/12/bin目录里新建一个文件,文件名叫做:OwsTimer.exe.config,

<configuration>

<appSettings>

   <add key="YourKey" value="YourValue" />

</appSettings>

</configuration>

然后用ConfigurationManager.AppSettings.Get("YourKey"); 来取得这个值。

记住每次修改代码以后进行调试前,一定要走上面说的步骤,特别是步骤4和5很容易被忘记,不然可能会遇到很莫名其妙的问题。

最后,happy SharePointING。

posted @ 2010-10-23 12:39  Glife  阅读(559)  评论(0编辑  收藏  举报