Hi/Lo algorithm
Hi/Lo is a algorithm used for generating unique keys for use in a database as a primary key. It uses a sequence-based hi-lo pattern to generate values. Can be used with domain-driven design (DDD).
Algorithm
The current_lo and current_hi variables are internal state variables. The internal state is maintained across invocations. The max_lo constant is an argument passed in at setup as a configuration option. get_next_hi
is a function that retrieves a new high value from a database server.
algorithm generate_key is output: key as a integer if current_lo ≥ max_lo then current_hi := get_next_hi() current_lo := 0 key := current_hi × max_lo + current_lo current_lo := current_lo + 1 return key
Example
Example implementation in Python.
class HiloKeyGenerator(object):
"""Key generator that uses a Hi/Lo algorithm."""
def __init__(self, max_lo: int = 1000):
self.max_lo = max_lo
self.current_hi = 0
self.current_lo = max_lo + 1
def generate_key(self) -> int:
"""Generate a new unique key."""
if self.current_lo >= self.max_lo:
self.current_hi = get_next_hi()
self.current_lo = 0
key = self.current_hi * self.max_lo + self.current_lo
self.current_lo += 1
return key
Output:
>>> generator = HiloKeyGenerator()
>>> generator.generate_key()
1000
>>> generator.generate_key()
1001
>>> generator.generate_key()
1002
Books
Very briefly mentioned in the 2003 book Java Persistence for Relational Databases by Richard Sperko on page 236.[1]
Very briefly mentioned in the 2004 book Better, Faster, Lighter Java by Bruce Tate and Justin Gehtland on page 137.[2]
Very briefly mentioned in the 2004 book Enterprise Java Development on a Budget: Leveraging Java Open Source by Brian Sam-Bodden and Christopher M Jud on page 386.[3]
Explained in the 2015 book Learning NHibernate 4 by Suhas Chatekar on page 53 and 144–145.[4]
Mentioned in the 2017 book NHibernate 4.x cookbook on page 35.[5]
Mentioned in the 2018 book ASP.NET Core 2 Fundamentals on page 219.[6]
This implementation uses hi/lo algorithm to generate identifiers. Algorithm uses a high value retrieved from database and combines it with range of low values to generate a unique identifier. High value is from column
next_id
of tablehibernate_unique_key
buy default. But you can override this to use a different table. This algorithm also supports specifying awhere
parameter which can be used to retrieve high value for different entities from different rows of thehibernate_unique_key
table.— Suhas Chatekar, Learning NHibernate 4 (2015-07-31)
hilo needs a set of two numbers to work with. One is hi which is sourced from a database table and other is lo which is calculated by NHibernate. NHibernate combines these two numbers using a formula to generate a unique number that can be used as identifier.
— Suhas Chatekar, Learning NHibernate 4 (2015-07-31)
While auto incremented IDs are simpler, whenever you add an entity to the context, this addition forces the entity to be inserted to the database. That is because we can only retrieve the ID if the actual insertion happens in the case of auto incremented IDs. The HiLo algorithm frees us from this restriction by reserving the IDs beforehand using a database sequence.
— Onur Gumus and Mugilan T. S. Ragupathi, ASP.NET Core 2 Fundamentals (2018-08-30)
Support
Supported by Entity Framework Core (ORM for .NET Core) with Microsoft SQL Server using the UseHiLo
extension method.[7] Not supported by the predecessor Entity Framework.
Supported by Hibernate (ORM for Java) and NHibernate (ORM for .NET) through SequenceHiLoGenerator
[8] and TableHiLoGenerator
[9]. Had support since at least 2002. Had support since at least version version 3.2 with code authored by Gavin King.
Supported by Doctrine[10] (ORM for PHP) through TableGenerator
.[11]
Supported by Marten[12] (persistence library for .NET) with PostgreSQL through HiLoSequence
.[13]
Supported by RavenDB.[14]
Not supported by Apache Cayenne, ServiceStack.OrmLite, Ruby on Rails Active Record, Dapper, and Dashing.
See also
References
- ^ Sperko, Richard. Java persistence for relational databases. Apress. p. 236. ISBN 9781590590713.
- ^ Tate, Bruce; Gehtland, Justin. Better, faster, lighter Java (1st ed.). O'Reilly. p. 137. ISBN 0-596-00676-4.
- ^ Sam-Bodden, Brian; M Jud, Christopher. Enterprise Java development on a budget : leveraging Java open source technologies. Apress. p. 386. ISBN 978-1-59059-125-3.
- ^ Chatekar, Suhas (2015-07-31). Learning NHibernate 4 : explore the full potential of NHibernate to build robust data access code. Packt Publishing Ltd. p. 53. ISBN 9781784392062.
- ^ Liljas, Gunnar; Zaytsev, Alexander; Dentler, Jason (2017-01-31). NHibernate 4.x cookbook : over 90 incredible and powerful recipes to help you efficiently use NHibernate in your application (Secondition ed.). Packt Publishing Ltd. p. 35. ISBN 9781784394110.
- ^ Gumus, Onur; T. S. Ragupathi, Mugilan (2018-08-30). ASP.NET Core 2 fundamentals : build cross-platform apps and dynamic web services with this server-side web application framework. Packt Publishing Ltd. p. 219. ISBN 9781789533552.
- ^ "SqlServerPropertyBuilderExtensions.UseHiLo Method (Microsoft.EntityFrameworkCore)". docs.microsoft.com.
- ^ "NHibernate Object Relational Mapper". GitHub. NHibernate. 14 November 2019. Retrieved 14 November 2019.
- ^ "NHibernate Object Relational Mapper". GitHub. NHibernate. 14 November 2019. Retrieved 14 November 2019.
- ^ "Doctrine\ORM\Sequencing\TableGenerator | API". www.doctrine-project.org.
- ^ "Doctrine Object Relational Mapper (ORM)". GitHub. Doctrine. 14 November 2019. Retrieved 14 November 2019.
- ^ "Marten - Sequential Identifiers with Hilo". jasperfx.github.io.
- ^ "Postgresql as a Document Database and Event Store for .Net Applications: JasperFx/marten". GitHub. The Jasper Framework and Related Projects. 14 November 2019. Retrieved 14 November 2019.
- ^ "HiLo Algorithm | RavenDB 4.1 Documentation". ravendb.net.