com.helger.commons.lang.CloneHelper Maven / Gradle / Ivy
/*
* 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.lang;
import java.lang.reflect.Constructor;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import javax.annotation.concurrent.Immutable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.helger.commons.annotation.PresentForCodeCoverage;
import com.helger.commons.annotation.ReturnsMutableCopy;
import com.helger.commons.collection.impl.CommonsArrayList;
import com.helger.commons.collection.impl.ICommonsList;
/**
* Helper class for cloning objects.
*
* @author Philip Helger
*/
@Immutable
public final class CloneHelper
{
private static final Logger LOGGER = LoggerFactory.getLogger (CloneHelper.class);
@PresentForCodeCoverage
private static final CloneHelper INSTANCE = new CloneHelper ();
private CloneHelper ()
{}
@Nullable
private static DATATYPE _getGenericClone (@Nonnull final DATATYPE aObject)
{
// 1. check ICloneable interface
if (aObject instanceof ICloneable >)
return GenericReflection.> uncheckedCast (aObject).getClone ();
try
{
// 2. find Object.clone method
return GenericReflection.invokeMethod (aObject, "clone");
}
catch (final Exception ex)
{
LOGGER.warn ("Failed to invoke clone on " + aObject.getClass ().getName ());
try
{
// 3. find copy-constructor
final Constructor aCtor = GenericReflection.findConstructor (aObject, aObject.getClass ());
if (aCtor != null)
return aCtor.newInstance (aObject);
}
catch (final IllegalAccessException ex2)
{
LOGGER.error ("Failed to clone object of type '" +
aObject.getClass ().getName () +
"' because it has neither a (visible) clone method nor a copy constructor or the methods are invisible.");
}
catch (final NoSuchMethodException ex2)
{
LOGGER.error ("Failed to clone object of type '" +
aObject.getClass ().getName () +
"' because it has neither a clone method nor a (visible) copy constructor or the methods are invisible.");
}
catch (final Exception ex2)
{
LOGGER.error ("Failed to clone object of type '" +
aObject.getClass ().getName () +
"' because it has neither a (visible) clone method nor a copy constructor.",
ex2);
}
}
return null;
}
/**
* Get a clone (= deep copy) of the passed value. The following things are
* tried for cloning:
*
* - If the object is immutable, it is returned as is (if it is a primitive
* type or marked with the {@link Immutable} annotation.
* - If the object implements {@link ICloneable} it is invoked.
* - If the object implements {@link Cloneable} it is invoked.
* - If a copy constructor (a constructor taking one argument of the same
* class as it declares)
*
* If all tries fail, null
is returned.
*
* @param
* The source and return type
* @param aObject
* The object to be copied.
* @return null
if the passed value is null
or if no
* cloning could be performed.
*/
@Nullable
public static DATATYPE getClonedValue (@Nullable final DATATYPE aObject)
{
// null -> null
if (aObject == null)
return null;
final Class > aClass = aObject.getClass ();
// special handling for immutable objects without equals or clone
if (ClassHelper.isPrimitiveWrapperType (aClass) ||
aObject instanceof String ||
aClass.getAnnotation (Immutable.class) != null)
return aObject;
// generic clone
return _getGenericClone (aObject);
}
/**
* Get a clone (= deep copy) of the passed value for all objects implementing
* {@link ICloneable}.
*
* @param aObject
* The object to be copied. May be null
.
* @return null
if the passed value is null
or a
* clone of the object.
* @param
* The data type to be cloned
*/
@Nullable
public static > DATATYPE getCloneIfNotNull (@Nullable final DATATYPE aObject)
{
// null -> null
if (aObject == null)
return null;
return aObject.getClone ();
}
/**
* Get a list where each contained item is also cloned. Like a deep copy.
*
* @param aList
* Source list. May be null
.
* @return The cloned list. Never null
but maybe empty if the
* source list is empty.
* @param
* The list element type to be cloned
*/
@Nonnull
@ReturnsMutableCopy
public static ICommonsList getGenericClonedList (@Nullable final Iterable aList)
{
final ICommonsList ret = new CommonsArrayList <> ();
if (aList != null)
for (final DATATYPE aItem : aList)
ret.add (getClonedValue (aItem));
return ret;
}
/**
* Get a list where each contained item is also cloned. Like a deep copy.
*
* @param aList
* Source list. May be null
.
* @return The cloned list. Never null
but maybe empty if the
* source list is empty.
* @param
* The set element type to be cloned
*/
@Nonnull
@ReturnsMutableCopy
public static > ICommonsList getClonedList (@Nullable final Iterable aList)
{
final ICommonsList ret = new CommonsArrayList <> ();
if (aList != null)
for (final DATATYPE aItem : aList)
ret.add (getCloneIfNotNull (aItem));
return ret;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy