Thursday, March 10, 2022

Multiple objects in single DML ?

This is an extension of our earlier post where we read about how we can insert both parent and child in a single DML. If you haven't read it yet visit it by clicking.

Note: Usage of single DML to insert/update different object is not just limited to objects which are in a relationship rather it could be simply the creation of unrelated different objects on the occurrence of a certain event.

We already know the limit of the number of records that can be updated by a DML is 10,000 (Remember the Limit is for the whole transaction, not just one DML but in our case, our transaction consists of only one DML).

Now questions come to mind, can I also insert List<Sobject> that contains a mixture of Accounts and Contact records in total (2,400) records.

Before reading further try to think of the answer...

The default expectation is "Yes", having multiple Objects in the List should not impact the 10,000 limits and 2,400 is way below 10,000. But unfortunately, that is not the case.

If you try to insert 1,200 Accounts and 1,200 Contacts that are unrelated, below is a snippet of code used to add records.



You will receive the below error.

Error



But why this error?

When we know we can easily insert 10,000 records of Accounts or any other object for that matter when inserted/updated individually. (Code example below).


On deep diving into the issue we figured, when working with multiple objects, salesforce traverses through the list and creates chunks of records. 

Chunk is nothing but a subset of input array or list and each chunk contains records as per the below rules.

  • Each chunk will only contain records of one object type.
  • If object changes while traversing the list, the chunk also changes.
Assume you have a list of records as A1, A2, C1, C2, A3. 
They will be distributed as 
Chunk 1 --> A1, A2 
Chunk 2 --> C1, C2
Chunk 3 --> A3

  • In a chunk maximum of 200 records can come, the 201st record will be placed in the 2nd chunk.
  • In total for a successful DML, you can have a maximum of 10 chunks. 
  • Derivation from the above rule we can also say we can have at most 10 different types of objects in a Single DML which will lead to 10 different chunks
  • Data is committed chunk by chunk ie if for example, any trigger exists on records it will be called once for each chunk of records.

Taking the above example as a base,  when the Account trigger is called ( Trigger.new will only contain A1 and A2 first time then it will be called again (for chunk 3) and this time it will only contain A3.


Note: Chunking comes into the picture only if you are working with multiple objects, when doing data manipulation on a single object no chunking takes place.


Let's look at another example where DML could fail.

Example 1 :

Here we are adding different instances of multiple objects in a single list and inserting them.


But if you try to run above code it will fail although we are only inserting 11 records, it fails as it will lead to 11 chunks.


Just because every time an object changes, chunk also changes we can simply sort all the same type of object instances so they are part of a single chunk (200 max). 
We can resolve the above code by simply using sort() method which will sort the list putting all same types of instances in order and which will reduce chunk size to 5.(Code below) 


To resolve the earlier example of 1200 Accounts and 1200 Contacts we will have to distribute it in two DMLs as it can't be achieved via a single DML.



To summarize the topic, I will say there can be 3 scenarios if we are working with multiple Sobjects.

  • We have a hybrid list that contains different types of object records which are scattered like (Account1, Contact1, Lead1, Account2 ...). In this type of hybrid list, we can simply sort the list to reduce the chunk size.
  • We have two or three objects but their numbers are huge for example 1200 Accounts, 1200 Contact which will lead to 12 chunks, in this case, we will have to simply divide the records into multiple DMLs in case the number of chunks is above 10 otherwise we can use single DML.
  • If you are working with more than 10 types of objects, you will again end up with the same error as every time object changes chunk also changes. In this case, you will have to separate out DML as well. Depending on the use case you can also use Async transaction to divide DML as doing multiple DML in a single context may not be very good.

Thanks.


If you liked the post do support me by buying a pizza for me :D.





No comments:

Post a Comment