Javaで1200から1.2kにフォーマットする方法


157

以下の数値をJavaでそれらの隣の数値にフォーマットしたいと思います。

1000 to 1k
5821 to 5.8k
10500 to 10k
101800 to 101k
2000000 to 2m
7800000 to 7.8m
92150000 to 92m
123200000 to 123m

右側の数字は長いか整数で、左側の数字は文字列になります。私はこれにどのように取り組むべきですか?私はすでにこれのためにほとんどアルゴリズムをしていませんでしたが、それでより良い仕事をする何かがすでに発明されているかもしれないと思いました、そして私が数十億と数兆を扱い始めれば追加のテストを必要としません:)

追加要件:

  • 形式は最大4文字にする必要があります
  • 上記の意味は、1.1kは大丈夫です11.2kは大丈夫ではありません。同じ7.8mは問題ありません19.1mは問題ありません。小数点を持つことができるのは、小数点の前の1桁のみです。小数点の前の2桁は、小数点の後の数字ではないことを意味します。
  • 丸めは必要ありません。(kとmが追加されて表示される数値は、正確なロジックの記事ではなく、近似を示すアナログゲージの数値です。したがって、主に変数の性質により、キャッシュされた結果を見ているときにも数桁が増減する可能性があるため、丸めは無関係です。)

1
誰もライブラリを持っていない場合は、コードを投稿してもよろしいですか?
Grammin、2011年

1
これは役に立ちますが、これは重複ではありません。stackoverflow.com/questions/529432
rfeak '20年

1
@マット私はあなたが以前どのようなソリューションを使用していたのか知​​りたいと思っていました。よろしければ、回答として投稿してください。
jzd

1
これの背後にある考えNo rounding is necessaryは私にはばかげているようです。物事を複雑にするだけですか?これを次のように言い換えた方がいいのではないでしょうRounding is not necessary, but welcomeか。
Wolf

1
kとmが追加されて表示されている数値に気付かなかった場合は、正確なロジックの記事ではなく、近似を示すアナログゲージが多くなっています。したがって、丸めは、主に変数の性質が原因で関係がないため、キャッシュされた結果を見ている間でも、数桁を増減できます。
Mat B.

回答:


155

これは、任意の長い値で機能するソリューションであり、私は非常に読みやすいと思います(コアロジックは、formatメソッドのます)。

TreeMap適切なサフィックスを見つけるために活用します。これは、配列を使用していて読みづらかった以前のソリューションよりも驚くほど効率的です。

private static final NavigableMap<Long, String> suffixes = new TreeMap<> ();
static {
  suffixes.put(1_000L, "k");
  suffixes.put(1_000_000L, "M");
  suffixes.put(1_000_000_000L, "G");
  suffixes.put(1_000_000_000_000L, "T");
  suffixes.put(1_000_000_000_000_000L, "P");
  suffixes.put(1_000_000_000_000_000_000L, "E");
}

public static String format(long value) {
  //Long.MIN_VALUE == -Long.MIN_VALUE so we need an adjustment here
  if (value == Long.MIN_VALUE) return format(Long.MIN_VALUE + 1);
  if (value < 0) return "-" + format(-value);
  if (value < 1000) return Long.toString(value); //deal with easy case

  Entry<Long, String> e = suffixes.floorEntry(value);
  Long divideBy = e.getKey();
  String suffix = e.getValue();

  long truncated = value / (divideBy / 10); //the number part of the output times 10
  boolean hasDecimal = truncated < 100 && (truncated / 10d) != (truncated / 10);
  return hasDecimal ? (truncated / 10d) + suffix : (truncated / 10) + suffix;
}

テストコード

public static void main(String args[]) {
  long[] numbers = {0, 5, 999, 1_000, -5_821, 10_500, -101_800, 2_000_000, -7_800_000, 92_150_000, 123_200_000, 9_999_999, 999_999_999_999_999_999L, 1_230_000_000_000_000L, Long.MIN_VALUE, Long.MAX_VALUE};
  String[] expected = {"0", "5", "999", "1k", "-5.8k", "10k", "-101k", "2M", "-7.8M", "92M", "123M", "9.9M", "999P", "1.2P", "-9.2E", "9.2E"};
  for (int i = 0; i < numbers.length; i++) {
    long n = numbers[i];
    String formatted = format(n);
    System.out.println(n + " => " + formatted);
    if (!formatted.equals(expected[i])) throw new AssertionError("Expected: " + expected[i] + " but found: " + formatted);
  }
}

1
素晴らしい解決策。これらの本当に大きな数(4兆、5億など)のサフィックスを追加するだけでよく、出力はスケーリングし続けます。
Cypher

あなたのコードは負の数では正しくありません:-5821として-5kではなくとして フォーマットされるべき-5.8kです
std.denis 2015年

1
@ std.denis OPは負の数値のフォーマット方法を示していませんでした。私はそれらを正の数のようにフォーマットすることを決定しましたが-、有効桁数を同じに保つために接頭辞を付けました。他のオプションがあります...
assylias

1
まず、明らかにあなたのせいではないので、私は悪いコメントを削除しました。第二に、良い答えが他のものよりも多く得られる限り、十分な注意を引きつけられないことは問題ではありませんが、良い答えを見つけるために掘る必要があり、いくつかの間違った、悪いまたは一般的な答えが賛成されます(本当に新しいことを学ぶのは悪いことです)。そして、すでに多くの回答が存在する場合に賞金を発行する人々にとって、不足しているものをより明確に特定し、基準に最も適合する回答を慎重に選択することを期待していました...
maraca

1
しかし、全世界はこの基準を理解していますか?世界中の人のためのアプリを作る場合は注意してください。英語、それのために10Mが、ロシアにとってはだ10млнなど
user924

101

これはCプログラムのように見えますが、超軽量です。

public static void main(String args[]) {
    long[] numbers = new long[]{1000, 5821, 10500, 101800, 2000000, 7800000, 92150000, 123200000, 9999999};
    for(long n : numbers) {
        System.out.println(n + " => " + coolFormat(n, 0));
    }
}

private static char[] c = new char[]{'k', 'm', 'b', 't'};

/**
 * Recursive implementation, invokes itself for each factor of a thousand, increasing the class on each invokation.
 * @param n the number to format
 * @param iteration in fact this is the class from the array c
 * @return a String representing the number n formatted in a cool looking way.
 */
private static String coolFormat(double n, int iteration) {
    double d = ((long) n / 100) / 10.0;
    boolean isRound = (d * 10) %10 == 0;//true if the decimal part is equal to 0 (then it's trimmed anyway)
    return (d < 1000? //this determines the class, i.e. 'k', 'm' etc
        ((d > 99.9 || isRound || (!isRound && d > 9.99)? //this decides whether to trim the decimals
         (int) d * 10 / 10 : d + "" // (int) d * 10 / 10 drops the decimal
         ) + "" + c[iteration]) 
        : coolFormat(d, iteration+1));

}

それは出力します:

1000 => 1k
5821 => 5.8k
10500 => 10k
101800 => 101k
2000000 => 2m
7800000 => 7.8m
92150000 => 92m
123200000 => 123m
9999999 => 9.9m

16
難読化されたコード。最近は、このようにコーディングする必要はありません。期待どおりに動作する可能性がありますが、著者にはRoger C. Martin:Clean Code
Andreas Dolk

30
難読化されていますか?申し訳ありませんが、おそらく1冊の本を読んで、最近ではなんとかして別の方法でコーディングできると思います。それについてJoel(joelonsoftware.com/articles/ThePerilsofJavaSchools.html)に伝えてください。私は、メソッドの速度に近い場所に到達するために記述できるコードをあえて用意します。
Elijah Saounkine、2011年

11
d、c、n変数をより読みやすい(より迅速な理解)ものに変更すると、このコードは私の見解では適切になります
Gennadiy Ryabkin

5
なぜこのパフォーマンスへのこだわり?なぜ誰もがパフォーマンスについて考えることさえ保証するのに十分な数のこれらの変換を実行したいのでしょうか...?読みやすさを第一に、パフォーマンスは必要な場合にのみ調整します。
Amos M. Carpenter 2015年

10
@ AmosM.Carpenterに同意する必要があります。4年前にこの回答を書いたとき、コードの保守性についてほとんど知りませんでした。最適化することは悪くありません。一般的に言えば、可読性が最初です。ちなみに、パフォーマンスに関してはそれほど悪くはありません。マラカが書いたものより5倍遅くはありません-ほぼ同じです(ベンチマークのソリューションのいくつかをここに示しますgithub.com/esaounkine/number-format-ベンチマーク)。
Elijah Saounkine

43

ここで、DecimalFormatの工学表記を利用するソリューション:

public static void main(String args[]) {
    long[] numbers = new long[]{7, 12, 856, 1000, 5821, 10500, 101800, 2000000, 7800000, 92150000, 123200000, 9999999};
    for(long number : numbers) {
        System.out.println(number + " = " + format(number));
    }
}

private static String[] suffix = new String[]{"","k", "m", "b", "t"};
private static int MAX_LENGTH = 4;

private static String format(double number) {
    String r = new DecimalFormat("##0E0").format(number);
    r = r.replaceAll("E[0-9]", suffix[Character.getNumericValue(r.charAt(r.length() - 1)) / 3]);
    while(r.length() > MAX_LENGTH || r.matches("[0-9]+\\.[a-z]")){
        r = r.substring(0, r.length()-2) + r.substring(r.length() - 1);
    }
    return r;
}

出力:

7 = 7
12 = 12
856 = 856
1000 = 1k
5821 = 5.8k
10500 = 10k
101800 = 102k
2000000 = 2m
7800000 = 7.8m
92150000 = 92m
123200000 = 123m
9999999 = 10m

@Mat新しい要件を処理するために更新
jzd '21年

これをCurrency Instanceと組み合わせて、同様の機能をCurrencyで取得する簡単な方法はありますか?
xdumaine

@roviuser、どういう意味かわかりませんが、これは別の質問のように聞こえます。
jzd

7
160000から200kを切り捨て、120000を100kに
切り捨てます

4
これが壊れている、私は数10000000000000.0を入力し、それが103語る
オリバー・ディクソン

23

多少の改善が必要ですが、StrictMathが助けになります。
あなたは文字列または配列にサフィックスを入れて、パワーまたはそのようなものに基づいてフェッチすることができます。
部門はパワーを中心に管理することもできます。ほとんどすべてがパワーの値に関するものだと思います。それが役に立てば幸い!

public static String formatValue(double value) {
int power; 
    String suffix = " kmbt";
    String formattedNumber = "";

    NumberFormat formatter = new DecimalFormat("#,###.#");
    power = (int)StrictMath.log10(value);
    value = value/(Math.pow(10,(power/3)*3));
    formattedNumber=formatter.format(value);
    formattedNumber = formattedNumber + suffix.charAt(power/3);
    return formattedNumber.length()>4 ?  formattedNumber.replaceAll("\\.[0-9]+", "") : formattedNumber;  
}

出力:

999
1.2k
98k
911k
1.1m
11b
712b
34t


2
読みやすさを少し改善しました。4文字の問題を解決するためにjzdからreturnステートメントを追加する必要があるだけです。また、AIOOB例外を回避するためにtを超える場合は、必ず接尾辞を追加してください。;)
jhurtado

このコードはロケールに依存します。たとえば、sv_SEの場合、ロケール1000は10x10に変換されますが、正規表現では正しく一致しません。
Joakim Lundborg、2014

2
0の例外をスローし、負の数では機能せず、9,999,999を適切に丸めません(10mを出力)...
assylias

16

現在の回答の問題

  • 現在のソリューションの多くは、これらの接頭辞k = 10 3、m = 10 6、b = 10 9、t = 10 12を使用しています。ただし、さまざまな ソースによると、正しい接頭辞はk = 10 3、M = 10 6、G = 10 9、T = 10 12です。
  • 負の数のサポートの欠如(または少なくとも負の数がサポートされていることを示すテストの欠如)
  • 逆演算のサポートの欠如、たとえば1.1kから1100への変換(これは元の質問の範囲外ですが)

Javaソリューション

このソリューション(この回答の拡張)は、上記の問題に対処します。

import org.apache.commons.lang.math.NumberUtils;

import java.text.DecimalFormat;
import java.text.FieldPosition;
import java.text.Format;
import java.text.ParsePosition;
import java.util.regex.Pattern;


/**
 * Converts a number to a string in <a href="http://en.wikipedia.org/wiki/Metric_prefix">metric prefix</a> format.
 * For example, 7800000 will be formatted as '7.8M'. Numbers under 1000 will be unchanged. Refer to the tests for further examples.
 */
class RoundedMetricPrefixFormat extends Format {

    private static final String[] METRIC_PREFIXES = new String[]{"", "k", "M", "G", "T"};

    /**
     * The maximum number of characters in the output, excluding the negative sign
     */
    private static final Integer MAX_LENGTH = 4;

    private static final Pattern TRAILING_DECIMAL_POINT = Pattern.compile("[0-9]+\\.[kMGT]");

    private static final Pattern METRIC_PREFIXED_NUMBER = Pattern.compile("\\-?[0-9]+(\\.[0-9])?[kMGT]");

    @Override
    public StringBuffer format(Object obj, StringBuffer output, FieldPosition pos) {

        Double number = Double.valueOf(obj.toString());

        // if the number is negative, convert it to a positive number and add the minus sign to the output at the end
        boolean isNegative = number < 0;
        number = Math.abs(number);

        String result = new DecimalFormat("##0E0").format(number);

        Integer index = Character.getNumericValue(result.charAt(result.length() - 1)) / 3;
        result = result.replaceAll("E[0-9]", METRIC_PREFIXES[index]);

        while (result.length() > MAX_LENGTH || TRAILING_DECIMAL_POINT.matcher(result).matches()) {
            int length = result.length();
            result = result.substring(0, length - 2) + result.substring(length - 1);
        }

        return output.append(isNegative ? "-" + result : result);
    }

    /**
     * Convert a String produced by <tt>format()</tt> back to a number. This will generally not restore
     * the original number because <tt>format()</tt> is a lossy operation, e.g.
     *
     * <pre>
     * {@code
     * def formatter = new RoundedMetricPrefixFormat()
     * Long number = 5821L
     * String formattedNumber = formatter.format(number)
     * assert formattedNumber == '5.8k'
     *
     * Long parsedNumber = formatter.parseObject(formattedNumber)
     * assert parsedNumber == 5800
     * assert parsedNumber != number
     * }
     * </pre>
     *
     * @param source a number that may have a metric prefix
     * @param pos if parsing succeeds, this should be updated to the index after the last parsed character
     * @return a Number if the the string is a number without a metric prefix, or a Long if it has a metric prefix
     */
    @Override
    public Object parseObject(String source, ParsePosition pos) {

        if (NumberUtils.isNumber(source)) {

            // if the value is a number (without a prefix) don't return it as a Long or we'll lose any decimals
            pos.setIndex(source.length());
            return toNumber(source);

        } else if (METRIC_PREFIXED_NUMBER.matcher(source).matches()) {

            boolean isNegative = source.charAt(0) == '-';
            int length = source.length();

            String number = isNegative ? source.substring(1, length - 1) : source.substring(0, length - 1);
            String metricPrefix = Character.toString(source.charAt(length - 1));

            Number absoluteNumber = toNumber(number);

            int index = 0;

            for (; index < METRIC_PREFIXES.length; index++) {
                if (METRIC_PREFIXES[index].equals(metricPrefix)) {
                    break;
                }
            }

            Integer exponent = 3 * index;
            Double factor = Math.pow(10, exponent);
            factor *= isNegative ? -1 : 1;

            pos.setIndex(source.length());
            Float result = absoluteNumber.floatValue() * factor.longValue();
            return result.longValue();
        }

        return null;
    }

    private static Number toNumber(String number) {
        return NumberUtils.createNumber(number);
    }
}

Groovyソリューション

以下に示すように、このソリューションは当初Groovyで作成されました。

import org.apache.commons.lang.math.NumberUtils

import java.text.DecimalFormat
import java.text.FieldPosition
import java.text.Format
import java.text.ParsePosition
import java.util.regex.Pattern


/**
 * Converts a number to a string in <a href="http://en.wikipedia.org/wiki/Metric_prefix">metric prefix</a> format.
 * For example, 7800000 will be formatted as '7.8M'. Numbers under 1000 will be unchanged. Refer to the tests for further examples.
 */
class RoundedMetricPrefixFormat extends Format {

    private static final METRIC_PREFIXES = ["", "k", "M", "G", "T"]

    /**
     * The maximum number of characters in the output, excluding the negative sign
     */
    private static final Integer MAX_LENGTH = 4

    private static final Pattern TRAILING_DECIMAL_POINT = ~/[0-9]+\.[kMGT]/

    private static final Pattern METRIC_PREFIXED_NUMBER = ~/\-?[0-9]+(\.[0-9])?[kMGT]/

    @Override
    StringBuffer format(Object obj, StringBuffer output, FieldPosition pos) {

        Double number = obj as Double

        // if the number is negative, convert it to a positive number and add the minus sign to the output at the end
        boolean isNegative = number < 0
        number = Math.abs(number)

        String result = new DecimalFormat("##0E0").format(number)

        Integer index = Character.getNumericValue(result.charAt(result.size() - 1)) / 3
        result = result.replaceAll("E[0-9]", METRIC_PREFIXES[index])

        while (result.size() > MAX_LENGTH || TRAILING_DECIMAL_POINT.matcher(result).matches()) {
            int length = result.size()
            result = result.substring(0, length - 2) + result.substring(length - 1)
        }

        output << (isNegative ? "-$result" : result)
    }

    /**
     * Convert a String produced by <tt>format()</tt> back to a number. This will generally not restore
     * the original number because <tt>format()</tt> is a lossy operation, e.g.
     *
     * <pre>
     * {@code
     * def formatter = new RoundedMetricPrefixFormat()
     * Long number = 5821L
     * String formattedNumber = formatter.format(number)
     * assert formattedNumber == '5.8k'
     *
     * Long parsedNumber = formatter.parseObject(formattedNumber)
     * assert parsedNumber == 5800
     * assert parsedNumber != number
     * }
     * </pre>
     *
     * @param source a number that may have a metric prefix
     * @param pos if parsing succeeds, this should be updated to the index after the last parsed character
     * @return a Number if the the string is a number without a metric prefix, or a Long if it has a metric prefix
     */
    @Override
    Object parseObject(String source, ParsePosition pos) {

        if (source.isNumber()) {

            // if the value is a number (without a prefix) don't return it as a Long or we'll lose any decimals
            pos.index = source.size()
            toNumber(source)

        } else if (METRIC_PREFIXED_NUMBER.matcher(source).matches()) {

            boolean isNegative = source[0] == '-'

            String number = isNegative ? source[1..-2] : source[0..-2]
            String metricPrefix = source[-1]

            Number absoluteNumber = toNumber(number)

            Integer exponent = 3 * METRIC_PREFIXES.indexOf(metricPrefix)
            Long factor = 10 ** exponent
            factor *= isNegative ? -1 : 1

            pos.index = source.size()
            (absoluteNumber * factor) as Long
        }
    }

    private static Number toNumber(String number) {
        NumberUtils.createNumber(number)
    }
}

テスト(Groovy)

テストはGroovyで記述されていますが、JavaまたはGroovyクラスのどちらかを検証するために使用できます(両方の名前とAPIが同じであるため)。

import java.text.Format
import java.text.ParseException

class RoundedMetricPrefixFormatTests extends GroovyTestCase {

    private Format roundedMetricPrefixFormat = new RoundedMetricPrefixFormat()

    void testNumberFormatting() {

        [
                7L         : '7',
                12L        : '12',
                856L       : '856',
                1000L      : '1k',
                (-1000L)   : '-1k',
                5821L      : '5.8k',
                10500L     : '10k',
                101800L    : '102k',
                2000000L   : '2M',
                7800000L   : '7.8M',
                (-7800000L): '-7.8M',
                92150000L  : '92M',
                123200000L : '123M',
                9999999L   : '10M',
                (-9999999L): '-10M'
        ].each { Long rawValue, String expectedRoundValue ->

            assertEquals expectedRoundValue, roundedMetricPrefixFormat.format(rawValue)
        }
    }

    void testStringParsingSuccess() {
        [
                '7'    : 7,
                '8.2'  : 8.2F,
                '856'  : 856,
                '-856' : -856,
                '1k'   : 1000,
                '5.8k' : 5800,
                '-5.8k': -5800,
                '10k'  : 10000,
                '102k' : 102000,
                '2M'   : 2000000,
                '7.8M' : 7800000L,
                '92M'  : 92000000L,
                '-92M' : -92000000L,
                '123M' : 123000000L,
                '10M'  : 10000000L

        ].each { String metricPrefixNumber, Number expectedValue ->

            def parsedNumber = roundedMetricPrefixFormat.parseObject(metricPrefixNumber)
            assertEquals expectedValue, parsedNumber
        }
    }

    void testStringParsingFail() {

        shouldFail(ParseException) {
            roundedMetricPrefixFormat.parseObject('notNumber')
        }
    }
}

1
CSプレフィックスについて考えていると思います。彼は何十億、何兆という話をしているので、彼は短いスケールの数字を望んでいると思います。
jhurtado 2014

1
9999999は9.9mと印刷するはずです(数値は切り捨てられ、丸められません)。
アッシリア2015年

このソリューションでは、u(マイクロ)やm(ミリ)など、1より小さい値のプレフィックスはサポートされていません。
gbmhunter 2016年

13

ICUのlib ICUはあなたに読めるとmaintanableソリューションを与えるだろうと思う使用してI数spelloutなどのために使用することができる番号のルールベースのフォーマッタを、持っています。

[使用法]

正しいクラスはRuleBasedNumberFormatです。フォーマット自体は、個別のファイル(または文字列定数、IIRC)として保存できます。

http://userguide.icu-project.org/formatparse/numbersの

double num = 2718.28;
NumberFormat formatter = 
    new RuleBasedNumberFormat(RuleBasedNumberFormat.SPELLOUT);
String result = formatter.format(num);
System.out.println(result);

同じページにはローマ数字が表示されているので、あなたのケースも可能だと思います。


ローカリゼーションが必要な場合に完全にバラバラにならない、スレッド内の唯一のソリューション。
Grozz、2018

2
Android開発に必要な場合、これはすでにフレームワークに含まれています。を探しCompactDecimalFormatます。APIレベル24+
Gokhan Arik

10

Java-12 +は、使用することができますNumberFormat.getCompactNumberInstance数値をフォーマットします。あなたはとしてNumberFormat最初を作成することができます

NumberFormat fmt = NumberFormat.getCompactNumberInstance(Locale.US, NumberFormat.Style.SHORT);

次にそれを使用してformat

fmt.format(1000)
$5 ==> "1K"

fmt.format(10000000)
$9 ==> "10M"

fmt.format(1000000000)
$11 ==> "1B"

8

重要:にキャストする回答doubleのような数字のために失敗する99999999999999999Lと返す100P代わりに99P理由double用途IEEE標準

有効数字最大15桁の 10進数文字列をIEEE 754倍精度表現に変換し、同じ有効桁数の文字列に戻す場合、最終的な文字列は元の文字列と一致する必要があります。[ 有効数字long最大19桁です。]

System.out.println((long)(double)99999999999999992L); // 100000000000000000
System.out.println((long)(double)99999999999999991L); //  99999999999999984
// it is even worse for the logarithm:
System.out.println(Math.log10(99999999999999600L)); // 17.0
System.out.println(Math.log10(99999999999999500L)); // 16.999999999999996

このソリューションは、不要な桁を切り取り、すべてのlong値で機能します。シンプルだがパフォーマンスの高い実装(以下の比較)。-120kは4文字では表現できません。-0.1Mは長すぎるため、負の数の場合は5文字で問題ありません。

private static final char[] magnitudes = {'k', 'M', 'G', 'T', 'P', 'E'}; // enough for long

public static final String convert(long number) {
    String ret;
    if (number >= 0) {
        ret = "";
    } else if (number <= -9200000000000000000L) {
        return "-9.2E";
    } else {
        ret = "-";
        number = -number;
    }
    if (number < 1000)
        return ret + number;
    for (int i = 0; ; i++) {
        if (number < 10000 && number % 1000 >= 100)
            return ret + (number / 1000) + '.' + ((number % 1000) / 100) + magnitudes[i];
        number /= 1000;
        if (number < 1000)
            return ret + number + magnitudes[i];
    }
}

でテストelse if分であるため、先頭にはnecessairyある-(2^63)とmaxがある(2^63)-1ため、割り当てがnumber = -numberあれば失敗するでしょうnumber == Long.MIN_VALUE。チェックを行う必要がある場合は、単にをチェックするのではなく、できるだけ多くの数を含めることもできnumber == Long.MIN_VALUEます。

この実装と最も多くの賛成票を得た(現在は最速と言われている)実装を比較すると、5倍以上高速であることがわかりました(テスト設定に依存しますが、数値が多いほど、ゲインが大きくなり、この実装にはすべてのケースを処理するため、より多くのチェックを行う必要があるため、他のケースが修正されると、差はさらに大きくなります)。浮動小数点演算、対数、パワー、再帰、正規表現、洗練されたフォーマッター、作成されるオブジェクトの量の最小化がないため、これは非常に高速です。


ここにテストプログラムがあります:

public class Test {

    public static void main(String[] args) {
        long[] numbers = new long[20000000];
        for (int i = 0; i < numbers.length; i++)
            numbers[i] = Math.random() < 0.5 ? (long) (Math.random() * Long.MAX_VALUE) : (long) (Math.random() * Long.MIN_VALUE);
        System.out.println(convert1(numbers) + " vs. " + convert2(numbers));
    }

    private static long convert1(long[] numbers) {
        long l = System.currentTimeMillis();
        for (int i = 0; i < numbers.length; i++)
            Converter1.convert(numbers[i]);
        return System.currentTimeMillis() - l;
    }

    private static long convert2(long[] numbers) {
        long l = System.currentTimeMillis();
        for (int i = 0; i < numbers.length; i++)
            Converter2.coolFormat(numbers[i], 0);
        return System.currentTimeMillis() - l;
    }

}

可能な出力:(2309 vs. 11591正の数のみを使用する場合とほぼ同じで、実行順序を逆にする場合ははるかに極端です。ガベージコレクションに関係している可能性があります)


8

これは、再帰なしの短い実装で、非常に小さなループです。負の数では機能しませんが、までのすべての正longの数をサポートしますLong.MAX_VALUE

private static final char[] SUFFIXES = {'k', 'm', 'g', 't', 'p', 'e' };

public static String format(long number) {
    if(number < 1000) {
        // No need to format this
        return String.valueOf(number);
    }
    // Convert to a string
    final String string = String.valueOf(number);
    // The suffix we're using, 1-based
    final int magnitude = (string.length() - 1) / 3;
    // The number of digits we must show before the prefix
    final int digits = (string.length() - 1) % 3 + 1;

    // Build the string
    char[] value = new char[4];
    for(int i = 0; i < digits; i++) {
        value[i] = string.charAt(i);
    }
    int valueLength = digits;
    // Can and should we add a decimal point and an additional number?
    if(digits == 1 && string.charAt(1) != '0') {
        value[valueLength++] = '.';
        value[valueLength++] = string.charAt(1);
    }
    value[valueLength++] = SUFFIXES[magnitude - 1];
    return new String(value, 0, valueLength);
}

出力:

1k
5.8k
10k
101k
2m
7.8m
92m
123m
9.2e(これはLong.MAX_VALUE

私はまた、いくつかの本当に単純なベンチマーク(1000万のランダムなlongをフォーマットする)を行いました。これは、Elijahの実装よりもかなり速く、assyliasの実装よりもわずかに高速です。

鉱山:1137.028 ms
エリヤ:2664.396 ms
アッシリア:1373.473 ms


1
前回の更新で、バグを追加しました。現在は、番号101800に対して1kを返します。
スーフィアン2015


8

丸めたい人のために。これは、Java.Lang.Mathライブラリーを利用する、優れた読みやすいソリューションです。

 public static String formatNumberExample(Number number) {
        char[] suffix = {' ', 'k', 'M', 'B', 'T', 'P', 'E'};
        long numValue = number.longValue();
        int value = (int) Math.floor(Math.log10(numValue));
        int base = value / 3;
        if (value >= 3 && base < suffix.length) {
            return new DecimalFormat("~#0.0").format(numValue / Math.pow(10, base * 3)) + suffix[base];
        } else {
            return new DecimalFormat("#,##0").format(numValue);
        }
    }

8

次のコードは、簡単な拡張を考慮してこれを行う方法を示しています。

「魔法」は主に makeDecimal関数にあり、渡された正しい値については、出力に4文字を超えることはありません。

最初に、指定された除数の全体と10分の1の部分を抽出するため、たとえば、12,345,678除数が1,000,000whole場合12、のtenths値と3ます。

それから、次のルールを使用して、全体を出力するか、全体と10分の1の両方を出力するかを決定できます。

  • 10分の1がゼロの場合は、全体とサフィックスのみを出力します。
  • 全体が9より大きい場合は、全体とサフィックスのみを出力します。
  • それ以外の場合は、全体、10分の1、および接尾辞を出力します。

そのためのコードは次のとおりです。

static private String makeDecimal(long val, long div, String sfx) {
    val = val / (div / 10);
    long whole = val / 10;
    long tenths = val % 10;
    if ((tenths == 0) || (whole >= 10))
        return String.format("%d%s", whole, sfx);
    return String.format("%d.%d%s", whole, tenths, sfx);
}

次に、ヘルパー関数を正しい値で呼び出すだけの簡単な方法です。定数を含めて、開発者の作業を簡単にします。

static final long THOU =                1000L;
static final long MILL =             1000000L;
static final long BILL =          1000000000L;
static final long TRIL =       1000000000000L;
static final long QUAD =    1000000000000000L;
static final long QUIN = 1000000000000000000L;

static private String Xlat(long val) {
    if (val < THOU) return Long.toString(val);
    if (val < MILL) return makeDecimal(val, THOU, "k");
    if (val < BILL) return makeDecimal(val, MILL, "m");
    if (val < TRIL) return makeDecimal(val, BILL, "b");
    if (val < QUAD) return makeDecimal(val, TRIL, "t");
    if (val < QUIN) return makeDecimal(val, QUAD, "q");
    return makeDecimal(val, QUIN, "u");
}

makeDecimal関数が大した仕事をするという事実は、それを超えて拡張999,999,999することが、に余分な行を追加するだけの問題であることを意味しXlatます。

64ビットの符号付きlongで保持できる最大値は約9.2兆であるため、最後の入力returnXlat条件付きである必要はありません。

しかし、奇妙な要件により、Oracleが128ビットlongerタイプまたは1024 ビットタイプを追加することを決定した場合、そのdamn_long準備ができます:-)


そして最後に、機能の検証に使用できる小さなテストハーネスです。

public static void main(String[] args) {
    long vals[] = {
        999L, 1000L, 5821L, 10500L, 101800L, 2000000L,
        7800000L, 92150000L, 123200000L, 999999999L,
        1000000000L, 1100000000L, 999999999999L,
        1000000000000L, 999999999999999L,
        1000000000000000L, 9223372036854775807L
    };
    for (long val: vals)
        System.out.println ("" + val + " -> " + Xlat(val));
    }
}

出力から、必要なものが得られることがわかります。

999 -> 999
1000 -> 1k
5821 -> 5.8k
10500 -> 10k
101800 -> 101k
2000000 -> 2m
7800000 -> 7.8m
92150000 -> 92m
123200000 -> 123m
999999999 -> 999m
1000000000 -> 1b
1100000000 -> 1.1b
999999999999 -> 999b
1000000000000 -> 1t
999999999999999 -> 999t
1000000000000000 -> 1q
9223372036854775807 -> 9.2u

また、余談ですが、この関数に負の数を渡すと、< THOUパスに続くため、文字列が要件に対して長すぎることに注意してください。質問で負でない値のみを言及しているので、それは大丈夫だと思いました。


6

それが最善のアプローチかどうかはわかりませんが、これは私がやったことです。

7=>7
12=>12
856=>856
1000=>1.0k
5821=>5.82k
10500=>10.5k
101800=>101.8k
2000000=>2.0m
7800000=>7.8m
92150000=>92.15m
123200000=>123.2m
9999999=>10.0m

---コード---

public String Format(Integer number){
    String[] suffix = new String[]{"k","m","b","t"};
    int size = (number.intValue() != 0) ? (int) Math.log10(number) : 0;
    if (size >= 3){
        while (size % 3 != 0) {
            size = size - 1;
        }
    }
    double notation = Math.pow(10, size);
    String result = (size >= 3) ? + (Math.round((number / notation) * 100) / 100.0d)+suffix[(size/3) - 1] : + number + "";
    return result
}

6

大きな数値を小さな数値(2桁)に変換するための関数。桁数の変更#.##DecimalFormat

public String formatValue(float value) {
    String arr[] = {"", "K", "M", "B", "T", "P", "E"};
    int index = 0;
    while ((value / 1000) >= 1) {
        value = value / 1000;
        index++;
    }
    DecimalFormat decimalFormat = new DecimalFormat("#.##");
    return String.format("%s %s", decimalFormat.format(value), arr[index]);
}

テスト中

System.out.println(formatValue(100));     //  100
System.out.println(formatValue(1000));    // 1 K
System.out.println(formatValue(10345));   // 10.35 K
System.out.println(formatValue(10012));   // 10.01 K
System.out.println(formatValue(123456));  // 123.46 K
System.out.println(formatValue(4384324)); // 4.38 M
System.out.println(formatValue(10000000)); // 10 M
System.out.println(formatValue(Long.MAX_VALUE)); // 9.22 E

それが役に立てば幸い


5

私のJavaは錆びていますが、C#で実装する方法は次のとおりです。

private string  FormatNumber(double value)
    {
    string[]  suffixes = new string[] {" k", " m", " b", " t", " q"};
    for (int j = suffixes.Length;  j > 0;  j--)
        {
        double  unit = Math.Pow(1000, j);
        if (value >= unit)
            return (value / unit).ToString("#,##0.0") + suffixes[--j];
        }
    return value.ToString("#,##0");
    }

これを調整して、メートルキロではなくCSキロ(1,024)を使用したり、単位を追加したりするのは簡単です。1,000を「1 k」ではなく「1.0 k」としてフォーマットしますが、それは重要ではないと信じています。

「4文字以内」というより具体的な要件を満たすには、接尾辞の前のスペースを削除し、中央のブロックを次のように調整します。

if (value >= unit)
  {
  value /= unit;
  return (value).ToString(value >= unit * 9.95 ? "#,##0" : "#,##0.0") + suffixes[--j];
  }

1
残念ながら、このToStringメソッドはJavaには存在しません-他の問題(ロケール依存など)を引き起こす可能性のあるNumberFormatが必要になります。
アッシリア2015年

5

お気に入り。電子ドメインで一般的なように、10進数のインジケータとして「k」などを使用することもできます。これにより、追加のスペースなしで追加の数字が得られます

2列目はできるだけ多くの桁を使用しようとします

1000 => 1.0k | 1000
5821 => 5.8k | 5821
10500 => 10k | 10k5
101800 => 101k | 101k
2000000 => 2.0m | 2m
7800000 => 7.8m | 7m8
92150000 => 92m | 92m1
123200000 => 123m | 123m
9999999 => 9.9m | 9m99

これはコードです

public class HTTest {
private static String[] unit = {"u", "k", "m", "g", "t"};
/**
 * @param args
 */
public static void main(String[] args) {
    int[] numbers = new int[]{1000, 5821, 10500, 101800, 2000000, 7800000, 92150000, 123200000, 9999999};
    for(int n : numbers) {
        System.out.println(n + " => " + myFormat(n) + " | " + myFormat2(n));
    }
}

private static String myFormat(int pN) {
    String str = Integer.toString(pN);
    int len = str.length ()-1;
    if (len <= 3) return str;
    int level = len / 3;
    int mode = len % 3;
    switch (mode) {
    case 0: return str.substring(0, 1) + "." + str.substring(1, 2) + unit[level];
    case 1: return str.substring(0, 2) + unit[level];
    case 2: return str.substring(0, 3) + unit[level];
    }
    return "how that?";
}
private static String trim1 (String pVal) {
    if (pVal.equals("0")) return "";
    return pVal;
}
private static String trim2 (String pVal) {
    if (pVal.equals("00")) return "";
    return pVal.substring(0, 1) + trim1(pVal.substring(1,2));
}
private static String myFormat2(int pN) {
    String str = Integer.toString(pN);
    int len = str.length () - 1;
    if (len <= 3) return str;
    int level = len / 3;
    int mode = len % 3;
    switch (mode) {
    case 0: return str.substring(0, 1) + unit[level] + trim2(str.substring(1, 3));
    case 2: return str.substring(0, 3) + unit[level];
    case 1: return str.substring(0, 2) + unit[level] + trim1(str.substring(2, 3));
    }
    return "how that?";
}
}

4

パフォーマンスよりも読みやすさを重視するという私のコメントに忠実であり、これは何が起こっているのかを明確にすべきバージョンです(あなたが使用したと仮定して) BigDecimal守り、パフォーマンスを心配することなく、過度のコメントを付けずに(以前にs(自己文書化コードを信じています)。 (私がこれを何百万回も実行したいシナリオを描くことができないので、パフォーマンスが考慮事項になることもあります)。

このバージョン:

  • BigDecimal精度と丸めの問題を回避するためにsを使用します
  • OPの要求に応じて切り捨てる
  • 他の丸めモードで機能します(HALF_UPテストなど)。
  • 精度を調整できます(変更REQUIRED_PRECISION
  • を使用しenumてしきい値を定義します。つまり、k / m / b / tなどの代わりにKB / MB / GB / TBを使用するように簡単に調整できます。もちろんTRILLION、必要に応じて拡張することもできます。
  • 問題のテストケースは境界をテストしていないため、完全なユニットテストが付属しています
  • ゼロと負の数で機能するはずです

Threshold.java

import java.math.BigDecimal;

public enum Threshold {
  TRILLION("1000000000000", 12, 't', null),
  BILLION("1000000000", 9, 'b', TRILLION),
  MILLION("1000000", 6, 'm', BILLION),
  THOUSAND("1000", 3, 'k', MILLION),
  ZERO("0", 0, null, THOUSAND);

  private BigDecimal value;
  private int zeroes;
  protected Character suffix;
  private Threshold higherThreshold;

  private Threshold(String aValueString, int aNumberOfZeroes, Character aSuffix,
      Threshold aThreshold) {
    value = new BigDecimal(aValueString);
    zeroes = aNumberOfZeroes;
    suffix = aSuffix;
    higherThreshold = aThreshold;
  }

  public static Threshold thresholdFor(long aValue) {
    return thresholdFor(new BigDecimal(aValue));
  }

  public static Threshold thresholdFor(BigDecimal aValue) {
    for (Threshold eachThreshold : Threshold.values()) {
      if (eachThreshold.value.compareTo(aValue) <= 0) {
        return eachThreshold;
      }
    }
    return TRILLION; // shouldn't be needed, but you might have to extend the enum
  }

  public int getNumberOfZeroes() {
    return zeroes;
  }

  public String getSuffix() {
    return suffix == null ? "" : "" + suffix;
  }

  public Threshold getHigherThreshold() {
    return higherThreshold;
  }
}

NumberShortener.java

import java.math.BigDecimal;
import java.math.RoundingMode;

public class NumberShortener {

  public static final int REQUIRED_PRECISION = 2;

  public static BigDecimal toPrecisionWithoutLoss(BigDecimal aBigDecimal,
      int aPrecision, RoundingMode aMode) {
    int previousScale = aBigDecimal.scale();
    int previousPrecision = aBigDecimal.precision();
    int newPrecision = Math.max(previousPrecision - previousScale, aPrecision);
    return aBigDecimal.setScale(previousScale + newPrecision - previousPrecision,
        aMode);
  }

  private static BigDecimal scaledNumber(BigDecimal aNumber, RoundingMode aMode) {
    Threshold threshold = Threshold.thresholdFor(aNumber);
    BigDecimal adjustedNumber = aNumber.movePointLeft(threshold.getNumberOfZeroes());
    BigDecimal scaledNumber = toPrecisionWithoutLoss(adjustedNumber, REQUIRED_PRECISION,
        aMode).stripTrailingZeros();
    // System.out.println("Number: <" + aNumber + ">, adjusted: <" + adjustedNumber
    // + ">, rounded: <" + scaledNumber + ">");
    return scaledNumber;
  }

  public static String shortenedNumber(long aNumber, RoundingMode aMode) {
    boolean isNegative = aNumber < 0;
    BigDecimal numberAsBigDecimal = new BigDecimal(isNegative ? -aNumber : aNumber);
    Threshold threshold = Threshold.thresholdFor(numberAsBigDecimal);
    BigDecimal scaledNumber = aNumber == 0 ? numberAsBigDecimal : scaledNumber(
        numberAsBigDecimal, aMode);
    if (scaledNumber.compareTo(new BigDecimal("1000")) >= 0) {
      scaledNumber = scaledNumber(scaledNumber, aMode);
      threshold = threshold.getHigherThreshold();
    }
    String sign = isNegative ? "-" : "";
    String printNumber = sign + scaledNumber.stripTrailingZeros().toPlainString()
        + threshold.getSuffix();
    // System.out.println("Number: <" + sign + numberAsBigDecimal + ">, rounded: <"
    // + sign + scaledNumber + ">, print: <" + printNumber + ">");
    return printNumber;
  }
}

printlnステートメントのコメントを外すか、お気に入りのロガーを使用してそれが何をしているかを確認するように変更します。)

そして最後に、NumberShortenerTest(プレーンJUnit 4)のテスト:

import static org.junit.Assert.*;

import java.math.BigDecimal;
import java.math.RoundingMode;

import org.junit.Test;

public class NumberShortenerTest {

  private static final long[] NUMBERS_FROM_OP = new long[] { 1000, 5821, 10500, 101800, 2000000, 7800000, 92150000, 123200000 };
  private static final String[] EXPECTED_FROM_OP = new String[] { "1k", "5.8k", "10k", "101k", "2m", "7.8m", "92m", "123m" };
  private static final String[] EXPECTED_FROM_OP_HALF_UP = new String[] { "1k", "5.8k", "11k", "102k", "2m", "7.8m", "92m", "123m" };
  private static final long[] NUMBERS_TO_TEST = new long[] { 1, 500, 999, 1000, 1001, 1009, 1049, 1050, 1099, 1100, 12345, 123456, 999999, 1000000,
      1000099, 1000999, 1009999, 1099999, 1100000, 1234567, 999999999, 1000000000, 9123456789L, 123456789123L };
  private static final String[] EXPECTED_FROM_TEST = new String[] { "1", "500", "999", "1k", "1k", "1k", "1k", "1k", "1k", "1.1k", "12k", "123k",
      "999k", "1m", "1m", "1m", "1m", "1m", "1.1m", "1.2m", "999m", "1b", "9.1b", "123b" };
  private static final String[] EXPECTED_FROM_TEST_HALF_UP = new String[] { "1", "500", "999", "1k", "1k", "1k", "1k", "1.1k", "1.1k", "1.1k", "12k",
      "123k", "1m", "1m", "1m", "1m", "1m", "1.1m", "1.1m", "1.2m", "1b", "1b", "9.1b", "123b" };

  @Test
  public void testThresholdFor() {
    assertEquals(Threshold.ZERO, Threshold.thresholdFor(1));
    assertEquals(Threshold.ZERO, Threshold.thresholdFor(999));
    assertEquals(Threshold.THOUSAND, Threshold.thresholdFor(1000));
    assertEquals(Threshold.THOUSAND, Threshold.thresholdFor(1234));
    assertEquals(Threshold.THOUSAND, Threshold.thresholdFor(9999));
    assertEquals(Threshold.THOUSAND, Threshold.thresholdFor(999999));
    assertEquals(Threshold.MILLION, Threshold.thresholdFor(1000000));
  }

  @Test
  public void testToPrecision() {
    RoundingMode mode = RoundingMode.DOWN;
    assertEquals(new BigDecimal("1"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 1, mode));
    assertEquals(new BigDecimal("1.2"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 2, mode));
    assertEquals(new BigDecimal("1.23"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 3, mode));
    assertEquals(new BigDecimal("1.234"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 4, mode));
    assertEquals(new BigDecimal("999").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999"), 4, mode).stripTrailingZeros()
        .toPlainString());
    assertEquals(new BigDecimal("999").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999"), 2, mode).stripTrailingZeros()
        .toPlainString());
    assertEquals(new BigDecimal("999").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999.9"), 2, mode).stripTrailingZeros()
        .toPlainString());

    mode = RoundingMode.HALF_UP;
    assertEquals(new BigDecimal("1"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 1, mode));
    assertEquals(new BigDecimal("1.2"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 2, mode));
    assertEquals(new BigDecimal("1.23"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 3, mode));
    assertEquals(new BigDecimal("1.235"), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("1.23456"), 4, mode));
    assertEquals(new BigDecimal("999").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999"), 4, mode).stripTrailingZeros()
        .toPlainString());
    assertEquals(new BigDecimal("999").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999"), 2, mode).stripTrailingZeros()
        .toPlainString());
    assertEquals(new BigDecimal("1000").toPlainString(), NumberShortener.toPrecisionWithoutLoss(new BigDecimal("999.9"), 2, mode)
        .stripTrailingZeros().toPlainString());
  }

  @Test
  public void testNumbersFromOP() {
    for (int i = 0; i < NUMBERS_FROM_OP.length; i++) {
      assertEquals("Index " + i + ": " + NUMBERS_FROM_OP[i], EXPECTED_FROM_OP[i],
          NumberShortener.shortenedNumber(NUMBERS_FROM_OP[i], RoundingMode.DOWN));
      assertEquals("Index " + i + ": " + NUMBERS_FROM_OP[i], EXPECTED_FROM_OP_HALF_UP[i],
          NumberShortener.shortenedNumber(NUMBERS_FROM_OP[i], RoundingMode.HALF_UP));
    }
  }

  @Test
  public void testBorders() {
    assertEquals("Zero: " + 0, "0", NumberShortener.shortenedNumber(0, RoundingMode.DOWN));
    assertEquals("Zero: " + 0, "0", NumberShortener.shortenedNumber(0, RoundingMode.HALF_UP));
    for (int i = 0; i < NUMBERS_TO_TEST.length; i++) {
      assertEquals("Index " + i + ": " + NUMBERS_TO_TEST[i], EXPECTED_FROM_TEST[i],
          NumberShortener.shortenedNumber(NUMBERS_TO_TEST[i], RoundingMode.DOWN));
      assertEquals("Index " + i + ": " + NUMBERS_TO_TEST[i], EXPECTED_FROM_TEST_HALF_UP[i],
          NumberShortener.shortenedNumber(NUMBERS_TO_TEST[i], RoundingMode.HALF_UP));
    }
  }

  @Test
  public void testNegativeBorders() {
    for (int i = 0; i < NUMBERS_TO_TEST.length; i++) {
      assertEquals("Index " + i + ": -" + NUMBERS_TO_TEST[i], "-" + EXPECTED_FROM_TEST[i],
          NumberShortener.shortenedNumber(-NUMBERS_TO_TEST[i], RoundingMode.DOWN));
      assertEquals("Index " + i + ": -" + NUMBERS_TO_TEST[i], "-" + EXPECTED_FROM_TEST_HALF_UP[i],
          NumberShortener.shortenedNumber(-NUMBERS_TO_TEST[i], RoundingMode.HALF_UP));
    }
  }
}

重要なテストケースを逃した場合、または期待値を調整する必要がある場合は、コメントで自由に指摘してください。


ソリューションの唯一の明らかな欠点は、コードのV + Hスクロールバーに見えるため、読みやすさが低下します。明確性を失わずに再フォーマットが可能だと思いますか?
Wolf

@Wolf:私のIDEからのコピー/貼り付けを回避することを望んでいましたが、そうです、読みやすさを主張し、水平スクロールを要求することは私にとって偽善的です。;-)コードの最初の2ビットを更新しました。これは、何が起こっているのかを確認するために見ているコードですが、テストコード自体はあまり役に立ちません。 dテストが機能することを確認したい場合は、ユニットテストを実行するために、それを独自のIDEに貼り付けます。大丈夫だと思います。
Amos M. Carpenter 2015年

あぁ、いいね。しかし、最後のボックスのテストケースでは、予想される結果は-光学的に-入力との関連性が高い(最初の6つの配列のリテラルを意味する)。
Wolf

@Wolf:私は、スペースまたはタブを使用してアイテムを一列に配置しようとするのは好きではありません-私のお気に入りのフォーマッター(Eclipse)のすべてのケースで一貫して簡単に構成することはできません。手動で行うと... 、アイテムを追加または削除するたびに行う必要があるすべての調整のため。実際にそれらを整列させて表示したい場合は、数値/値をCSVとしてスプレッドシートに貼り付けます。
Amos M. Carpenter 2015年

1
@assylias、あなたが何を求めているかによります。1回限りの使用例を解決した直後であれば、ソリューションは正常に機能するはずです。私はTreeMapアプローチが好きです。「読みやすさ」はもちろん主観的です。;-)では、誰かがあなたのバージョンで切り捨てるのとは異なる方法で丸めたい場合はどうでしょうか?(たとえば、これを使用してファイルサイズを示す場合、誰を切り捨てますか?)10ではなく2の累乗が必要な場合は、かなり書き直さないといけませんね。私が言ったように、私は意図的にコードをゴルフしようとはしていませんでした。
Amos M. Carpenter

4

これは私のコードです。清潔でシンプル。

public static String getRoughNumber(long value) {
    if (value <= 999) {
        return String.valueOf(value);
    }

    final String[] units = new String[]{"", "K", "M", "B", "P"};
    int digitGroups = (int) (Math.log10(value) / Math.log10(1000));
    return new DecimalFormat("#,##0.#").format(value / Math.pow(1000, digitGroups)) + "" + units[digitGroups];

}

2

私自身の答え、Javaコード、自明のコードを追加します。

import java.math.BigDecimal;

/**
 * Method to convert number to formatted number.
 * 
 * @author Gautham PJ
 */
public class ShortFormatNumbers
{

    /**
     * Main method. Execution starts here.
     */
    public static void main(String[] args)
    {

        // The numbers that are being converted.
        int[] numbers = {999, 1400, 2500, 45673463, 983456, 234234567};


        // Call the "formatNumber" method on individual numbers to format 
        // the number.
        for(int number : numbers)
        {
            System.out.println(number + ": " + formatNumber(number));
        }

    }


    /**
     * Format the number to display it in short format.
     * 
     * The number is divided by 1000 to find which denomination to be added 
     * to the number. Dividing the number will give the smallest possible 
     * value with the denomination.
     * 
     * @param the number that needs to be converted to short hand notation.
     * @return the converted short hand notation for the number.
     */
    private static String formatNumber(double number)
    {
        String[] denominations = {"", "k", "m", "b", "t"};
        int denominationIndex = 0;

        // If number is greater than 1000, divide the number by 1000 and 
        // increment the index for the denomination.
        while(number > 1000.0)
        {
            denominationIndex++;
            number = number / 1000.0;
        }

        // To round it to 2 digits.
        BigDecimal bigDecimal = new BigDecimal(number);
        bigDecimal = bigDecimal.setScale(2, BigDecimal.ROUND_HALF_EVEN);


        // Add the number with the denomination to get the final value.
        String formattedNumber = bigDecimal + denominations[denominationIndex];
        return formattedNumber;
    }

}

1

このコードスニペットは、非常にシンプルでクリーンなコードであり、完全に機能します。

private static char[] c = new char[]{'K', 'M', 'B', 'T'};
private String formatK(double n, int iteration) {
    if (n < 1000) {
        // print 999 or 999K
        if (iteration <= 0) {
            return String.valueOf((long) n);
        } else {
            return String.format("%d%s", Math.round(n), c[iteration-1]);
        }
    } else if (n < 10000) {
        // Print 9.9K
        return String.format("%.1f%s", n/1000, c[iteration]);
    } else {
        // Increase 1 iteration
        return formatK(Math.round(n/1000), iteration+1);
    }
}

1

これを試して :

public String Format(Integer number){
    String[] suffix = new String[]{"k","m","b","t"};
    int size = (number.intValue() != 0) ? (int) Math.log10(number) : 0;
    if (size >= 3){
        while (size % 3 != 0) {
            size = size - 1;
        }
    }
    double notation = Math.pow(10, size);
    String result = (size >= 3) ? + (Math.round((number / notation) * 100) / 100.0d)+suffix[(size/3) - 1] : + number + "";
    return result
}

1
public class NumberToReadableWordFormat {

    public static void main(String[] args) {
        Integer[] numbers = new Integer[]{1000, 5821, 10500, 101800, 2000000, 7800000, 92150000, 123200000, 9999999,999};
        for(int n : numbers) {
            System.out.println(n + " => " + coolFormat(n));
        }
    }

    private static String[] c = new String[]{"K", "L", "Cr"};
    private static String coolFormat(int n) {
        int size = String.valueOf(n).length();
        if (size>=4 && size<6) {
                int value = (int) Math.pow(10, 1);
                double d = (double) Math.round(n/1000.0 * value) / value;
                return (double) Math.round(n/1000.0 * value) / value+" "+c[0];
        } else if(size>5 && size<8) {
                int value = (int) Math.pow(10, 1);
                return (double) Math.round(n/100000.0 * value) / value+" "+c[1];
        } else if(size>=8) {
                int value = (int) Math.pow(10, 1);
                return (double) Math.round(n/10000000.0 * value) / value+" "+c[2];
        } else {
            return n+"";
        }
    }
}

出力:

1000 => 1.0 K

5821 => 5.8 K

10500 => 10.5 K

101800 => 1.0 L

2000000 => 20.0 L

7800000 => 78.0 L

92150000 => 9.2 Cr

123200000 => 12.3 Cr

9999999 => 100.0 L

999 => 999

0
//code longer but work sure...

public static String formatK(int number) {
    if (number < 999) {
        return String.valueOf(number);
    }

    if (number < 9999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 1);
        String str2 = strNumber.substring(1, 2);
        if (str2.equals("0")) {
            return str1 + "k";
        } else {
            return str1 + "." + str2 + "k";
        }
    }

    if (number < 99999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 2);
        return str1 + "k";
    }

    if (number < 999999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 3);
        return str1 + "k";
    }

    if (number < 9999999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 1);
        String str2 = strNumber.substring(1, 2);
        if (str2.equals("0")) {
            return str1 + "m";
        } else {
            return str1 + "." + str2 + "m";
        }
    }

    if (number < 99999999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 2);
        return str1 + "m";
    }

    if (number < 999999999) {
        String strNumber = String.valueOf(number);
        String str1 = strNumber.substring(0, 3);
        return str1 + "m";
    }

    NumberFormat formatterHasDigi = new DecimalFormat("###,###,###");
    return formatterHasDigi.format(number);
}

2
これは、すべてのエッジケースで機能するわけではありません。たとえば、999を試してください。
jzd 2013年
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.