दिलचस्प पोस्ट
वर्ग कोष्ठक के बीच पाठ को निकालने के लिए नियमित अभिव्यक्ति कक्षा में जावास्क्रिप्ट में तत्व कैसे प्राप्त करें? यदि ऑपरेटर के बराबर काम नहीं करता है, तो यह अंतरिक्ष से घिरा नहीं है? एंड्रॉइड पर SQLite के लिए सर्वोत्तम प्रथाएं क्या हैं? एंड्रॉइड: फ़ोकस एक संपादन टेक्स्ट पर है, तो स्वचालित रूप से नरम कीबोर्ड दिखाएं एंड्रॉइड में सर्कुलर इमेजिव्यू कैसे बनाएं? पर्ल के आईएनसी का निर्माण कैसे किया जाता है? (उर्फ क्या पर्ल मॉड्यूल की खोज करने के सभी तरीके क्या हैं?) जावास्क्रिप्ट आज तक प्रमुख शून्य जोड़ते हैं ASP.NET में समय क्षेत्र के साथ कैसे काम करें? किसी सेट से एक यादृच्छिक तत्व को चुनना टीएफएस 2013 निर्माण .NET 4.6 / सी # 6.0 एक्सटेंशन प्रकारों के संकलन में संगतता की जांच करना, और उन्हें सीडीएफ़ के साथ प्रयोग करना jQuery अजाक्स सफलता अनाम समारोह गुंजाइश आपको जेफ्राम और अन्य घटकों का विस्तार क्यों नहीं करना चाहिए? एंड्रॉइड में एक गतिविधि से किसी अन्य मूल्य का मूल्य कैसे पास करें?

कैसे जावा में 1200 से 1.2k के स्वरूपण के बारे में जाना

मैं निम्नलिखित संख्याओं को उनके साथ जावा के साथ संख्याओं में स्वरूपित करना चाहता हूं:

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 

दाईं ओर की संख्या लंबी या पूर्णांक होगी, बाईं ओर की संख्या स्ट्रिंग होगी। मुझे इस दृष्टिकोण से कैसे जाना चाहिए I मैंने पहले से ही इस के लिए थोड़ा एल्गोरिदम किया था लेकिन मैंने सोचा था कि वहां पहले से ही कुछ ऐसा आविष्कार किया जा सकता है जो उस पर अच्छा काम करता है और अगर मैं अरबों और ट्रिलियन्स के साथ काम करना शुरू कर देता हूं तो अतिरिक्त परीक्षण की आवश्यकता नहीं है 🙂

अतिरिक्त आवश्यकताएं:

  • प्रारूप में अधिकतम 4 वर्ण होना चाहिए
  • उपर्युक्त साधन 1.1k ठीक है 11.2k नहीं है। 7.8 एम के लिए ठीक है 19.1 मीटर ठीक नहीं है। दशमलव अंक से पहले केवल दशमलव अंक दशमलव बिंदु होने की अनुमति है दशमलव अंक से दो अंकों का मतलब दशमलव अंक के बाद अंक नहीं है।
  • कोई गोलाई आवश्यक नहीं है (कश्मीर और एम के साथ प्रदर्शित संख्याएं एनालॉग गेज की अधिक होती हैं जो तर्क के सटीक लेख नहीं दर्शाती हैं। इसलिए मुख्य रूप से चर के प्रकृति के कारण अप्रासंगिक है क्योंकि आप कैश्ड परिणाम को देख रहे हैं, तब भी कई अंकों को बढ़ा या कमी कर सकते हैं।)

वेब के समाधान से एकत्रित समाधान "कैसे जावा में 1200 से 1.2k के स्वरूपण के बारे में जाना"

यहां एक समाधान है जो किसी भी लंबे मूल्य के लिए काम करता है और मुझे काफी पठनीय (मुख्य तर्क format विधि के निचले तीन रेखाओं में किया जाता है)।

यह उपयुक्त प्रत्यय को खोजने के लिए TreeMap को 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); } } 

मुझे पता है, यह सी प्रोग्राम की तरह दिखता है, लेकिन यह हल्का सुपर है!

 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, ie '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 

यहां एक समाधान है जो दशमलव के इंजीनियरिंग नोटेशन का उपयोग करता है:

 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]+\\.[az]")){ 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 

कुछ सुधार की आवश्यकता है, लेकिन: बचाव के लिए सख्त मैथ!
आप एक स्ट्रिंग या सरणी में प्रत्यय डाल सकते हैं और शक्ति पर आधारित fetch'em, या ऐसा कुछ कर सकते हैं।
विभाजन को सत्ता के आसपास भी प्रबंधित किया जा सकता है, मुझे लगता है कि लगभग सब कुछ बिजली मूल्य के बारे में है। आशा करता हूँ की ये काम करेगा!

 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

वर्तमान उत्तर के साथ समस्याएं

  • मौजूदा समाधानों में से कई इन उपसर्गों का उपयोग कर रहे हैं k = 10 3 , m = 10 6 , b = 10 9 , t = 10 12 हालांकि, विभिन्न स्रोतों के अनुसार, सही उपसर्गों के = 10 3 , एम = 10 6 , जी = 10 9 , टी = 10 12
  • नकारात्मक संख्याओं के लिए समर्थन का अभाव (या कम से कम परीक्षणों की कमी है जो दर्शाती है कि नकारात्मक संख्याएं समर्थित हैं)
  • व्युत्क्रम ऑपरेशन के लिए समर्थन की कमी, जैसे कि 1.1k से 1100 को परिवर्तित करना (हालांकि यह मूल प्रश्न के दायरे के बाहर है)

जावा समाधान

यह समाधान ( इस उत्तर का विस्तार) उपरोक्त मुद्दों को संबोधित करता है

 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, eg * * <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 समाधान

समाधान मूलतः ग्रोवी में लिखा गया था जैसा कि नीचे दिखाया गया है।

 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, eg * * <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) } } 

टेस्ट (ग्रूवी)

परीक्षण ग्रूवी में लिखे गए हैं लेकिन इसका उपयोग जावा या ग्रूवी क्लास या तो सत्यापित करने के लिए किया जा सकता है (क्योंकि वे दोनों एक ही नाम और एपीआई हैं)।

 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') } } } 

आईसीयू लिब के लिए एक नियम आधारित फॉर्मेटर है, जो नंबर स्पेलआउट इत्यादि के लिए इस्तेमाल किया जा सकता है। मुझे लगता है कि आईसीयू का उपयोग करके आपको एक पठनीय और रखरखाव योग्य समाधान मिल जाएगा।

[उपयोग]

सही वर्ग का नियम-बिज़्ड संख्याफ़ॉर्मेट है। स्वरूप को अलग फाइल के रूप में संग्रहित किया जा सकता है (या स्ट्रिंग स्थिर, आईआईआरसी)।

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); 

वही पृष्ठ रोमन अंकों को दिखाता है, इसलिए मुझे लगता है कि आपका केस भी संभव होना चाहिए, भी।

महत्वपूर्ण: double करने के लिए कास्टिंग करने का जवाब 99999999999999999L जैसी संख्याओं के लिए असफल हो जायेगा और 99P बजाए 100P लौटाएगा क्योंकि double IEEE मानक का उपयोग करता है:

अगर एक 15 दशमलव अंकों के साथ एक दशमलव स्ट्रिंग आईईईई 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 मूल्यों के लिए काम करता है । सरल लेकिन कार्यान्वयन कार्यान्वयन (नीचे तुलना) -120 क 4 अक्षरों के साथ व्यक्त नहीं किया जा सकता, यहां तक ​​कि -0.1 एम बहुत लंबा है, इसलिए नकारात्मक संख्या 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 परीक्षा शुरू हो जाती है, else if यह न्यूनतम -(2^63) और अधिकतम है (2^63)-1 और इसलिए असाइनमेंट number = -number number == Long.MIN_VALUE विफल होगा अगर number == Long.MIN_VALUE अगर हमें एक जांच करना है, तो हम संख्या के लिए जाँच करने के बजाय जितनी संभव हो सके उतनी संख्याओं को भी शामिल कर सकते हैं number == Long.MIN_VALUE

जिसने सबसे ज्यादा अपवोट्स प्राप्त करने वाले (वर्तमान में सबसे तेज़ कहा जा रहा है) के साथ इस कार्यान्वयन की तुलना में यह दिखाया गया है कि यह 5 गुना तेज है (यह टेस्ट सेटिंग्स पर निर्भर करता है, लेकिन अधिक संख्याओं के साथ लाभ अधिक हो जाता है और यह कार्यान्वयन अधिक जांच करने के लिए क्योंकि यह सभी मामलों को संभालता है, इसलिए यदि दूसरा तय हो जाए तो अंतर भी बड़ा हो जाएगा) यह वह उपवास है क्योंकि कोई फ़्लोटिंग प्वाइंट ऑपरेशन नहीं है, कोई लॉगरिथ्म नहीं, कोई शक्ति नहीं, कोई पुनरावर्ती नहीं, कोई regex नहीं, कोई अत्याधुनिक फॉर्मैटर्स और बनाए गए ऑब्जेक्ट की मात्रा का न्यूनतम स्तर


यहां परीक्षा कार्यक्रम है:

 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 (उसी के बारे में जब केवल निष्पादन के आदेश को उलटते समय सकारात्मक संख्याओं और अधिक चरम का उपयोग किया जाता है, तो शायद कचरा संग्रहण के साथ कुछ ऐसा होता है)

मेरा जावा जंग खाए है, लेकिन यहां बताया गया है कि मैं इसे सी # में कैसे लागू करूं:

 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"); } 

मीट्रिक किलो की बजाय सीएस किलो (1,024) का उपयोग करने के लिए या अधिक इकाइयों को जोड़ने के लिए इसे समायोजित करना आसान होगा यह 1,000 "1 k" के बजाय "1.0 k" के रूप में प्रारूपित करता है, लेकिन मुझे विश्वास है कि यह बेकार है।

अधिक विशिष्ट आवश्यकता "चार अक्षरों से अधिक नहीं" को पूरा करने के लिए, प्रत्ययों से पहले रिक्त स्थान निकालें और इस तरह से मध्य ब्लॉक समायोजित करें:

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

मुझे नहीं पता कि यह सबसे अच्छा तरीका है, लेकिन मैं यही करता हूं।

 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 } 

मेरा मनपसंद। आप "कश्मीर" का उपयोग कर सकते हैं और इसी तरह से दशमलव के लिए सूचक भी, जैसा कि इलेक्ट्रॉनिक डोमेन में आम है इससे आपको अतिरिक्त स्थान के बिना एक अतिरिक्त अंक मिलेगा

दूसरा स्तंभ जितना संभव हो उतना अंक उपयोग करने की कोशिश करता है

 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?"; } } 

यहां रिकर्सन के बिना एक छोटा कार्यान्वयन है और केवल एक बहुत ही छोटा लूप है नकारात्मक संख्याओं के साथ काम नहीं करता है, लेकिन सभी सकारात्मक 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.2 ई (यह Long.MAX_VALUE है। Long.MAX_VALUE )

मैंने कुछ सचमुच सरल बेंचमार्किंग (10 मिलियन यादृच्छिक लम्बे स्वरूपण) किया और यह एलियाह के कार्यान्वयन की तुलना में काफी तेज है और एथलीयस के कार्यान्वयन से थोड़ा अधिक तेज है।

मेरा: 1137.028 एमएस
एलियास: 2664.396 एमएस
एनालिअस ': 1373.473 एमएस

मेरी टिप्पणी पर सच रहना है कि मैं प्रदर्शन के मुकाबले पठनीयता को मानता हूं, यहां एक ऐसा संस्करण है जहां बिना चिंता किए बगैर, अत्यधिक टिप्पणी (मैं आत्म-दस्तावेजीकरण कोड में विश्वास करता है) के बिना, जो कुछ हो रहा है (यह मानते हुए कि आपने BigDecimal पहले उपयोग किया है) मान लिया जाना चाहिए प्रदर्शन (क्योंकि मैं ऐसी परिदृश्य को चित्रित नहीं कर सकता जहां आप ऐसा कई लाखों बार करना चाहते हैं कि प्रदर्शन भी विचार हो जाता है)।

यह संस्करण:

  • परिशुद्धता के लिए BigDecimal का उपयोग करता है और गोलाई मुद्दों से बचने के लिए
  • ओपी द्वारा अनुरोध के रूप में गोल करने के लिए काम करता है
  • अन्य गोलाई मोडों के लिए काम करता है, जैसे HALF_UP जैसे कि परीक्षण में
  • आपको सटीक समायोजित करने की अनुमति देता है ( REQUIRED_PRECISION बदलें)
  • थ्रेशोल्ड को परिभाषित करने के लिए एक enum का उपयोग करता है, अर्थात् आसानी से के / एमबी / जीबी / टीबी / केबी / एम / बी / टी के उपयोग के लिए समायोजित किया जा सकता है, और जरूर आवश्यक हो सकता है कि TRILLION से परे विस्तार किया जा सकता है
  • पूर्ण इकाई परीक्षण के साथ आता है, क्योंकि प्रश्न में परीक्षण के मामले सीमाओं का परीक्षण नहीं कर रहे थे
  • शून्य और ऋणात्मक संख्याओं के लिए काम करना चाहिए

थ्रेसहोल्ड.जावा :

 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; } } 

(Uncomment the println statements or change to use your favourite logger to see what it's doing.)

And finally, the tests in NumberShortenerTest (plain 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)); } } } 

Feel free to point out in the comments if I missed a significant test case or if expected values should be adjusted.

For anyone that wants to round. This is a great, easy to read solution, that takes advantage of the Java.Lang.Math library

  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); } } 

The following code shows how you can do this with easy expansion in mind.

The "magic" lies mostly in the makeDecimal function which, for the correct values passed in, guarantees you will never have more than four characters in the output.

It first extracts the whole and tenths portions for a given divisor so, for example, 12,345,678 with a divisor of 1,000,000 will give a whole value of 12 and a tenths value of 3 .

From that, it can decide whether it outputs just the whole part or both the whole and tenths part, using the rules:

  • If tenths part is zero, just output whole part and suffix.
  • If tenths part is greater than nine, just output whole part and suffix.
  • Otherwise, output whole part, tenths part and suffix.

The code for that follows:

 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); } 

Then, it's a simple matter of calling that helper function with the correct values, including some constants to make life easier for the developer:

 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"); } 

The fact that the makeDecimal function does the grunt work means that expanding beyond 999,999,999 is just a matter of adding an extra line to Xlat , so easy that I've done it for you.

The final return in Xlat doesn't need a conditional since the largest value you can hold in a 64-bit signed long is only about 9.2 quintillion.

But if, by some bizarre requirement, Oracle decides to add a 128-bit longer type or a 1024-bit damn_long type, you'll be ready for it 🙂


And, finally, a little test harness you can use for validating the functionality.

 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)); } } 

You can see from the output that it gives you what you need:

 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 

And, as an aside, be aware that passing in a negative number to this function will result in a string too long for your requirements, since it follows the < THOU path). I figured that was okay since you only mention non-negative values in the question.

Adding my own answer, Java code, self explanatory code..

 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; } } 
 //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); } 

This code snippet just deadly simple, and clean code, and totally works:

 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); } } 

इसे इस्तेमाल करे :

 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 }