Apex-거버너 제한

관리자 실행 제한은 Force.com 다중 테넌트 플랫폼에서 리소스를 효율적으로 사용하도록합니다. 효율적인 처리를 위해 코드 실행에 대해 Salesforce.com에서 지정한 제한입니다.

주지사 한도 란 무엇입니까?

아시다시피 Apex는 다중 테넌트 환경에서 실행됩니다. 즉, 모든 고객과 조직이 단일 리소스를 공유합니다. 따라서 아무도 리소스를 독점하지 않도록해야합니다. 따라서 Salesforce.com은 코드 실행을 제어하고 제한하는 제한 집합을 만들었습니다. 거버너 한도를 초과 할 때마다 오류가 발생하고 프로그램 실행이 중지됩니다.

개발자의 관점에서 볼 때 코드가 확장 가능하고 한계에 도달하지 않도록하는 것이 중요합니다.

이러한 모든 한도는 거래별로 적용됩니다. 단일 트리거 실행은 하나의 트랜잭션입니다.

앞서 살펴본 것처럼 트리거 디자인 패턴은 한계 오류를 방지하는 데 도움이됩니다. 이제 다른 중요한 한계를 살펴 보겠습니다.

SOQL 쿼리 제한 방지

트랜잭션 당 100 개의 쿼리 만 실행할 수 있습니다. 즉, 코드가 100 개 이상의 SOQL 쿼리를 실행하면 오류가 발생합니다.

이 예는 SOQL 쿼리 제한에 도달하는 방법을 보여줍니다.

다음 트리거는 고객 목록을 반복하고 '결제 확인'문자열로 하위 레코드의 (인보이스) 설명을 업데이트합니다.

// Helper class:Below code needs o be checked.
public class CustomerTriggerHelper {
  
  public static void isAfterUpdateCall(Trigger.new) {
      createInvoiceRecords(trigger.new);//Method call
      updateCustomerDescription(trigger.new);
   }
   
   // 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
   }
   
   // Method to update the invoice records
   public static updateCustomerDescription (List<apex_customer__c> customerList) {
      for (APEX_Customer__c objCust: customerList) {
         List<apex_customer__c> invList = [SELECT Id, Name,
            APEX_Description__c FROM APEX_Invoice__c WHERE APEX_Customer__c = :objCust.id];
         
         // This query will fire for the number of records customer list has and will
         // hit the governor limit when records are more than 100
         for (APEX_Invoice__c objInv: invList) {
            objInv.APEX_Description__c = 'OK To Pay';
            update objInv;
            // Update invoice, this will also hit the governor limit for DML if large
            // number(150) of records are there
         }
      }
   }
}

'updateCustomerDescription'메서드가 호출되고 고객 레코드 수가 100 개를 초과하면 SOQL 제한에 도달합니다. 이를 방지하려면 For 루프에 SOQL 쿼리를 작성하지 마십시오. 이 경우 SOQL 쿼리가 For 루프에 작성되었습니다.

다음은 DML과 SOQL 제한을 피하는 방법을 보여주는 예입니다. 중첩 된 관계 쿼리를 사용하여 송장 레코드를 가져오고 컨텍스트 변수를 사용했습니다.trigger.newMap ID 및 고객 레코드의지도를 가져옵니다.

// SOQL-Good Way to Write Query and avoid limit exception
// Helper Class
public class CustomerTriggerHelper {
   public static void isAfterUpdateCall(Trigger.new) {
      createInvoiceRecords(trigger.new);  //Method call
      updateCustomerDescription(trigger.new, trigger.newMap);
   }
   
   // 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
   }
   
   // Method to update the invoice records
   public static updateCustomerDescription (List<apex_customer__c>
      customerList, Map<id, apex_customer__c> newMapVariable) {
      List<apex_customer__c> customerListWithInvoice = [SELECT id,
         Name,(SELECT Id, Name, APEX_Description__c FROM APEX_Invoice__r) FROM
         APEX_Customer__c WHERE Id IN :newMapVariable.keySet()];
      
      // Query will be for only one time and fetches all the records
      List<apex_invoice__c> invoiceToUpdate = new
      List<apex_invoice__c>();
      
      for (APEX_Customer__c objCust: customerList) {
         for (APEX_Invoice__c objInv: invList) {
            objInv.APEX_Description__c = 'OK To Pay';
            invoiceToUpdate.add(objInv);
            // Add the modified records to List
         }
      }
      update invoiceToUpdate;
   }
}

DML 대량 호출

이 예제는 트리거 도우미 클래스 패턴과 함께 벌크 트리거를 보여줍니다. 먼저 도우미 클래스를 저장 한 다음 트리거를 저장해야합니다.

Note − 앞서 생성 한 'CustomerTriggerHelper'클래스에 아래 코드를 붙여 넣습니다.

// Helper Class
public class CustomerTriggerHelper {
   public static void isAfterUpdateCall(List<apex_customer__c> customerList,
      Map<id, apex_customer__c> mapIdToCustomers, Map<id, apex_customer__c>
      mapOldItToCustomers) {
      createInvoiceRecords(customerList, mapOldItToCustomers);   //Method call
      updateCustomerDescription(customerList,mapIdToCustomers,
      mapOldItToCustomers);
   }
   
   // Method To Create Invoice Records
   public static void createInvoiceRecords (List<apex_customer__c>
      customerList, Map<id, apex_customer__c> mapOldItToCustomers) {
      List<apex_invoice__c> InvoiceList = new List<apex_invoice__c>();
      List<apex_customer__c> customerToInvoice = [SELECT id, Name FROM
         APEX_Customer__c LIMIT 1];
      
      for (APEX_Customer__c objCustomer: customerList) {
         if (objCustomer.APEX_Customer_Status__c == 'Active' &&
            mapOldItToCustomers.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';
            objInvoice.APEX_Customer__c = objCustomer.id;
            InvoiceList.add(objInvoice);
         }
      }
      system.debug('InvoiceList&&&'+InvoiceList);
      insert InvoiceList;
      // DML to insert the Invoice List in SFDC. This also follows the Bulk pattern
   }
   
   // Method to update the invoice records
   public static void updateCustomerDescription (List<apex_customer__c>
      customerList, Map<id, apex_customer__c> newMapVariable, Map<id,
      apex_customer__c> oldCustomerMap) {
      List<apex_customer__c> customerListWithInvoice = [SELECT id,
      Name,(SELECT Id, Name, APEX_Description__c FROM Invoices__r) FROM
         APEX_Customer__c WHERE Id IN :newMapVariable.keySet()];
   
      // Query will be for only one time and fetches all the records
      List<apex_invoice__c> invoiceToUpdate = new List<apex_invoice__c>();
      List<apex_invoice__c> invoiceFetched = new List<apex_invoice__c>();
      invoiceFetched = customerListWithInvoice[0].Invoices__r;
      system.debug('invoiceFetched'+invoiceFetched);
      system.debug('customerListWithInvoice****'+customerListWithInvoice);
   
      for (APEX_Customer__c objCust: customerList) {
         system.debug('objCust.Invoices__r'+objCust.Invoices__r);
         if (objCust.APEX_Active__c == true &&
            oldCustomerMap.get(objCust.id).APEX_Active__c == false) {
            for (APEX_Invoice__c objInv: invoiceFetched) {
               system.debug('I am in For Loop'+objInv);
               objInv.APEX_Description__c = 'OK To Pay';
               invoiceToUpdate.add(objInv);
               // Add the modified records to List
            }
         }
      }
     system.debug('Value of List ***'+invoiceToUpdate);
     update invoiceToUpdate;
      // This statement is Bulk DML which performs the DML on List and avoids
      // the DML Governor limit
   }
}

// Trigger Code for this class: Paste this code in 'Customer_After_Insert'
// trigger on Customer Object
trigger Customer_After_Insert on APEX_Customer__c (after update) {
   CustomerTriggerHelper.isAfterUpdateCall(Trigger.new, trigger.newMap,
      trigger.oldMap);
   // Trigger calls the helper class and does not have any code in Trigger
}

기타 Salesforce 관리자 제한

다음 표에는 중요한 총재 제한이 나열되어 있습니다.

기술 한도
총 힙 크기 6MB / 12MB
발행 된 총 DML 문 수 150
단일 SOSL 쿼리로 검색된 총 레코드 수 2000 년
발행 된 총 SOSL 쿼리 수 20
Database.getQueryLocator에서 검색 한 총 레코드 수 10000
SOQL 쿼리에서 검색 한 총 레코드 수 50000