All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.helger.commons.id.factory.AbstractPersistingIntIDFactory Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2014-2024 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *         http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.helger.commons.id.factory;

import javax.annotation.Nonnegative;
import javax.annotation.concurrent.GuardedBy;
import javax.annotation.concurrent.ThreadSafe;

import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.ELockType;
import com.helger.commons.annotation.IsLocked;
import com.helger.commons.concurrent.SimpleLock;
import com.helger.commons.hashcode.HashCodeGenerator;
import com.helger.commons.string.ToStringGenerator;

/**
 * This implementation of {@link IIntIDFactory} reads IDs from a device. It does
 * it by reserving a range of n IDs so that not each ID reservation
 * requires IO. If only 1 ID is effectively used, the other n-1 IDs are
 * lost and will never be assigned to any object again.
 *
 * @author Philip Helger
 */
@ThreadSafe
public abstract class AbstractPersistingIntIDFactory implements IIntIDFactory
{
  protected final SimpleLock m_aLock = new SimpleLock ();
  private final int m_nReserveCount;
  @GuardedBy ("m_aLock")
  private int m_nID = 0;
  @GuardedBy ("m_aLock")
  private int m_nLastID = -1;

  /**
   * Constructor.
   *
   * @param nReserveCount
   *        The number of IDs to reserve per persistence layer access. Must be
   *        > 0.
   */
  protected AbstractPersistingIntIDFactory (@Nonnegative final int nReserveCount)
  {
    ValueEnforcer.isGT0 (nReserveCount, "ReserveCount");
    m_nReserveCount = nReserveCount;
  }

  /**
   * @return The number of IDs to reserve, as provided in the constructor.
   *         Always > 0.
   */
  @Nonnegative
  public final int getReserveCount ()
  {
    // As reserve count is final, we don't need to lock access to it!
    return m_nReserveCount;
  }

  /**
   * Read the current ID from the device. In case the method is called for a
   * non-initialized device, 0 should be returned.
* The update should write the read value plus the passed reserve count back * to the device. This method should perform an atomic read and update to * avoid that ID can be reused.
* Pseudo code: * *
   * protected int readAndUpdateIDCounter (int nReserveCount)
   * {
   *   final int nRead = FileIO.read (file);
   *   FileIO.write (file, nRead + nReserveCount);
   *   return nRead;
   * }
   * 
* * @param nReserveCount * the number that should be added to the read value. Always > 0. * @return 0 if this method is called for a non-initialized device, the value * read from the device otherwise or * {@link com.helger.commons.CGlobal#ILLEGAL_UINT} in case of an * error. */ @IsLocked (ELockType.WRITE) protected abstract int readAndUpdateIDCounter (@Nonnegative int nReserveCount); /* * Note: this implementation must be synchronized because the method calling * this only uses a readLock! */ public final int getNewID () { m_aLock.lock (); try { if (m_nID >= m_nLastID) { // Read new IDs final int nNewID = readAndUpdateIDCounter (m_nReserveCount); // the existing ID may not be < than the previously used ID! if (m_nLastID >= 0 && nNewID < m_nID) throw new IllegalStateException ("The read value " + nNewID + " is smaller than the last known ID " + m_nID + "!"); m_nID = nNewID; m_nLastID = nNewID + m_nReserveCount; } final int ret = m_nID; m_nID++; return ret; } finally { m_aLock.unlock (); } } @Override public boolean equals (final Object o) { if (o == this) return true; if (o == null || !getClass ().equals (o.getClass ())) return false; final AbstractPersistingIntIDFactory rhs = (AbstractPersistingIntIDFactory) o; return m_nReserveCount == rhs.m_nReserveCount && m_nID == rhs.m_nID; } @Override public int hashCode () { return new HashCodeGenerator (this).append (m_nReserveCount).append (m_nID).getHashCode (); } @Override public String toString () { return new ToStringGenerator (this).append ("ReserveCount", m_nReserveCount) .append ("ID", m_nID) .append ("LastID", m_nLastID) .getToString (); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy