第十七篇的Trigger用法为通过Handler方式实现Trigger的封装,此种好处是一个Handler对应一个sObject,使本该在Trigger中写的代码分到Handler中,代码更加清晰。
十七篇链接:salesforce 零基础学习(十七)Trigger用法
有的时候对于sObject的trigger处理复杂的情况下,比如一个sObject的before update要实现功能1,2.....n功能情况下,Handler中需要在before update写实现功能1--n的代码。然而有些时候,我们在执行update情况下只需要让他触发功能i的功能代码,使用上述Handler方法也可以搞定,只不过处理起来比较尴尬,此篇针对trigger中对于不同业务分成模块进行处理。
一.Triggers基类
Triggers基类主要有以下内容:
1.枚举:封装的枚举包含了Trigger中的所有情况;
2.Handler接口:此接口中声明了一个方法handle,所有实现此接口的类都需要重写次方法;
3.bind方法,用于绑定事件以及实现Handler接口的类,即绑定的事件会使用实现Handler接口的类进行业务逻辑处理;
4.execute方法,用于执行triggers中绑定的Handler。
代码如下:
public class Triggers { public enum Evt
{
AfterDelete,
AfterInsert,
AfterUndelete,
AfterUpdate,
BeforeDelete,
BeforeInsert,
BeforeUpdate
} public interface Handler
{
void handle();
} Map<String, List<Handler>> eventHandlerMapping = new Map<String, List<Handler>>(); public Triggers bind(Evt event, Handler eh)
{
List<Handler> handlers = eventHandlerMapping.get(event.name());
if (handlers == null)
{
handlers = new List<Handler>();
eventHandlerMapping.put(event.name(), handlers);
}
handlers.add(eh);
return this;
} public void execute()
{
Evt ev = null;
if(Trigger.isInsert && Trigger.isBefore)
{
ev = Evt.beforeinsert;
}
else if(Trigger.isInsert && Trigger.isAfter)
{
ev = Evt.afterinsert;
}
else if(Trigger.isUpdate && Trigger.isBefore)
{
ev = Evt.beforeupdate;
}
else if(Trigger.isUpdate && Trigger.isAfter)
{
ev = Evt.afterupdate;
}
else if(Trigger.isDelete && Trigger.isBefore)
{
ev = Evt.beforedelete;
}
else if(Trigger.isDelete && Trigger.isAfter)
{
ev = Evt.afterdelete;
}
else if(Trigger.isundelete)
{
ev = Evt.afterundelete;
}
List<Handler> handlers = eventHandlerMapping.get(ev.name());
if (handlers != null && !handlers.isEmpty())
{
for (Handler h : handlers)
{
h.handle();
}
}
}
}
Triggers
二.相关的实现Handler接口的类
此处例举两个类,分别实现Triggers.Handler并且实现相关的Handle方法
1.F1Handler
public without sharing class F1Handler implements Triggers.Handler {
public void Handle(){
List<Company_Info__c> companyInfoList = trigger.new;
//TODO
//do something start
system.debug('===============executeF1Handler');
//do something end
}
}
2.F2Handler
public with sharing class F2Handler implements Triggers.Handler {
public void Handle(){
List<Company_Info__c> companyInfoList = trigger.new;
//TODO
//do something start
system.debug('===============executeF2Handler');
//do something end
}
}
三.相关Trigger中Handler是否执行的Helper类
有的时候业务需要不同地方的入口进行相同的操作,需要执行不同的trigger业务模块,比如在controller正常的更新需要全部执行trigger方法模块,在其他trigger中更新此sObject则只需要一部分模块,这个时候需要相关Helper类方法来控制是否执行哪块逻辑。
public without sharing class TriggerExecutionHelper {
public static Boolean enableExecuteF1 {
get{
if(enableExecuteF1 == null) {
enableExecuteF1 = true;
}
return enableExecuteF1;
}
set;
} public static Boolean enableExecuteF2{
get{
if(enableExecuteF2 == null) {
enableExecuteF2 = true;
}
return enableExecuteF2;
}
set;
} }
四.Trigger代码部分
trigger代码部分用来判断是否需要执行哪些情况trigger
trigger CompanyInfoTrigger on Company_Info__c (after delete, after insert, after undelete, after update, before delete, before insert, before update) { Triggers companyInfoTrigger = new Triggers();
if(TriggerExecutionHelper.enableExecuteF1) {
companyInfoTrigger.bind(Triggers.Evt.BeforeInsert, new F1Handler());
} if(TriggerExecutionHelper.enableExecuteF2) {
companyInfoTrigger.bind(Triggers.Evt.BeforeInsert, new F2Handler());
} companyInfoTrigger.Execute(); }
效果展示:
1.通过系统页面插入一条数据,默认走全部的trigger内容
相关log可以查看执行结果
2.通过匿名块执行insert则只执行trigger中一个业务模块代码
总结:如果业务逻辑特别清晰并且可以分模块处理,则可以使用此种方法,达到的效果为业务清晰明了,后期人员也便于维护。缺点为如果相关模块的Handler都对一个字段进行处理,则会有相关先后处理的问题以及出bug调试时间增多等,具体使用哪种还是看具体的业务以及个人使用习惯,业务简单直接写在trigger中也未尝不可。篇中有问题地方欢迎指出,有不懂的欢迎留言。