001/*
002 * ============================================================================
003 * Copyright © 2002-2023 by Thomas Thrien.
004 * All Rights Reserved.
005 * ============================================================================
006 * Licensed to the public under the agreements of the GNU Lesser General Public
007 * License, version 3.0 (the "License"). You may obtain a copy of the License at
008 *
009 *      http://www.gnu.org/licenses/lgpl.html
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
013 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
014 * License for the specific language governing permissions and limitations
015 * under the License.
016 */
017
018package org.tquadrat.foundation.lang.internal;
019
020import static org.apiguardian.api.API.Status.INTERNAL;
021import static org.tquadrat.foundation.lang.CommonConstants.NULL_STRING;
022import static org.tquadrat.foundation.lang.Objects.isNull;
023import static org.tquadrat.foundation.lang.Objects.nonNull;
024import static org.tquadrat.foundation.lang.Objects.requireNonNullArgument;
025
026import java.util.Optional;
027import java.util.function.Supplier;
028
029import org.apiguardian.api.API;
030import org.tquadrat.foundation.annotation.ClassVersion;
031import org.tquadrat.foundation.lang.AutoLock;
032import org.tquadrat.foundation.lang.Lazy;
033import org.tquadrat.foundation.lang.Objects;
034
035/**
036 *  <p>{@summary The implementation of the interface
037 *  {@link Lazy}.}</p>
038 *  <p>An instance of this class holds a value that will be initialised by a
039 *  call to the supplier (provided with the constructor
040 *  {@link #LazyImpl(Supplier)})
041 *  on a first call to
042 *  {@link #get()}.</p>
043 *  <p>For special purposes, the constructor
044 *  {@link #LazyImpl(Object)}
045 *  creates an already initialised instance of {@code Lazy}.</p>
046 *  <p>Use
047 *  {@link #isPresent()}
048 *  to avoid unnecessary initialisation.</p>
049 *  <p>As a lazy initialisation makes the value unpredictable, it is necessary
050 *  that the implementations of
051 *  {@link #equals(Object)}
052 *  and
053 *  {@link #hashCode()}
054 *  force the initialisation.</p>
055 *  <p>{@link #toString()} do not force the initialisation.</p>
056 *
057 *  @extauthor Thomas Thrien - thomas.thrien@tquadrat.org
058 *  @version $Id: LazyImpl.java 1060 2023-09-24 19:21:40Z tquadrat $
059 *  @since 0.0.5
060 *
061 *  @UMLGraph.link
062 *
063 *  @param  <T> The type of the value for this instance of {@code Lazy}.
064 */
065@ClassVersion( sourceVersion = "$Id: LazyImpl.java 1060 2023-09-24 19:21:40Z tquadrat $" )
066@API( status = INTERNAL, since = "0.0.5" )
067public final class LazyImpl<T> implements Lazy<T>
068{
069        /*------------*\
070    ====** Attributes **=======================================================
071        \*------------*/
072    /**
073     *  The lock that protects the value creation. It will be set to
074     *  {@linkplain Optional#empty() empty}
075     *  after
076     *  {@link #m_Value} is initialised.
077     */
078    @SuppressWarnings( "OptionalUsedAsFieldOrParameterType" )
079    private Optional<AutoLock> m_Lock;
080
081    /**
082     *  The supplier for the value of this {@code Lazy} instance. It will be
083     *  set to {@code null} after
084     *  {@link #m_Value}
085     *  is initialised.
086     */
087    private Supplier<T> m_Supplier;
088
089    /**
090     *  The value of this {@code Lazy} instance; it is {@code null} if it was
091     *  not yet initialised.
092     */
093    private T m_Value;
094
095        /*--------------*\
096    ====** Constructors **=====================================================
097        \*--------------*/
098    /**
099     *  Creates a new {@code Lazy} instance.
100     *
101     *  @param  supplier    The supplier that initialises the value for this
102     *      instance on the first call to
103     *      {@link #get()}.
104     */
105    public LazyImpl( final Supplier<T> supplier )
106    {
107        m_Supplier = requireNonNullArgument( supplier, "supplier" );
108        m_Lock = Optional.of( AutoLock.of() );
109        m_Value = null;
110    }   //  LazyImpl()
111
112    /**
113     *  <p>{@summary Creates a new {@code Lazy} instance that is already
114     *  initialised.}</p>
115     *  <p>This allows to use {@code Lazy} instances also for
116     *  {@linkplain java.lang.Cloneable cloneable}
117     *  objects, given that {@code T} is either cloneable itself or
118     *  immutable.</p>
119     *
120     *  @param  value   The value; can be {@code null}.
121     *
122     *  @see Object#clone()
123     */
124    public LazyImpl( final T value )
125    {
126        m_Supplier = null;
127        m_Lock = Optional.empty();
128        m_Value = value;
129    }   //  LazyImpl()
130
131        /*---------*\
132    ====** Methods **==========================================================
133        \*---------*/
134    /**
135     *  {@inheritDoc}
136     */
137    @Override
138    public final boolean equals( final Object obj )
139    {
140        var retValue = this == obj;
141        if( !retValue && nonNull( obj ) )
142        {
143            if( obj instanceof final Lazy<?> other )
144            {
145                retValue = get().equals( other.get() );
146            }
147            else
148            {
149                retValue = get().equals( obj );
150            }
151        }
152
153        //---* Done *----------------------------------------------------------
154        return retValue;
155    }   //  equals()
156
157    /**
158     *  {@inheritDoc}
159     */
160    @Override
161    public final T get()
162    {
163        m_Lock.ifPresent( lck ->
164        {
165            try( @SuppressWarnings( "unused" ) final var lock = lck.lock() )
166            {
167                /*
168                 * When the lock is present, the supplier must have been not
169                 * null; this is enforced by the respective constructor.
170                 * So when the supplier is NOW null, another thread has
171                 * performed the initialisation while this one was waiting for
172                 * the lock.
173                 */
174                if( nonNull( m_Supplier ) )
175                {
176                    m_Value = m_Supplier.get();
177                    m_Lock = Optional.empty();
178                    m_Supplier = null;
179                }
180            }
181        } );
182
183        //---* Done *----------------------------------------------------------
184        return m_Value;
185    }   //  get()
186
187    /**
188     *  {@inheritDoc}
189     */
190    @Override
191    public final int hashCode() { return get().hashCode(); }
192
193    /**
194     *  {@inheritDoc}
195     */
196    @Override
197    public final boolean isPresent() { return isNull( m_Supplier ); }
198
199    /**
200     *  {@inheritDoc}
201     */
202    @Override
203    public <X extends Throwable> T orElseThrow( final Supplier<? extends X> exceptionSupplier ) throws X
204    {
205        if( !isPresent() ) throw exceptionSupplier.get();
206
207        //---* Done *----------------------------------------------------------
208        return m_Value;
209    }   //  orElseThrow()
210
211    /**
212     *  {@inheritDoc}
213     */
214    @Override
215    public final String toString()
216    {
217        final var retValue = Objects.toString( m_Value, isPresent() ? NULL_STRING : "[Not initialized]" );
218
219        //---* Done *----------------------------------------------------------
220        return retValue;
221    }   //  toString()
222}
223//  class LazyImpl
224
225/*
226 *  End of File
227 */