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

com.helger.commons.io.misc.SizeHelper 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.io.misc;

import java.math.BigDecimal;
import java.math.BigInteger;
import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.util.Locale;

import javax.annotation.Nonnegative;
import javax.annotation.Nonnull;
import javax.annotation.concurrent.NotThreadSafe;

import com.helger.commons.CGlobal;
import com.helger.commons.ValueEnforcer;
import com.helger.commons.string.ToStringGenerator;

/**
 * A utility class that converts sizes into the corresponding Kilobyte, Megabyte
 * etc. notation.
* This class is not thread-safe, because the {@link DecimalFormat} class is not * thread-safe! * * @author Philip Helger */ @NotThreadSafe public final class SizeHelper { /** Suffix for Bytes */ public static final String B_SUFFIX = "B"; /** Suffix for Kilobytes */ public static final String KB_SUFFIX = "KB"; /** Suffix for Megabytes */ public static final String MB_SUFFIX = "MB"; /** Suffix for Gigabytes */ public static final String GB_SUFFIX = "GB"; /** Suffix for Terabytes */ public static final String TB_SUFFIX = "TB"; /** Suffix for Petabytes */ public static final String PB_SUFFIX = "PB"; private final DecimalFormatSymbols m_aDFS; private DecimalFormat m_aDF0; private DecimalFormat m_aDF1; private DecimalFormat m_aDF2; public SizeHelper (@Nonnull final Locale aDisplayLocale) { ValueEnforcer.notNull (aDisplayLocale, "DisplayLocale"); m_aDFS = DecimalFormatSymbols.getInstance (aDisplayLocale); } public SizeHelper (@Nonnull final DecimalFormatSymbols aDFS) { m_aDFS = ValueEnforcer.notNull (aDFS, "DecimalFormatSymbols"); } @Nonnull private String _format (final long nSize) { // Lazy init if (m_aDF0 == null) m_aDF0 = new DecimalFormat ("0", m_aDFS); return m_aDF0.format (nSize); } @Nonnull private String _format (final double dSize, @Nonnegative final int nDecimals) { ValueEnforcer.isGE0 (nDecimals, "Decimals"); if (nDecimals == 0) return _format ((long) dSize); // Cache for the most common formats if (nDecimals == 1) { if (m_aDF1 == null) m_aDF1 = new DecimalFormat ("0.0", m_aDFS); return m_aDF1.format (dSize); } if (nDecimals == 2) { if (m_aDF2 == null) m_aDF2 = new DecimalFormat ("0.00", m_aDFS); return m_aDF2.format (dSize); } // build formatting string with at least 3 decimals final StringBuilder aFormat = new StringBuilder ("0.000"); for (int i = 3; i < nDecimals; ++i) aFormat.append ('0'); return new DecimalFormat (aFormat.toString (), m_aDFS).format (dSize); } @Nonnull public String getAsKB (final long nSize) { return _format (nSize / CGlobal.BYTES_PER_KILOBYTE) + KB_SUFFIX; } @Nonnull public String getAsKB (final long nSize, @Nonnegative final int nDecimals) { return _format ((double) nSize / CGlobal.BYTES_PER_KILOBYTE, nDecimals) + KB_SUFFIX; } @Nonnull public String getAsMB (final long nSize) { return _format (nSize / CGlobal.BYTES_PER_MEGABYTE) + MB_SUFFIX; } @Nonnull public String getAsMB (final long nSize, @Nonnegative final int nDecimals) { return _format ((double) nSize / CGlobal.BYTES_PER_MEGABYTE, nDecimals) + MB_SUFFIX; } @Nonnull public String getAsGB (final long nSize) { return _format (nSize / CGlobal.BYTES_PER_GIGABYTE) + GB_SUFFIX; } @Nonnull public String getAsGB (final long nSize, @Nonnegative final int nDecimals) { return _format ((double) nSize / CGlobal.BYTES_PER_GIGABYTE, nDecimals) + GB_SUFFIX; } @Nonnull public String getAsTB (final long nSize) { return _format (nSize / CGlobal.BYTES_PER_TERABYTE) + TB_SUFFIX; } @Nonnull public String getAsTB (final long nSize, @Nonnegative final int nDecimals) { return _format ((double) nSize / CGlobal.BYTES_PER_TERABYTE, nDecimals) + TB_SUFFIX; } @Nonnull public String getAsPB (final long nSize) { return _format (nSize / CGlobal.BYTES_PER_PETABYTE) + PB_SUFFIX; } @Nonnull public String getAsPB (final long nSize, @Nonnegative final int nDecimals) { return _format ((double) nSize / CGlobal.BYTES_PER_PETABYTE, nDecimals) + PB_SUFFIX; } private static void _checkConvertibility (@Nonnull final BigInteger aSize) { if (aSize.compareTo (CGlobal.BIGINT_MAX_LONG) > 0) throw new IllegalArgumentException ("The passed BigInteger is too large to be converted into a long value: " + aSize.toString ()); if (aSize.compareTo (CGlobal.BIGINT_MIN_LONG) < 0) throw new IllegalArgumentException ("The passed BigInteger is too small to be converted into a long value: " + aSize.toString ()); } /** * Get the best matching formatting of the passed value. No fraction digits * will be emitted. * * @param aSize * The value to be converted to a size value. May not be * null. * @return The string representation * @throws IllegalArgumentException * If the passed value cannot be fit in a long */ @Nonnull public String getAsMatching (@Nonnull final BigInteger aSize) { _checkConvertibility (aSize); return getAsMatching (aSize.longValue ()); } /** * Get the best matching formatting of the passed value. * * @param aSize * The value to be converted to a size value. May not be * null. * @param nDecimals * The number of fraction digits. * @return The string representation * @throws IllegalArgumentException * If the passed value cannot be fit in a long */ @Nonnull public String getAsMatching (@Nonnull final BigInteger aSize, @Nonnegative final int nDecimals) { _checkConvertibility (aSize); return getAsMatching (aSize.longValue (), nDecimals); } private static void _checkConvertibility (@Nonnull final BigDecimal aSize) { if (aSize.compareTo (CGlobal.BIGDEC_MAX_LONG) > 0) throw new IllegalArgumentException ("The passed BigDecimal is too large to be converted into a long value: " + aSize.toString ()); if (aSize.compareTo (CGlobal.BIGDEC_MIN_LONG) < 0) throw new IllegalArgumentException ("The passed BigDecimal is too small to be converted into a long value: " + aSize.toString ()); } /** * Get the best matching formatting of the passed value. No fraction digits * will be emitted. * * @param aSize * The value to be converted to a size value. May not be * null. * @return The string representation * @throws IllegalArgumentException * If the passed value cannot be fit in a long */ @Nonnull public String getAsMatching (@Nonnull final BigDecimal aSize) { _checkConvertibility (aSize); return getAsMatching (aSize.longValue ()); } /** * Get the best matching formatting of the passed value. * * @param aSize * The value to be converted to a size value. May not be * null. * @param nDecimals * The number of fraction digits. * @return The string representation * @throws IllegalArgumentException * If the passed value cannot be fit in a long */ @Nonnull public String getAsMatching (@Nonnull final BigDecimal aSize, @Nonnegative final int nDecimals) { _checkConvertibility (aSize); return getAsMatching (aSize.longValue (), nDecimals); } @Nonnull public String getAsMatching (final long nSize) { if (nSize >= CGlobal.BYTES_PER_PETABYTE) return getAsPB (nSize); if (nSize >= CGlobal.BYTES_PER_TERABYTE) return getAsTB (nSize); if (nSize >= CGlobal.BYTES_PER_GIGABYTE) return getAsGB (nSize); if (nSize >= CGlobal.BYTES_PER_MEGABYTE) return getAsMB (nSize); if (nSize >= CGlobal.BYTES_PER_KILOBYTE) return getAsKB (nSize); return _format (nSize) + B_SUFFIX; } @Nonnull public String getAsMatching (final long nSize, @Nonnegative final int nDecimals) { if (nSize >= CGlobal.BYTES_PER_PETABYTE) return getAsPB (nSize, nDecimals); if (nSize >= CGlobal.BYTES_PER_TERABYTE) return getAsTB (nSize, nDecimals); if (nSize >= CGlobal.BYTES_PER_GIGABYTE) return getAsGB (nSize, nDecimals); if (nSize >= CGlobal.BYTES_PER_MEGABYTE) return getAsMB (nSize, nDecimals); if (nSize >= CGlobal.BYTES_PER_KILOBYTE) return getAsKB (nSize, nDecimals); return _format (nSize, nDecimals) + B_SUFFIX; } @Override public String toString () { return new ToStringGenerator (this).append ("dfs", m_aDFS).getToString (); } /** * Get the size helper for the passed locale. The locale determines the * formatting of the numeric value. * * @param aDisplayLocale * The locale to be used. * @return The non-null {@link SizeHelper} object for the passed * locale. */ @Nonnull public static SizeHelper getSizeHelperOfLocale (@Nonnull final Locale aDisplayLocale) { return new SizeHelper (aDisplayLocale); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy