103 — Minimum Coins Basic¶
এই level-এর প্রথম problem, আর এটা শুরুই হচ্ছে একটা ধাক্কা দিয়ে — greedy এখানে কখনো ঠিক, কখনো ডাহা ভুল! ভয় পেও না, বরং উপভোগ করো; কারণ greedy-র fail case নিজের চোখে দেখলে বাকি 9টা problem-এ তুমি অনেক সাবধান চোখ নিয়ে ঢুকবে। ধীরে পড়ো, {1, 3, 4}-এর গল্পটা যেন গেঁথে যায়।
Source¶
- Original source: Classic exercise (greedy fails on {1, 3, 4} — নিজে দেখাও)
- Platform: Classic exercise / Coin change-এর পরিচিত শিক্ষামূলক রূপ —
— - Topic: Greedy math tricks → canonical coin system, counterexample শিকার
- Difficulty: Easy
- Pattern: canonical greedy (বড় coin আগে নাও — কিন্তু প্রমাণসহ)
- Basic idea inherited from: — (greedy level-এর শিকড়, এটাই প্রথম)
1. Problem in simple words¶
তোমার কাছে কয়েক রকম মূল্যের coin আছে — যেমন {1, 2, 5, 10} টাকা। প্রতিটা coin তুমি যতবার খুশি ব্যবহার করতে পারো (unlimited supply)। একটা নির্দিষ্ট amount (যেমন 27 টাকা) ঠিক বানাতে হবে — প্রশ্ন: সবচেয়ে কম কয়টা coin লাগবে?
দুটো ছোট কথা মনে রাখো:
- প্রতিটা coin বারবার নেওয়া যায়
- কখনো amount বানানোই অসম্ভব হতে পারে (যেমন শুধু
{2}দিয়ে 3 বানানো যায় না) — তখন উত্তর "অসম্ভব"
দেখতে সহজ, কিন্তু এর ভেতরেই লুকিয়ে আছে greedy-র সবচেয়ে বিখ্যাত ফাঁদ।
2. What is the math idea?¶
মূল প্রশ্ন: locally লোভী choice (সবচেয়ে বড় coin আগে নাও) কি সবসময় globally সেরা?
উত্তর: না — এটা নির্ভর করে coin set-এর গঠনের উপর।
- Canonical system (যেমন আসল টাকার নোট: 1, 2, 5, 10, 20, 50, 100) — এখানে greedy ঠিক উত্তর দেয়
- Arbitrary system (যেমন
{1, 3, 4}) — এখানে greedy ভুল করতে পারে; তখন DP লাগে (যেটা পরে level 12-এ আসবে)
তাই এই problem আসলে একটা greedy algorithm আর একটা সাবধানবাণী — দুটো একসাথে।
3. Which basic idea does this inherit from?¶
এটা greedy level-এর একদম প্রথম problem, তাই আগের কোনো greedy problem-এর কাঁধে দাঁড়ায় না (inherited from: —)।
বরং এটাই বাকি সব greedy-র জন্য সবচেয়ে জরুরি অভ্যাসটা শেখায়: greedy idea পেলে আগে counterexample খোঁজো। concept-notes-এ যে নিয়ম পড়েছ — "5 মিনিট counterexample খুঁজে না পেলে তবে exchange argument দিয়ে প্রমাণের চেষ্টা" — তার প্রথম হাতে-কলমে প্রয়োগ এখানেই। % (modulo) আর integer division // — এই দুই tool level 0 থেকেই চেনা, সেগুলোই এখানে greedy-র হাতিয়ার।
4. Real-life analogy¶
ভাবো তুমি দোকানে খুচরো ফেরত দিচ্ছ। স্বাভাবিকভাবেই তুমি সবচেয়ে বড় নোটটা আগে দাও — 27 টাকা দিতে হলে আগে 20, তারপর 5, তারপর 2। এই "বড়টা আগে" সহজাত অভ্যাসটাই greedy।
আসল টাকার সিস্টেমে এটা কাজ করে বলেই আমরা রোজ এভাবে খুচরো দিই। কিন্তু কল্পনা করো এক অদ্ভুত দেশ যেখানে শুধু {1, 3, 4} টাকার coin আছে। সেখানে 6 টাকা দিতে গিয়ে "বড়টা আগে" করলে তুমি 4 + 1 + 1 = 3টা coin দেবে — অথচ 3 + 3 = 2টা coin-এই কাজ হতো! সেই দেশে তোমার অভ্যাস তোমাকে ঠকাবে।
5. Visual explanation¶
আগে canonical system-এ greedy কেমন সুন্দর কাজ করে দেখো:
amount = 27, coins = {10, 5, 2, 1} (বড় থেকে ছোট)
27 │ 10 নাও → বাকি 17 (count = 1)
17 │ 10 নাও → বাকি 7 (count = 2)
7 │ 5 নাও → বাকি 2 (count = 3)
2 │ 2 নাও → বাকি 0 (count = 4) ✓
মোট 4টা coin
এবার সেই কুখ্যাত fail case — {1, 3, 4} দিয়ে 6:
greedy: 6 → 4 নাও (বাকি 2) → 1 (বাকি 1) → 1 (বাকি 0) = 3টা coin
optimal: 6 → 3 নাও (বাকি 3) → 3 (বাকি 0) = 2টা coin ✓
বড় coin (4) নেওয়াটা লোভনীয় ছিল,
কিন্তু বাকি 2 গড়তে দুটো 1 লাগল — ভবিষ্যৎ নষ্ট হলো!
লক্ষ করো — greedy কোনো ভুল হিসাব করেনি, প্রতি step-এ নিয়ম মেনেই বড়টা নিয়েছে। তবু হেরে গেল। এটাই greedy-র বিপদ।
6. Example input and output¶
amount, coins -> output (ব্যাখ্যা)
---------------------------------------------------------
27, {10, 5, 2, 1} -> 4 (10+10+5+2)
6, {1, 2, 5} -> 2 (5+1) canonical, greedy ঠিক
0, {1, 5} -> 0 (কিছুই লাগে না)
3, {2} -> -1 (অসম্ভব)
6, {1, 3, 4} -> greedy দেয় 3, কিন্তু সত্যিকারের min = 2 ✗
শেষ line-টাই এই note-এর হৃদয় — canonical system-এ আমরা greedy-র উত্তরে বিশ্বাস করি, arbitrary system-এ করি না।
7. Brute force thinking¶
"সত্যিকারের" minimum (যেকোনো coin set-এর জন্য সঠিক) বের করতে সবচেয়ে সরল চিন্তা — সব সম্ভাবনা চেষ্টা করো। প্রতিটা coin বেছে নাও, বাকি amount-এর জন্য আবার একই প্রশ্ন করো (recursion), আর সবচেয়ে কম coin-এর path টা রাখো:
def min_coins_brute(amount, coins):
if amount == 0:
return 0
best = float("inf")
for c in coins:
if c <= amount:
sub = min_coins_brute(amount - c, coins)
if sub + 1 < best:
best = sub + 1
return best
এটা সব coin set-এ ঠিক উত্তর দেয় — কারণ এটা কোনো শর্টকাট নেয় না, প্রতিটা পথ দেখে। তাই আমরা এটাকে "সত্য" হিসেবে ব্যবহার করে greedy-কে যাচাই করতে পারব।
8. Why brute force may be slow¶
এই recursion প্রতিটা step-এ প্রতিটা coin চেষ্টা করে, তাই একই sub-amount বারবার গণনা হয় — গাছটা exponential-ভাবে ফাঁপে। amount একটু বড় হলেই (যেমন 60-70) এটা অসহনীয় ধীর হয়ে যায়।
amount = 30, coins = {1, 3, 4} হলে recursion tree বিশাল —
একই amount (যেমন 10) বহুবার আলাদা পথে গণনা হয়।
brute force : exponential (ধীর) — কিন্তু সব set-এ সঠিক
greedy : O(coins) প্রায় তাৎক্ষণিক — কিন্তু শুধু canonical-এ সঠিক
DP : O(amount × coins) — সব set-এ সঠিক ও দ্রুত (পরে level 12)
মাঝখানের সমাধান DP — কিন্তু সেটা পরের গল্প। এখন আমরা greedy আর তার সীমা বুঝছি।
9. Better thinking¶
Canonical system ধরে নিলে greedy অসাধারণ দ্রুত: coins বড় থেকে ছোট sort করো; প্রতিটা coin যতবার সম্ভব নাও (amount // c বার), তারপর বাকি অংশ (amount % c) নিয়ে এগোও।
def min_coins_greedy(amount, coins):
coins = sorted(coins, reverse=True)
count = 0
for c in coins:
count += amount // c
amount %= c
return count if amount == 0 else -1
Greedy choice: প্রতি step-এ amount-এর চেয়ে ছোট/সমান সবচেয়ে বড় coin-টা নাও।
কেন কাজ করে (canonical-এ, এক লাইন): canonical system-এ সবচেয়ে বড় coin c না নিলে, তার জায়গা ছোট coin দিয়ে ভরাট করতে হয় — আর সেই ছোট coin গুলোকে সবসময় কম সংখ্যায় একটা c দিয়ে বদলে (exchange) দেওয়া যায়, তাই বড়টা নেওয়া কখনো খারাপ হয় না। (Arbitrary system-এ ঠিক এই exchange-টাই ভেঙে পড়ে — {1, 3, 4}-এ দুটো 1-কে এক 4 দিয়ে বদলালে বাকি অংশ গড়া কঠিন হয়ে যায়।)
10. Thinking tweak¶
আসল "আহা!" এখানে উল্টো — শেখার জিনিসটা code নয়, সন্দেহ:
"Greedy লিখলাম মানেই হয়ে গেল না — coin set টা canonical তো? একটা ছোট counterexample চালিয়ে দেখি।"
{1, 3, 4} দিয়ে 6 — এই এক লাইনের পরীক্ষাটাই brute force-এর উত্তরের সাথে greedy মিলিয়ে দেখায় ওরা আলাদা। এই "আগে counterexample খুঁজি" reflex-টাই পুরো greedy level-এর সবচেয়ে দামি tweak। code-এ আমরা তাই greedy আর brute দুটোই রেখেছি, আর assert দিয়ে দেখিয়েছি কোথায় মেলে, কোথায় মেলে না।
11. Step-by-step dry run¶
{1, 3, 4} দিয়ে amount = 6 — greedy কীভাবে এগোয়:
| step | বর্তমান amount | বাছাই coin | amount // c |
নতুন amount (% c) |
মোট count |
|---|---|---|---|---|---|
| 1 | 6 | 4 | 1 | 6 % 4 = 2 | 1 |
| 2 | 2 | 3 | 0 | 2 % 3 = 2 | 1 |
| 3 | 2 | 1 | 2 | 2 % 1 = 0 | 3 |
greedy থামল count = 3-এ। এবার brute force (min_coins_brute(6, [1,3,4])):
6 = 3 + 3 → 2টা coin ✓ (সত্যিকারের minimum)
greedy দিল 3, brute দিল 2 → মেলে না ⇒ এই set canonical নয়!
একই amount, দুই উত্তর — চোখের সামনে greedy-র সীমা ধরা পড়ল।
12. Python solution¶
def min_coins_greedy(amount, coins):
"""বড় coin আগে — canonical system-এ সঠিক, নাহলে ভুল হতে পারে।"""
coins = sorted(coins, reverse=True)
count = 0
for c in coins:
count += amount // c
amount %= c
return count if amount == 0 else -1
def min_coins_brute(amount, coins):
"""সব পথ দেখে সত্যিকারের minimum — সব set-এ সঠিক, কিন্তু ধীর।"""
if amount == 0:
return 0
best = float("inf")
for c in coins:
if c <= amount:
sub = min_coins_brute(amount - c, coins)
if sub + 1 < best:
best = sub + 1
return best
# --- ছোট test (সব pass করা উচিত) ---
# canonical system-এ greedy == সত্যিকারের minimum
assert min_coins_greedy(27, [10, 5, 2, 1]) == 4 # 10+10+5+2
assert min_coins_greedy(6, [5, 2, 1]) == 2 # 5+1
assert min_coins_greedy(0, [1, 5]) == 0 # কিছুই লাগে না
assert min_coins_greedy(3, [2]) == -1 # অসম্ভব
# canonical-এ greedy আর brute একমত (ছোট amount-এ যাচাই, brute ধীর বলে)
for amt in range(0, 21):
assert min_coins_greedy(amt, [10, 5, 2, 1]) == min_coins_brute(amt, [10, 5, 2, 1])
# কুখ্যাত fail case: {1, 3, 4} দিয়ে 6
assert min_coins_greedy(6, [1, 3, 4]) == 3 # greedy ভুল করে 3 দেয়
assert min_coins_brute(6, [1, 3, 4]) == 2 # সত্যিকারের minimum 2
assert min_coins_greedy(6, [1, 3, 4]) != min_coins_brute(6, [1, 3, 4]) # মেলে না!
print("canonical 27:", min_coins_greedy(27, [10, 5, 2, 1])) # 4
print("{1,3,4} greedy 6:", min_coins_greedy(6, [1, 3, 4])) # 3
print("{1,3,4} true 6:", min_coins_brute(6, [1, 3, 4])) # 2
print("সব test pass!")
13. Line-by-line explanation¶
greedy-র প্রথম শর্ত — বড় coin আগে। তাই descending sort।
এই দুই লাইনই greedy-র হৃদয়। amount // c বলে দেয় এই coin সর্বোচ্চ কতবার নেওয়া যায়, আর amount %= c বাকি অংশ রাখে পরের (ছোট) coin-এর জন্য। এক loop-এই সব coin শেষ।
সব coin শেষেও amount বাকি থাকলে এই set দিয়ে বানানো অসম্ভব → -1।
min_coins_brute হলো আমাদের "সত্যের পরিমাপক" — প্রতিটা coin চেষ্টা করে recursion-এ বাকিটা সমাধান করে সবচেয়ে কম path রাখে। ধীর, কিন্তু সব set-এ নির্ভুল, তাই greedy-কে যাচাই করতে আদর্শ।
বাকি assert লাইনগুলো নিজেরাই চেক করছে: canonical-এ greedy আর brute সমান (range(0, 40) লুপে সব মিলিয়ে দেখা), আর {1, 3, 4}-এ ওরা আলাদা — এই অমিলটাই আমরা assert দিয়ে প্রমাণ করছি।
14. Output walkthrough¶
চালালে ছাপবে:
প্রথম line — canonical টাকায় greedy 27-কে 4 coin-এ ভাঙল (সঠিক)। পরের দুই line — একই amount 6-এ greedy দিল 3, কিন্তু সত্যিকারের minimum 2। সব assert চুপচাপ pass করেছে (assert fail করলেই কেবল চিৎকার করত), তাই শেষে সব test pass!।
15. Time complexity¶
- Greedy: O(k log k) — k = coin সংখ্যা, মূল খরচ sorting; তারপর এক pass O(k)। amount-এর আকারের সাথে কাজ বাড়ে না।
- Brute force: exponential — প্রতিটা step-এ k-টা শাখা, গভীরতা amount পর্যন্ত; তাই amount বাড়লে দ্রুত অসহনীয়।
16. Space complexity¶
- Greedy: O(1) বাড়তি (sorting বাদ দিলে) — শুধু গুটিকয় variable।
- Brute force: O(amount) — recursion stack-এর গভীরতা amount পর্যন্ত যেতে পারে।
17. Common mistakes¶
- প্রমাণ ছাড়া greedy বিশ্বাস করা — সবচেয়ে বড় ভুল। coin set canonical কিনা না দেখে greedy চালালে arbitrary set-এ ভুল উত্তর আসবে। আগে অন্তত একটা ছোট counterexample খোঁজো।
{1, 3, 4}counterexample ভুলে যাওয়া — interview-তে "greedy কেন সবসময় কাজ করে না" প্রশ্নের one-shot উত্তর এটাই; না দেখে লিখতে পারা চাই।- Coin sort না করা — descending sort না করলে greedy logic-ই ভেঙে পড়ে; বড়টা আগে না নিলে এটা greedy-ই নয়।
- অসম্ভব case না সামলানো — amount বানানো না গেলে -1 (বা "অসম্ভব") ফেরত দিতে ভুলো না; নাহলে ভুল count চলে যাবে।
- DP-র সাথে গুলিয়ে ফেলা — মনে রেখো: arbitrary set-এ সঠিক minimum চাইলে greedy নয়, DP লাগে — সেটা পরে level 12-এ।
18. Harder problems that inherit this idea¶
- LeetCode — Coin Change (https://leetcode.com/problems/coin-change/) — যেখানে greedy ভেঙে পড়ে, সেখানে DP দিয়ে যেকোনো coin set-এ সঠিক minimum বের করা।
- LeetCode — Coin Change II (https://leetcode.com/problems/coin-change-ii/) — কত ভাবে amount বানানো যায়, সেই counting DP।
- LeetCode — Lemonade Change (https://leetcode.com/problems/lemonade-change/) — এখানে greedy আবার ঠিকঠাক কাজ করে (canonical-ঘেঁষা change-making), তুলনা করার ভালো সঙ্গী।
19. Pattern learned¶
Canonical greedy with counterexample check — বড় coin আগে নাও, কিন্তু আগে নিশ্চিত হও coin set canonical কিনা; না হলে greedy ভুল করে, DP লাগে। বড় শিক্ষা: greedy লিখেই থেমো না, একটা ছোট counterexample চালিয়ে যাচাই করো।
20. Final summary¶
ভবিষ্যতের আমাকে এক লাইনে: "min coins = বড় coin আগে (greedy) — কিন্তু শুধু canonical system-এ সঠিক; {1, 3, 4} দিয়ে 6-এ greedy দেয় 3, সত্যি 2। greedy দেখলেই প্রথম প্রশ্ন: counterexample আছে কি?"
পরের ধাপ → 104 — Jump Game (এবার greedy যেখানে প্রমাণসহ কাজ করে — furthest reach)।
ফিরে দেখা: এই level-এর map → ../README.md · concept ঝালাই → ../concept-notes.md · note-টা যে ছকে লেখা → ../../../templates/math-problem-note-template.md।
মনে রাখার নিয়ম (legal): সব নিজের ভাষায় লেখা; কোনো problem statement copy করা হয়নি; official link শুধু attribution-এর জন্য; "asked by Google/Amazon" এমন দাবি নয় — বরং "common interview pattern" / "Google-style pattern"।