- 1 ১. ভূমিকা
- 2 ২. BigDecimal কী?
- 3 ৩. BigDecimal এর মৌলিক ব্যবহার
- 4 ৪. BigDecimal-এর উন্নত ব্যবহার
- 5 ৫. সাধারণ ত্রুটি এবং সেগুলো ঠিক করার উপায়
- 6 ৬. ব্যবহারিক উদাহরণ
- 7 7. Summary
- 8 8. FAQ: Frequently Asked Questions About BigDecimal
- 8.1 Q1. Why should I use BigDecimal instead of float or double?
- 8.2 Q2. What is the safest way to construct BigDecimal instances?
- 8.3 Q3. Why does divide() throw an exception?
- 8.4 Q4. What’s the difference between compareTo() and equals()?
- 8.5 Q5. How do I perform rounding?
- 8.6 Q6. Can I check decimal digits (scale)?
- 8.7 Q7. How should I handle null/empty input safely?
১. ভূমিকা
জাভাতে সংখ্যাত্মক গণনায় নির্ভুলতা সমস্যাগুলি
জাভা প্রোগ্রামিং-এ সংখ্যাত্মক গণনা দৈনন্দিন কাজের অংশ। উদাহরণস্বরূপ, পণ্যের দাম গণনা, কর বা সুদ নির্ধারণ—এই ধরনের অপারেশন অনেক অ্যাপ্লিকেশনে প্রয়োজনীয়। তবে, যখন এই গণনাগুলি float বা double এর মতো ফ্লোটিং‑পয়েন্ট টাইপ ব্যবহার করে করা হয়, তখন অপ্রত্যাশিত ত্রুটি ঘটতে পারে।
এটি ঘটে কারণ float এবং double মানগুলোকে বাইনারি আনুমানিকতা হিসেবে উপস্থাপন করে। “0.1” বা “0.2” এর মতো দশমিক মানগুলো, যা দশমিকভাবে সঠিকভাবে প্রকাশ করা যায়, বাইনারিতে ঠিকভাবে উপস্থাপন করা যায় না—ফলে ছোট ত্রুটি জমা হয়।
আর্থিক বা নির্ভুল গণনার জন্য BigDecimal অপরিহার্য
এ ধরনের ত্রুটি আর্থিক গণনা ও উচ্চ নির্ভুলতার বৈজ্ঞানিক/প্রকৌশলগত গণনা ক্ষেত্রে গুরুতর হতে পারে। উদাহরণস্বরূপ, বিলিং গণনায় ১ ইয়েনের পার্থক্যও বিশ্বাসযোগ্যতার সমস্যার দিকে নিয়ে যেতে পারে।
এখানেই জাভার BigDecimal ক্লাসের গুরুত্ব প্রকাশ পায়। BigDecimal যেকোনো দশমিক সংখ্যা উচ্চ নির্ভুলতার সাথে পরিচালনা করতে পারে এবং float বা double এর পরিবর্তে ব্যবহার করলে সংখ্যাত্মক গণনা ত্রুটি ছাড়াই করা যায়।
এই প্রবন্ধ থেকে আপনি কী শিখবেন
এই প্রবন্ধে আমরা জাভায় BigDecimal ব্যবহারের মৌলিক বিষয়, উন্নত কৌশল, পাশাপাশি সাধারণ ত্রুটি ও সতর্কতা পদ্ধতিগতভাবে ব্যাখ্যা করব।
এটি সেইসব ডেভেলপারদের জন্য উপকারী, যারা জাভায় আর্থিক গণনা সঠিকভাবে করতে চান অথবা তাদের প্রকল্পে BigDecimal গ্রহণের কথা ভাবছেন।
২. BigDecimal কী?
BigDecimal এর সংক্ষিপ্ত বিবরণ
BigDecimal হল জাভার একটি ক্লাস যা উচ্চ নির্ভুলতার দশমিক গাণিতিক সক্ষম করে। এটি java.math প্যাকেজের অংশ এবং বিশেষভাবে ত্রুটির সহনশীলতা না থাকা গণনা, যেমন আর্থিক/হিসাব/কর গণনার জন্য ডিজাইন করা হয়েছে।
জাভার float এবং double দিয়ে সংখ্যাগুলোকে বাইনারি আনুমানিকতা হিসেবে সংরক্ষণ করা হয়—অর্থাৎ “0.1” বা “0.2” এর মতো দশমিকগুলো ঠিকভাবে উপস্থাপন করা যায় না, যা ত্রুটির মূল কারণ। এর বিপরীতে, BigDecimal মানগুলোকে স্ট্রিং‑ভিত্তিক দশমিক উপস্থাপনা হিসেবে সংরক্ষণ করে, ফলে রাউন্ডিং ও আনুমানিকতার ত্রুটি দমন করা যায়।
যেকোনো নির্ভুলতার সংখ্যা পরিচালনা
BigDecimal এর সবচেয়ে বড় বৈশিষ্ট্য হল “যেকোনো নির্ভুলতা”। পূর্ণসংখ্যা ও দশমিক উভয় অংশই তাত্ত্বিকভাবে প্রায় অসীম সংখ্যক অঙ্ক পরিচালনা করতে পারে, ফলে অঙ্কের সীমাবদ্ধতার কারণে রাউন্ডিং বা অঙ্ক হারানোর সমস্যা হয় না।
উদাহরণস্বরূপ, নিম্নলিখিত বড় সংখ্যা সঠিকভাবে পরিচালনা করা যায়:
BigDecimal bigValue = new BigDecimal("12345678901234567890.12345678901234567890");
এভাবে নির্ভুলতা বজায় রেখে গাণিতিক ক্রিয়াকলাপ করা BigDecimal‑এর প্রধান শক্তি।
প্রধান ব্যবহার ক্ষেত্রসমূহ
BigDecimal নিম্নলিখিত পরিস্থিতিতে ব্যবহার করা সুপারিশ করা হয়:
- আর্থিক গণনা — আর্থিক অ্যাপ্লিকেশনে সুদ, কর হার ইত্যাদি গণনা
- ইনভয়েস / কোটেশন পরিমাণ প্রক্রিয়াকরণ
- উচ্চ নির্ভুলতা প্রয়োজনীয় বৈজ্ঞানিক/প্রকৌশলগত গণনা
- দীর্ঘমেয়াদী সঞ্চয়ে ত্রুটি জমা হওয়া প্রক্রিয়া
উদাহরণস্বরূপ, হিসাব ব্যবস্থা ও বেতন গণনা-এ ১ ইয়েনের পার্থক্য বড় ক্ষতি বা বিরোধের কারণ হতে পারে; তাই BigDecimal‑এর নির্ভুলতা অপরিহার্য।
৩. BigDecimal এর মৌলিক ব্যবহার
BigDecimal ইনস্ট্যান্স কীভাবে তৈরি করবেন
সাধারণ সংখ্যামূলক লিটারালগুলোর বিপরীতে, BigDecimal সাধারণত স্ট্রিং থেকে তৈরি করা উচিত। কারণ double বা float থেকে তৈরি করা মানগুলো ইতিমধ্যে বাইনারি আনুমানিকতার ত্রুটি বহন করতে পারে।
প্রস্তাবিত (স্ট্রিং থেকে তৈরি):
BigDecimal value = new BigDecimal("0.1");
এড়িয়ে চলুন (double থেকে তৈরি):
BigDecimal value = new BigDecimal(0.1); // may contain error
গাণিতিক ক্রিয়াকলাপ কীভাবে সম্পাদন করবেন
BigDecimal সাধারণ গাণিতিক অপারেটর (+, -, *, /) দিয়ে ব্যবহার করা যায় না। পরিবর্তে, নির্দিষ্ট মেথড ব্যবহার করতে হয়।
যোগ (add)
BigDecimal a = new BigDecimal("10.5");
BigDecimal b = new BigDecimal("2.3");
BigDecimal result = a.add(b); // 12.8
বিয়োগ (subtract)
BigDecimal result = a.subtract(b); // 8.2
গুণন (multiply)
BigDecimal result = a.multiply(b); // 24.15
ভাগ (divide) এবং গোলাকার মোড
ভাগ করার ক্ষেত্রে সতর্কতা প্রয়োজন। যদি সমানভাবে বিভাজ্য না হয়, তাহলে ArithmeticException ঘটবে যদি না গোলাকার মোড নির্দিষ্ট করা হয়।
BigDecimal a = new BigDecimal("10");
BigDecimal b = new BigDecimal("3");
BigDecimal result = a.divide(b, 2, RoundingMode.HALF_UP); // 3.33
এখানে আমরা “২ দশমিক স্থান” এবং “অর্ধেক উপরে গোলাকার” নির্দিষ্ট করছি।
স্কেল এবং গোলাকার মোড সেট করা setScale দিয়ে
setScale নির্দিষ্ট সংখ্যক অঙ্কে গোলাকার করার জন্য ব্যবহার করা যায়।
BigDecimal value = new Big BigDecimal("123.456789");
BigDecimal rounded = value.setScale(2, RoundingMode.HALF_UP); // 123.46
সাধারণ RoundingMode মানগুলো:
| Mode Name | Description |
|---|---|
HALF_UP | Round half up (standard rounding) |
HALF_DOWN | Round half down |
HALF_EVEN | Banker’s rounding |
UP | Always round up |
DOWN | Always round down |
BigDecimal অপরিবর্তনীয়
BigDecimal অপরিবর্তনীয়। অর্থাৎ — অঙ্কগণিত পদ্ধতিগুলো (add, subtract ইত্যাদি) মূল মান পরিবর্তন করে না — এগুলো একটি নতুন ইনস্ট্যান্স ফেরত দেয়।
BigDecimal original = new BigDecimal("5.0");
BigDecimal result = original.add(new BigDecimal("1.0"));
System.out.println(original); // still 5.0
System.out.println(result); // 6.0
৪. BigDecimal-এর উন্নত ব্যবহার
মান তুলনা করা: compareTo এবং equals-এর মধ্যে পার্থক্য
BigDecimal-এ মান তুলনা করার দুটি উপায় রয়েছে: compareTo() এবং equals() , এবং এগুলো ভিন্নভাবে কাজ করে।
compareTo()শুধুমাত্র সংখ্যাগত মান তুলনা করে (স্কেল উপেক্ষা করে)।equals()স্কেলসহ তুলনা করে (দশমিক অঙ্কের সংখ্যা)।BigDecimal a = new BigDecimal("10.0"); BigDecimal b = new BigDecimal("10.00"); System.out.println(a.compareTo(b)); // 0 (values are equal) System.out.println(a.equals(b)); // false (scale differs)
পয়েন্ট: সংখ্যাগত সমতার চেকের জন্য — যেমন অর্থের সমতা — compareTo() সাধারণত সুপারিশ করা হয়।
String থেকে/প্রতি রূপান্তর
ব্যবহারকারীর ইনপুট এবং বাহ্যিক ফাইল আমদানিতে, String টাইপের সাথে রূপান্তর সাধারণ।
String → BigDecimal
BigDecimal value = new Big BigDecimal("1234.56");
BigDecimal → String
String str = value.toString(); // "1234.56"
valueOf ব্যবহার করা
জাভাতেও BigDecimal.valueOf(double val) রয়েছে, কিন্তু এটিও অভ্যন্তরীণভাবে double-এর ত্রুটি ধারণ করে, তাই স্ট্রিং থেকে নির্মাণ করা এখনও নিরাপদতর।
BigDecimal unsafe = BigDecimal.valueOf(0.1); // contains internal error
MathContext-এর মাধ্যমে নির্ভুলতা এবং গোলাকার নিয়ম
MathContext একসাথে নির্ভুলতা এবং গোলাকার মোড নিয়ন্ত্রণ করতে দেয় — অনেক অপারেশনে সাধারণ নিয়ম প্রয়োগ করার সময় উপযোগী।
MathContext mc = new MathContext(4, RoundingMode.HALF_UP);
BigDecimal result = new BigDecimal("123.4567").round(mc); // 123.5
অঙ্কগণিতে এটি ব্যবহারযোগ্য:
BigDecimal a = new BigDecimal("10.456");
BigDecimal b = new BigDecimal("2.1");
BigDecimal result = a.multiply(b, mc); // 4-digit precision
null চেক এবং নিরাপদ ইনিশিয়ালাইজেশন
ফর্মগুলো null বা খালি মান পাঠাতে পারে — গার্ড কোড স্ট্যান্ডার্ড।
String input = ""; // empty
BigDecimal value = (input == null || input.isEmpty()) ? BigDecimal.ZERO : new BigDecimal(input);
BigDecimal-এর স্কেল চেক করা
দশমিক অঙ্ক জানার জন্য, scale() ব্যবহার করুন:
BigDecimal value = new BigDecimal("123.45");
System.out.println(value.scale()); // 3
৫. সাধারণ ত্রুটি এবং সেগুলো ঠিক করার উপায়
ArithmeticException: অসীম দশমিক বিস্তার
ত্রুটির উদাহরণ:
BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("3");
BigDecimal result = a.divide(b); // exception
এটি “১ ÷ ৩” — যেহেতু এটি অসীম দশমিক হয়ে যায়, যদি কোনো গোলাকার মোড/স্কেল দেওয়া না হয়, তাহলে একটি এক্সেপশন ফেলা হয়।
ঠিক করুন: স্কেল + গোলাকার মোড নির্দিষ্ট করুন
BigDecimal result = a.divide(b, 2, RoundingMode.HALF_UP); // OK (3.33)
double থেকে সরাসরি নির্মাণ করার সময় ত্রুটি
Passing a double directly may contain binary error already — producing unexpected values.
Bad Example:
BigDecimal val = new BigDecimal(0.1);
System.out.println(val); // 0.100000000000000005551115123...
Correct: Use a String
BigDecimal val = new BigDecimal("0.1"); // exact 0.1
Note: BigDecimal.valueOf(0.1) uses Double.toString() internally, so it is “almost same” as new BigDecimal("0.1") — but string is 100% safest.

স্কেল মিসম্যাচের কারণে equals ভুল বোঝা
equals() স্কেল তুলনা করে, তাই মানগুলো সংখ্যাগতভাবে সমান হলেও এটি false রিটার্ন করতে পারে।
BigDecimal a = new BigDecimal("10.0");
BigDecimal b = new BigDecimal("10.00");
System.out.println(a.equals(b)); // false
Solution: use compareTo() for numeric equality
System.out.println(a.compareTo(b)); // 0
অপর্যাপ্ত নির্ভুলতার কারণে অপ্রত্যাশিত ফলাফল
setScale ব্যবহার করার সময় রাউন্ডিং মোড নির্দিষ্ট না করলে — এক্সসেপশন ঘটতে পারে।
Bad Example:
BigDecimal value = new BigDecimal("1.2567");
BigDecimal rounded = value.setScale(2); // exception
Solution:
BigDecimal rounded = value.setScale(2, RoundingMode.HALF_UP); // OK
ইনপুট মান অবৈধ হলে NumberFormatException
যদি এমন অবৈধ টেক্সট পাস করা হয় যা সংখ্যা হিসেবে পার্স করা যায় না (যেমন, ব্যবহারকারীর ইনপুট / CSV ফিল্ড), তবে NumberFormatException ঘটবে।
Solution: use exception handling
try {
BigDecimal value = new BigDecimal(userInput);
} catch (NumberFormatException e) {
// show error message or fallback logic
}
৬. ব্যবহারিক উদাহরণ
এখানে আমরা বাস্তবিক দৃশ্যপট উপস্থাপন করছি যা দেখায় কীভাবে BigDecimal ব্যবহারিকভাবে ব্যবহার করা যায়। বিশেষ করে আর্থিক/হিসাব/কর গণনায়, সঠিক সংখ্যাগত হ্যান্ডলিংয়ের গুরুত্ব স্পষ্ট হয়।
মূল্য গণনায় দশমিক হ্যান্ডলিং (ভগ্নাংশ রাউন্ডিং)
উদাহরণ: ১০% ভোগ করসহ মূল্য গণনা
BigDecimal price = new BigDecimal("980"); // price w/o tax
BigDecimal taxRate = new BigDecimal("0.10");
BigDecimal tax = price.multiply(taxRate).setScale(0, RoundingMode.HALF_UP);
BigDecimal total = price.add(tax);
System.out.println("Tax: " + tax); // Tax: 98
System.out.println("Total: " + total); // Total: 1078
বিন্দুসমূহ:
- কর গণনার ফলাফল প্রায়শই পূর্ণসংখ্যা হিসেবে প্রক্রিয়াকৃত হয়, রাউন্ড করার জন্য
setScale(0, RoundingMode.HALF_UP)ব্যবহার করা হয়। doubleত্রুটি তৈরি করতে পারে —BigDecimalব্যবহার করার পরামর্শ দেওয়া হয়।
ছাড় গণনা (% OFF)
উদাহরণ: ২০% ছাড়
BigDecimal originalPrice = new BigDecimal("3500");
BigDecimal discountRate = new BigDecimal("0.20");
BigDecimal discount = originalPrice.multiply(discountRate).setScale(0, RoundingMode.HALF_UP);
BigDecimal discountedPrice = originalPrice.subtract(discount);
System.out.println("Discount: " + discount); // Discount: 700
System.out.println("After discount: " + discountedPrice); // 2800
বিন্দু: মূল্য ছাড়ের গণনা নির্ভুলতা হারানো উচিত নয়।
একক মূল্য × পরিমাণ গণনা (সাধারণ ব্যবসায়িক অ্যাপ দৃশ্যপট)
উদাহরণ: 298.5 ইয়েন × ৭টি আইটেম
BigDecimal unitPrice = new BigDecimal("298.5");
BigDecimal quantity = new BigDecimal("7");
BigDecimal total = unitPrice.multiply(quantity).setScale(2, RoundingMode.HALF_UP);
System.out.println("Total: " + total); // 2089.50
বিন্দুসমূহ:
- ভগ্নাংশ গুণনের জন্য রাউন্ডিং সামঞ্জস্য করুন।
- হিসাব / অর্ডার সিস্টেমের জন্য গুরুত্বপূর্ণ।
যৌগিক সুদ গণনা (আর্থিক উদাহরণ)
উদাহরণ: ৩% বার্ষিক সুদ × ৫ বছর
BigDecimal principal = new BigDecimal("1000000"); // base: 1,000,000
BigDecimal rate = new BigDecimal("0.03");
int years = 5;
BigDecimal finalAmount = principal;
for (int i = 0; i < years; i++) {
finalAmount = finalAmount.multiply(rate.add(BigDecimal.ONE)).setScale(2, RoundingMode.HALF_UP);
}
System.out.println("After 5 years: " + finalAmount); // approx 1,159,274.41
Point:
- Repeated calculations accumulate errors — BigDecimal avoids this.
Validation & Conversion of User Input
public static BigDecimal parseAmount(String input) {
try {
return new BigDecimal(input).setScale(2, RoundingMode.HALF_UP);
} catch (NumberFormatException e) {
return BigDecimal.ZERO; // treat invalid input as 0
}
}
Points:
- Safely convert user-provided numeric strings.
- Validation + error fallback improves robustness.
7. Summary
The Role of BigDecimal
In Java’s numeric processing — especially monetary or precision-required logic — the BigDecimal class is indispensable. Errors inherent in float / double can be dramatically avoided by using BigDecimal.
This article covered fundamentals, arithmetic, comparisons, rounding, error handling, and real-world examples.
Key Review Points
BigDecimalhandles arbitrary-precision decimal — ideal for money and precision math- Initialization should be via string literal , e.g.
new BigDecimal("0.1") - Use
add(),subtract(),multiply(),divide(), and always specify rounding mode when dividing - Use
compareTo()for equality — understand difference vsequals() setScale()/MathContextlet you finely control scale + rounding- Real business logic cases include money, tax, quantity × unit price etc.
For Those About to Use BigDecimal
Although “handling numbers in Java” looks simple — precision / rounding / numeric error problems always exist behind it. BigDecimal is a tool that directly addresses those problems — mastering it lets you write more reliable code.
At first you may struggle with rounding modes — but with real project usage, it becomes natural.
Next chapter is an FAQ section summarizing common questions about BigDecimal — useful for review and specific semantic searches.
8. FAQ: Frequently Asked Questions About BigDecimal
Q1. Why should I use BigDecimal instead of float or double?
A1.
Because float/double represent numbers as binary approximations — decimal fractions cannot be represented exactly. This causes results such as “0.1 + 0.2 ≠ 0.3.”
BigDecimal preserves decimal values exactly — ideal for money or precision-critical logic.
Q2. What is the safest way to construct BigDecimal instances?
A2.
Always construct from string.
Bad (error):
new BigDecimal(0.1)
Correct:
new BigDecimal("0.1")
BigDecimal.valueOf(0.1) uses Double.toString() internally, so it’s almost same — but string is the safest.
Q3. Why does divide() throw an exception?
A3.
Because BigDecimal.divide() throws ArithmeticException when result is a non-terminating decimal.
Solution: specify scale + rounding mode
BigDecimal result = a.divide(b, 2, RoundingMode.HALF_UP);
Q4. What’s the difference between compareTo() and equals()?
A4.
compareTo()checks numeric equality (scale ignored)equals()checks exact equality including scalenew BigDecimal("10.0").compareTo(new BigDecimal("10.00")); // → 0 new BigDecimal("10.0").equals(new BigDecimal("10.00")); // → false
Q5. How do I perform rounding?
A5.
Use setScale() with explicit rounding mode.
BigDecimal value = new BigDecimal("123.4567");
BigDecimal rounded = value.setScale(2, RoundingMode.HALF_UP); // 123.46
Main rounding modes:
RoundingMode.HALF_UP(round half up)RoundingMode.DOWN(round down)RoundingMode.UP(round up)
Q6. Can I check decimal digits (scale)?
A6.
Yes — use scale().
BigDecimal val = new BigDecimal("123.45");
System.out.println(val.scale()); // → 3
Q7. How should I handle null/empty input safely?
A7.
সর্বদা null চেক + এক্সসেপশন হ্যান্ডলিং অন্তর্ভুক্ত করুন।
public static BigDecimal parseSafe(String input) {
if (input == null || input.trim().isEmpty()) return BigDecimal.ZERO;
try {
return new BigDecimal(input.trim());
} catch (NumberFormatException e) {
return BigDecimal.ZERO;
}
}

