Puncak - Pola Desain Pemicu
Pola desain digunakan untuk membuat kode kita lebih efisien dan untuk menghindari mencapai batas pengatur. Seringkali pengembang dapat menulis kode yang tidak efisien yang dapat menyebabkan contoh objek berulang. Hal ini dapat mengakibatkan kode yang tidak efisien dan berkinerja buruk, dan berpotensi melanggar batas gubernur. Ini paling sering terjadi pada pemicu, karena dapat beroperasi pada sekumpulan record.
Kita akan melihat beberapa strategi pola desain penting dalam bab ini.
Pola Desain Pemicu Massal
Dalam kasus bisnis nyata, Anda mungkin perlu memproses ribuan catatan sekaligus. Jika pemicu Anda tidak dirancang untuk menangani situasi seperti itu, pemicu mungkin gagal saat memproses rekaman. Ada beberapa praktik terbaik yang perlu Anda ikuti saat menerapkan pemicu. Semua pemicu adalah pemicu massal secara default, dan dapat memproses beberapa catatan dalam satu waktu. Anda harus selalu merencanakan untuk memproses lebih dari satu rekaman dalam satu waktu.
Pertimbangkan kasus bisnis, di mana, Anda perlu memproses sejumlah besar catatan dan Anda telah menulis pemicunya seperti yang diberikan di bawah ini. Ini adalah contoh yang sama yang kami ambil untuk memasukkan catatan faktur ketika Status Pelanggan berubah dari Tidak Aktif menjadi Aktif.
// Bad Trigger Example
trigger Customer_After_Insert on APEX_Customer__c (after update) {
for (APEX_Customer__c objCustomer: Trigger.new) {
if (objCustomer.APEX_Customer_Status__c == 'Active' &&
trigger.oldMap.get(objCustomer.id).APEX_Customer_Status__c == 'Inactive') {
// condition to check the old value and new value
APEX_Invoice__c objInvoice = new APEX_Invoice__c();
objInvoice.APEX_Status__c = 'Pending';
insert objInvoice; //DML to insert the Invoice List in SFDC
}
}
}
Anda sekarang dapat melihat bahwa Pernyataan DML telah ditulis untuk blok loop yang akan berfungsi saat memproses hanya beberapa catatan tetapi ketika Anda memproses beberapa ratus catatan, itu akan mencapai batas Pernyataan DML per transaksi yang merupakan governor limit. Kami akan melihat secara mendetail tentang Batas Gubernur di bab berikutnya.
Untuk menghindarinya, kita harus membuat pemicu efisien untuk memproses banyak record dalam satu waktu.
Contoh berikut akan membantu Anda memahami hal yang sama -
// Modified Trigger Code-Bulk Trigger
trigger Customer_After_Insert on APEX_Customer__c (after update) {
List<apex_invoice__c> InvoiceList = new List<apex_invoice__c>();
for (APEX_Customer__c objCustomer: Trigger.new) {
if (objCustomer.APEX_Customer_Status__c == 'Active' &&
trigger.oldMap.get(objCustomer.id).APEX_Customer_Status__c == 'Inactive') {
//condition to check the old value and new value
APEX_Invoice__c objInvoice = new APEX_Invoice__c();
objInvoice.APEX_Status__c = 'Pending';
InvoiceList.add(objInvoice);//Adding records to List
}
}
insert InvoiceList;
// DML to insert the Invoice List in SFDC, this list contains the all records
// which need to be modified and will fire only one DML
}
Pemicu ini hanya akan mengaktifkan 1 pernyataan DML karena akan beroperasi di atas Daftar dan Daftar memiliki semua catatan yang perlu dimodifikasi.
Dengan cara ini, Anda dapat menghindari batasan gubernur pernyataan DML.
Kelas Trigger Helper
Menulis seluruh kode di trigger juga bukan praktik yang baik. Karenanya Anda harus memanggil kelas Apex dan mendelegasikan pemrosesan dari kelas Trigger ke Apex seperti yang ditunjukkan di bawah ini. Kelas Trigger Helper adalah kelas yang melakukan semua pemrosesan untuk pemicu.
Mari kita pertimbangkan lagi contoh pembuatan catatan faktur kita.
// Below is the Trigger without Helper class
trigger Customer_After_Insert on APEX_Customer__c (after update) {
List<apex_invoice__c> InvoiceList = new List<apex_invoice__c>();
for (APEX_Customer__c objCustomer: Trigger.new) {
if (objCustomer.APEX_Customer_Status__c == 'Active' &&
trigger.oldMap.get(objCustomer.id).APEX_Customer_Status__c == 'Inactive') {
// condition to check the old value and new value
APEX_Invoice__c objInvoice = new APEX_Invoice__c();
objInvoice.APEX_Status__c = 'Pending';
InvoiceList.add(objInvoice);
}
}
insert InvoiceList; // DML to insert the Invoice List in SFDC
}
// Below is the trigger with helper class
// Trigger with Helper Class
trigger Customer_After_Insert on APEX_Customer__c (after update) {
CustomerTriggerHelper.createInvoiceRecords(Trigger.new, trigger.oldMap);
// Trigger calls the helper class and does not have any code in Trigger
}
Kelas Pembantu
public class CustomerTriggerHelper {
public static void createInvoiceRecords (List<apex_customer__c>
customerList, Map<id, apex_customer__c> oldMapCustomer) {
List<apex_invoice__c> InvoiceList = new Listvapex_invoice__c>();
for (APEX_Customer__c objCustomer: customerList) {
if (objCustomer.APEX_Customer_Status__c == 'Active' &&
oldMapCustomer.get(objCustomer.id).APEX_Customer_Status__c == 'Inactive') {
// condition to check the old value and new value
APEX_Invoice__c objInvoice = new APEX_Invoice__c();
// objInvoice.APEX_Status__c = 'Pending';
InvoiceList.add(objInvoice);
}
}
insert InvoiceList; // DML to insert the Invoice List in SFDC
}
}
Dalam hal ini, semua pemrosesan telah didelegasikan ke kelas helper dan ketika kita membutuhkan fungsionalitas baru, kita cukup menambahkan kode ke kelas helper tanpa mengubah pemicu.
Pemicu Tunggal pada Setiap sObject
Selalu buat satu pemicu pada setiap objek. Beberapa pemicu pada objek yang sama dapat menyebabkan konflik dan kesalahan jika mencapai batas pengatur.
Anda dapat menggunakan variabel konteks untuk memanggil metode yang berbeda dari kelas helper sesuai kebutuhan. Perhatikan contoh kami sebelumnya. Misalkan metode createInvoice kami harus dipanggil hanya saat rekaman diperbarui dan di beberapa acara. Kemudian kita dapat mengontrol eksekusi seperti di bawah ini -
// Trigger with Context variable for controlling the calling flow
trigger Customer_After_Insert on APEX_Customer__c (after update, after insert) {
if (trigger.isAfter && trigger.isUpdate) {
// This condition will check for trigger events using isAfter and isUpdate
// context variable
CustomerTriggerHelper.createInvoiceRecords(Trigger.new);
// Trigger calls the helper class and does not have any code in Trigger
// and this will be called only when trigger ids after update
}
}
// Helper Class
public class CustomerTriggerHelper {
//Method To Create Invoice Records
public static void createInvoiceRecords (List<apex_customer__c> customerList) {
for (APEX_Customer__c objCustomer: customerList) {
if (objCustomer.APEX_Customer_Status__c == 'Active' &&
trigger.oldMap.get(objCustomer.id).APEX_Customer_Status__c == 'Inactive') {
// condition to check the old value and new value
APEX_Invoice__c objInvoice = new APEX_Invoice__c();
objInvoice.APEX_Status__c = 'Pending';
InvoiceList.add(objInvoice);
}
}
insert InvoiceList; // DML to insert the Invoice List in SFDC
}
}