All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.javamoney.moneta.internal.DefaultCashRounding Maven / Gradle / Ivy

Go to download

JSR 354 provides an API for representing, transporting, and performing comprehensive calculations with Money and Currency. This module implements JSR 354.

There is a newer version: 1.4.1
Show newest version
/**
 * Copyright (c) 2012, 2014, Credit Suisse (Anatole Tresch), Werner Keil and others by the @author tag.
 *
 * 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 org.javamoney.moneta.internal;

import javax.money.CurrencyUnit;
import javax.money.MonetaryAmount;
import javax.money.MonetaryRounding;
import javax.money.RoundingContext;
import javax.money.RoundingContextBuilder;

import java.io.Serializable;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.util.Objects;

/**
 * Implementation class providing cash rounding {@link javax.money.MonetaryOperator}
 * instances for {@link CurrencyUnit} instances. modeling rounding based on
 * minimal minor units available for cash payments.
 * 

* This class is thread safe. * * @author Anatole Tresch */ final class DefaultCashRounding implements MonetaryRounding, Serializable { /** * The scale key to be used. */ private static final String SCALE_KEY = "scale"; /** * The scale key to be used. */ private static final String MINMINORS_KEY = "minimalMinors"; /** * The provider class key to be used. */ private static final String PROVCLASS_KEY = "providerClass"; /** * The cash rounding flag key to be used. */ private static final String CASHROUNDING_KEY = "cashRounding"; private RoundingContext context; /** * Creates an rounding instance. * * @param roundingMode The {@link RoundingMode} to be used, not {@code null}. * @param scale The target scale. */ DefaultCashRounding(int scale, RoundingMode roundingMode, int minimalMinors) { if (scale < 0) { throw new IllegalArgumentException("scale < 0"); } RoundingContextBuilder b = RoundingContextBuilder.of("default", "default").set(CASHROUNDING_KEY, true). set(PROVCLASS_KEY, getClass().getName()).set(MINMINORS_KEY, minimalMinors).set(SCALE_KEY, scale); if(roundingMode==null){ throw new IllegalArgumentException("roundingMode missing"); } this.context = b.set(roundingMode).build(); } /** * Creates an {@link DefaultCashRounding} for rounding * {@link MonetaryAmount} instances given a currency. * * @param currency The currency, which determines the required precision. As * {@link RoundingMode}, by default, {@link RoundingMode#HALF_UP} * is used. */ DefaultCashRounding(CurrencyUnit currency, RoundingMode roundingMode, int minimalMinors) { this(currency.getDefaultFractionDigits(), roundingMode, minimalMinors); } /** * Creates an {@link javax.money.MonetaryOperator} for rounding {@link MonetaryAmount} * instances given a currency. * * @param currency The currency, which determines the required precision. As * {@link RoundingMode}, by default, {@link RoundingMode#HALF_UP} * is used. */ DefaultCashRounding(CurrencyUnit currency, int minimalMinors) { this(currency, RoundingMode.HALF_UP, minimalMinors); } /* * (non-Javadoc) * * @see org.javamoney.bp.api.MonetaryFunction#apply(java.lang.Object) */ @Override public MonetaryAmount apply(MonetaryAmount value) { Objects.requireNonNull(value, "Amount required."); // 1 extract BD value, round according the default fraction units int scale = this.context.getInt(SCALE_KEY); RoundingMode roundingMode = this.context.get(RoundingMode.class); BigDecimal num = value.getNumber().numberValue(BigDecimal.class).setScale(scale, roundingMode); // 2 evaluate minor units and remainder long minors = num.movePointRight(num.scale()).longValueExact(); int minimalMinors = this.context.getInt(MINMINORS_KEY); long factor = minors / minimalMinors; long low = minimalMinors * factor; long high = minimalMinors * (factor + 1); if (minors - low > high - minors) { minors = high; } else if (minors - low < high - minors) { minors = low; } else { switch (roundingMode) { case HALF_UP: case UP: case HALF_EVEN: minors = high; break; default: minors = low; } } return value.getFactory().setCurrency(value.getCurrency()) .setNumber(BigDecimal.valueOf(minors).movePointLeft(scale)).create(); } @Override public RoundingContext getRoundingContext() { return context; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy