001/* 002 * ============================================================================ 003 * Copyright © 2002-2020 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; 019 020import static org.apiguardian.api.API.Status.STABLE; 021import static org.tquadrat.foundation.lang.Objects.isNull; 022import static org.tquadrat.foundation.lang.internal.StringConverterService.listInstances; 023import static org.tquadrat.foundation.lang.internal.StringConverterService.retrieveConverterForClass; 024import static org.tquadrat.foundation.lang.internal.StringConverterService.retrieveConverterForEnum; 025 026import java.io.Serializable; 027import java.util.Collection; 028import java.util.Optional; 029 030import org.apiguardian.api.API; 031import org.tquadrat.foundation.annotation.ClassVersion; 032 033/** 034 * <p>{@summary Defines the conversion between Strings and object instances, 035 * where the concrete behaviour is defined by the implementation of the 036 * interface.}</p> 037 * <p>Implementations of {@code StringConverter} are meant as a tool to 038 * translate Strings to Objects for technical purposes, not necessarily for 039 * human convenience.</p> 040 * <p>Each implementation should ensure that</p> 041 * <pre><code> StringConverter<T> c = … 042 * T v = … 043 * true == ( v.equals( c.fromString( c.toString( v ) );</code></pre> 044 * <p>is always valid.</p> 045 * <p>If that condition could not be met, this must be clearly 046 * documented.</p> 047 * <p>For the other direction (from String to object back to String), this 048 * condition is weaker, as long as the result is equivalent to the original, 049 * as for numbers that can be represented as decimal, octal or 050 * hexadecimal.</p> 051 * <p>An implementation of {@code StringConverter} usually holds no state and 052 * it is possible to reuse an instance. Therefore it is recommended to provide 053 * a {@code public static final} field {@code INSTANCE} that is initialised 054 * with a reference for a single instance of the implementation. 055 * 056 * @note Both 057 * {@code fromString(CharSequence)} and {@code toString(Object)} will 058 * return {@code null} if called with a {@code null} argument. 059 * 060 * @note Usually the {@code java.util.Locale.ROOT} locale is used when locale 061 * specific conversion or parsing is required; this means in particular 062 * that numbers will use the decimal point as separator. 063 * 064 * @note In order to make an implementation of this interface visible for 065 * {@code forClass()}, {@code forEnum()} and {@code list()}, that 066 * implementation needs to be exposed as a service provider for 067 * {@code org.tquadrat.foundation.lang.StringConverter}. 068 * 069 * @note If the implementation is published as a service provider, and if it 070 * has a static {@code INSTANCE} field holding the reference to an 071 * instance of the converter, the implementation should also provide a 072 * static final method {@code provider()} that returns that reference. 073 * 074 * @note Two implementations of {@code StringConverter} for the same subject 075 * class may cause unpredictable results if both are exposed as service 076 * providers. 077 * 078 * @note When an implementation of {@code StringConverter} is published as a 079 * service, the subject class usually is guessed from the return value of 080 * {@code fromString()}, but sometimes this does not work reliably. In 081 * that case it is required that the implementation of 082 * {@code StringConverter} provides a method 083 * {@code public Collection<Class<?>> getSubjectClass()} that returns the 084 * respective classes.<br> 085 * This is definitely required, when the concrete implementation of the 086 * string converter is derived from an abstract base class; this abstract 087 * class must be visible for this module, too. 088 * 089 * @extauthor Thomas Thrien - thomas.thrien@tquadrat.org 090 * @version $Id: StringConverter.java 1107 2024-03-01 15:33:11Z tquadrat $ 091 * @since 0.1.0 092 * 093 * @param <T> The Object type for the conversion. 094 * 095 * @UMLGraph.link 096 */ 097@SuppressWarnings( "InterfaceMayBeAnnotatedFunctional" ) 098@ClassVersion( sourceVersion = "$Id: StringConverter.java 1107 2024-03-01 15:33:11Z tquadrat $" ) 099@API( status = STABLE, since = "0.1.0" ) 100public interface StringConverter<T> extends Serializable 101{ 102 /*-----------*\ 103 ====** Constants **======================================================== 104 \*-----------*/ 105 /** 106 * The name for the method that returns the subject classes: {@value}. 107 */ 108 @SuppressWarnings( "StaticMethodOnlyUsedInOneClass" ) 109 public static final String METHOD_NAME_GetSubjectClass = "getSubjectClass"; 110 111 /** 112 * The name for the method that returns the instance: {@value}. 113 */ 114 @SuppressWarnings( "StaticMethodOnlyUsedInOneClass" ) 115 public static final String METHOD_NAME_Provider = "provider"; 116 117 /*---------*\ 118 ====** Methods **========================================================== 119 \*---------*/ 120 /** 121 * Returns an instance of {@code StringConverter} for the given 122 * {@link Class}. 123 * If there is no converter for the given type, or the type is 124 * {@code null}, the return value is 125 * {@link Optional#empty()}. 126 * 127 * @param <C> The class a converter is needed for. 128 * @param type The instance of the class a converter is needed for. 129 * @return An instance of 130 * {@link Optional} 131 * that holds the instance of {@code StringConverter}. 132 */ 133 @API( status = STABLE, since = "0.1.0" ) 134 public static <C> Optional<StringConverter<C>> forClass( final Class<C> type ) 135 { 136 return retrieveConverterForClass( type ); 137 } // forClass() 138 139 /** 140 * Returns an instance of {@code StringConverter} for the given 141 * {@link Enum} type. 142 * If there is no converter for the given type in the registry, a new 143 * instance of {@code StringConverter} will be created, based on a 144 * default implementation. 145 * 146 * @param <E> The class a converter is needed for. 147 * @param type The instance of the class a converter is needed for. 148 * @return The requested instance of {@code StringConverter}. 149 */ 150 @API( status = STABLE, since = "0.0.6" ) 151 public static <E extends Enum<E>> StringConverter<E> forEnum( final Class<E> type ) 152 { 153 return retrieveConverterForEnum( type ); 154 } // forEnum() 155 156 /** 157 * Converts the given String to an object instance. 158 * 159 * @param source The String representation for the object instance; 160 * can be {@code null}. 161 * @return The resulting object instance; will be {@code null} if 162 * {@code source} was already {@code null}. 163 * @throws IllegalArgumentException The format of the given String is 164 * invalid and cannot be parsed into the object instance. 165 */ 166 public T fromString( final CharSequence source ) throws IllegalArgumentException; 167 168 /** 169 * Converts the given object instance to a String. 170 * 171 * @note Even if an implementation of 172 * {@link org.tquadrat.foundation.lang.Stringer} 173 * might exist for {@code T}, it cannot be used to implement this 174 * method; 175 * {@link org.tquadrat.foundation.lang.Stringer#toString(Object)} 176 * will never return {@code null} (if implemented accordingly), for a 177 * {@code null} argument, it will return the String "null". 178 * That contradicts the contract for this method. 179 * 180 * @param source The object to convert; can be {@code null}. 181 * @return The resulting String; will be {@code null} if {@code source} 182 * was already {@code null}. 183 */ 184 public default String toString( final T source ) 185 { 186 final var retValue = isNull( source ) ? null : source.toString(); 187 188 //---* Done *---------------------------------------------------------- 189 return retValue; 190 } // toString() 191 192 /** 193 * Returns the classes for that an instance of {@code StringConverter} is 194 * already registered, 195 * 196 * @return The classes with a string converter. 197 */ 198 @API( status = STABLE, since = "0.1.0" ) 199 public static Collection<Class<?>> list() { return listInstances(); } 200} 201// interface StringConverter 202 203/* 204 * End of File 205 */