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 */