FieldGradientField.java
- /*
- * Licensed to the Hipparchus project under one or more
- * contributor license agreements. See the NOTICE file distributed with
- * this work for additional information regarding copyright ownership.
- * The Hipparchus project licenses this file to You 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
- *
- * https://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 org.hipparchus.analysis.differentiation;
- import java.lang.reflect.Array;
- import java.util.HashMap;
- import java.util.Map;
- import org.hipparchus.CalculusFieldElement;
- import org.hipparchus.Field;
- import org.hipparchus.util.MathArrays;
- /** Field for {@link Gradient} instances.
- * @param <T> the type of the function parameters and value
- * @since 1.7
- */
- public class FieldGradientField<T extends CalculusFieldElement<T>> implements Field<FieldGradient<T>> {
- /** Cached fields. */
- private static final Map<Field<?>, FieldGradientField<?>[]> CACHE = new HashMap<>();
- /** Zero constant. */
- private final FieldGradient<T> zero;
- /** One constant. */
- private final FieldGradient<T> one;
- /** Associated factory for conversions to {@link DerivativeStructure}. */
- private final FDSFactory<T> factory;
- /** Private constructor.
- * @param valueField field for the function parameters and value
- * @param parameters number of free parameters
- */
- private FieldGradientField(final Field<T> valueField, final int parameters) {
- zero = new FieldGradient<>(valueField.getZero(), MathArrays.buildArray(valueField, parameters));
- one = new FieldGradient<>(valueField.getOne(), MathArrays.buildArray(valueField, parameters));
- factory = new FDSFactory<>(valueField, parameters, 1);
- }
- /** Get the field for number of free parameters.
- * @param valueField field for the function parameters and value
- * @param parameters number of free parameters
- * @param <T> the type of the function parameters and value
- * @return cached field
- */
- public static <T extends CalculusFieldElement<T>> FieldGradientField<T> getField(final Field<T> valueField, final int parameters) {
- FieldGradientField<?>[] cachedFields;
- synchronized (CACHE) {
- cachedFields = CACHE.get(valueField);
- if (cachedFields == null || cachedFields.length <= parameters) {
- FieldGradientField<?>[] newCachedFields =
- (FieldGradientField<?>[]) Array.newInstance(FieldGradientField.class, parameters + 1);
- if (cachedFields != null) {
- // preserve the already created fields
- System.arraycopy(cachedFields, 0, newCachedFields, 0, cachedFields.length);
- }
- cachedFields = newCachedFields;
- CACHE.put(valueField, cachedFields);
- }
- }
- if (cachedFields[parameters] == null) {
- // we need to create a new field
- cachedFields[parameters] = new FieldGradientField<>(valueField, parameters);
- }
- @SuppressWarnings("unchecked")
- final FieldGradientField<T> tCached = (FieldGradientField<T>) cachedFields[parameters];
- return tCached;
- }
- /** {@inheritDoc} */
- @Override
- public FieldGradient<T> getOne() {
- return one;
- }
- /** {@inheritDoc} */
- @Override
- public FieldGradient<T> getZero() {
- return zero;
- }
- /** {@inheritDoc} */
- @SuppressWarnings("unchecked")
- @Override
- public Class<FieldGradient<T>> getRuntimeClass() {
- return (Class<FieldGradient<T>>) getZero().getClass();
- }
- /** Get the factory for converting to {@link DerivativeStructure}.
- * <p>
- * This factory is used only for conversions. {@code Gradient} by
- * itself does not rely at all on {@link DSFactory}, {@link DSCompiler}
- * or {@link DerivativeStructure} for its computation. For this reason,
- * the factory here is hidden and this method is package private, so
- * only {@link Gradient#toDerivativeStructure()} can call it on an
- * existing {@link Gradient} instance
- * </p>
- * @return factory for conversions
- */
- FDSFactory<T> getConversionFactory() {
- return factory;
- }
- /** {@inheritDoc} */
- @Override
- public boolean equals(final Object other) {
- return this == other;
- }
- /** {@inheritDoc} */
- @Override
- public int hashCode() {
- return 0xcd3e92ee;
- }
- }