[erlang-questions] Better way to check if a set of keys exists in a mnesia table?

Mikael Pettersson mikpelinux@REDACTED
Sat Jul 16 16:26:44 CEST 2016


Chaitanya Chalasani writes:
 > I have a table with an UUID as the primary key / first element of the record. 
 > 
 > What is the efficient way to check if a given set of UUIDs are valid primary key for that table. 
 > 
 > I can think for three different solutions -
 > use mnesia:all_leys(TableName) and perform lists subset check. However, if the table contains over a million records, fetching all the keys for every check isn’t a nice solution.
 > use mnesia:read(TableName, Key) and check on the response. However, if the row is a big enough, trying to get the whole row for a simple key check isn’t that good either. 
 > use ets:member(TableName, Key). A better solution than the above but doesn’t work on remote tables. 
 > 
 > Which one of the above is the least bad solution or is there a better one hidden under the documents. 

all_keys can be horribly expensive and should be avoided if possible, but for small tables it may be acceptable.

I'd do one of the following:

1. mnesia:dirty_read(T, K) and check result for [] vs [_|_]
   Pro: easy, works
   Con: the data copy may be expensive for large records

2. Make the table an ordered_set; mnesia:dirty_prev(T, mnesia:dirty_next(T, K)) and check if K is returned
   Pro: avoids the data copy
   Con: requires an ordered_set, requires code to handle boundary conditions wrt '$end_of_table'

3. Store the keys w/o data in a separate table, then do a dirty_read in that
   Pro: reduces copying 
   Con: requires more storage, the lookup in the side table won't provide cache hints to help your access
        in the main table (but that may be Ok if the side table is hit orders of magnitude more often)

One could implement some sort of sparse bitmap or range tree and use that to record key presence, but I'm
not sure it would be worthwhile in Erlang.



More information about the erlang-questions mailing list