007 — Digital Root¶
002-এ একবার digit যোগ করেছিলাম। আজ সেটা বারবার করব — যোগফল এক digit না হওয়া পর্যন্ত। 9875 → 29 → 11 → 2। এটুকু হলো সহজ অংশ। কিন্তু এই problem-এর আসল রত্ন হলো একটা চমকে দেওয়া shortcut —
1 + (n - 1) % 9— যেটা পুরো loop-টাকে এক লাইনে নামিয়ে দেয়।% 9-এর এই জাদু বুঝে গেলে modular arithmetic-এর জগতে তোমার প্রথম পা পড়ে যাবে। ধীরে যাও, dry run-টা নিজে করো।
Source¶
- Original source: Classic exercise (related practice: LeetCode — Add Digits — https://leetcode.com/problems/add-digits/)
- Platform: Classic exercise / LeetCode (related)
- Topic: Math-based programming fundamentals → Level 0: Absolute Basics
- Difficulty: Easy
- Pattern: repeated digit sum, mod 9
- Basic idea inherited from: 002 (Sum of Digits)
1. Problem in simple words¶
একটা non-negative integer n দেওয়া আছে। তার digital root বের করতে হবে।
নিয়ম: সব digit যোগ করো। যোগফল যদি এক-digit হয় (0-9), সেটাই উত্তর। নাহলে সেই যোগফলের digit আবার যোগ করো — এভাবে চলতে থাকো যতক্ষণ না এক digit থাকে।
যেমন 9875:
2. What is the math idea?¶
দুটো স্তরে বোঝা যায় — সহজ আর জাদু:
সহজ (loop): 002-এর digit sum loop-টাই, কিন্তু একবার নয় — বারবার, যতক্ষণ না সংখ্যাটা এক digit (< 10) হয়।
জাদু (mod 9): আসলে digital root-এর একটা গোপন সূত্র আছে — % 9-এর সাথে গভীর সম্পর্ক। উত্তর সবসময় 1 থেকে 9-এর মধ্যে ঘোরে (n = 0 হলে 0), আর সেটা ধরা যায় এক লাইনে:
কেন % 9? কারণ 10 % 9 = 1, 100 % 9 = 1 — মানে প্রতিটা place value 9-এর জগতে 1-এর সমান। তাই একটা সংখ্যা আর তার digit sum, mod 9-এ সবসময় একই থাকে। বারবার digit যোগ করলেও mod 9 বদলায় না — শেষে যে এক-digit সংখ্যাটা থাকে, সেটাই mod 9-এর প্রতিনিধি।
3. Which basic idea does this inherit from?¶
সরাসরি 002 (Sum of Digits) থেকে — ভেতরের digit sum loop হুবহু 002-এর।
002 আর 007-এর সম্পর্ক এক বাক্যে:
002: digit sum একবার নাও -> 9875 হলে 29
007: digit sum বারবার নাও -> 9875 -> 29 -> 11 -> 2
যতক্ষণ না এক digit থাকে
মানে 007 হলো 002-কে একটা বাইরের loop-এ মুড়ে দেওয়া — "digit sum নাও, যদি বড় থাকে আবার নাও"। আগের problem-টা এখানে ভেতরের ইঞ্জিন হয়ে কাজ করছে। আর % 9-এর shortcut-টা concept-notes-এ যে "mod-এর জাদু" দেখেছিলে, তারই সরাসরি প্রয়োগ।
4. Real-life analogy¶
ভাবো একটা ভাঁজ করা কাগজ। তোমাকে একটা বড় সংখ্যা দেওয়া হলো, আর বলা হলো — "এটাকে ছোট করতে থাকো, প্রতিবার digit গুলো যোগ করে, যতক্ষণ না হাতের তালুতে ধরার মতো এক-অঙ্কের একটা সংখ্যা থাকে।"
- প্রতিটা ভাঁজ = একবার digit যোগ করা (সংখ্যা ছোট হলো)
- কাগজ যথেষ্ট ছোট (এক digit) হলে → থামো; হাতে যা রইল সেটাই digital root
আর mod 9-এর জাদুটা হলো — তুমি কাগজটা না ভাঁজ করেই বলে দিতে পারো শেষে কী থাকবে! কারণ digit যোগ করা সংখ্যার "9-এর জগতের পরিচয়" বদলায় না। অনেকটা accountant-দের পুরোনো "casting out nines" কৌশল — হিসাব ঠিক আছে কিনা যাচাই করতে digit যোগ করে 9 দিয়ে মেলানো।
5. Visual explanation¶
n = 9875 ধাপে ধাপে ছোট হচ্ছে, প্রতিটা ধাপে digit sum নিয়ে:
9875 --(9+8+7+5)--> 29 (>= 10, চালিয়ে যাও)
29 --(2+9)------> 11 (>= 10, চালিয়ে যাও)
11 --(1+1)------> 2 (< 10, থামো!)
digital root = 2
এবার mod 9 shortcut একই উত্তরে পৌঁছায় — এক লাফে:
1 + (9875 - 1) % 9
= 1 + 9874 % 9
= 1 + 1 (9874 = 9×1097 + 1, তাই remainder 1)
= 2 <- হুবহু loop-এর উত্তর!
কেন মেলে? কারণ প্রতি ধাপে digit sum নিলেও n % 9 একই থাকে — loop শুধু সেই mod 9 মানটাকেই খুঁজে বের করছে, ধীরে ধীরে।
6. Example input and output¶
input -> output (কীভাবে)
------------------------------
9875 -> 2 (9875 -> 29 -> 11 -> 2)
38 -> 2 (3+8 = 11 -> 1+1 = 2)
0 -> 0 <-- একমাত্র 0 যেখানে উত্তর 0
9 -> 9 (এক digit, নিজেই)
18 -> 9 (1+8 = 9)
99 -> 9 (9+9 = 18 -> 1+8 = 9)
123 -> 6 (1+2+3 = 6)
দুটো জিনিস খেয়াল করো: উত্তর সবসময় 1 থেকে 9 (শুধু n = 0 হলে 0), আর 9-এর গুণিতকের digital root সবসময় 9 (18, 99, 9 — সব 9)। এই দুটোই mod 9-এর জাদুর সরাসরি ফল।
7. Brute force thinking¶
সবচেয়ে সৎ, সরাসরি পদ্ধতি — 002-এর digit sum-টাকে একটা বাইরের loop-এ বসিয়ে বারবার চালাই, যতক্ষণ না এক digit থাকে:
def digital_root_loop(n):
n = abs(n)
while n >= 10: # যতক্ষণ একাধিক digit
s = 0
while n > 0: # ভেতরের 002-এর digit sum loop
s += n % 10
n //= 10
n = s # যোগফলকে নতুন n ধরো, আবার পরীক্ষা করো
return n
ভেতরের loop digit sum করে, বাইরের loop সেটা বারবার চালায়। পরিষ্কার, ঠিক উত্তর দেয়, আর 002 জানলে সহজেই লেখা যায়।
8. Why brute force may be slow¶
সত্যি বলতে এই loop "ধীর" নয় — খুব দ্রুত এক digit-এ নেমে আসে (প্রতি ধাপে সংখ্যা নাটকীয়ভাবে ছোট হয়; কয়েক ধাপেই শেষ)। তাহলে আরও ভাবার দরকার কেন?
- পুনরাবৃত্ত কাজ: আমরা একই ধরনের digit sum বারবার করছি, অথচ গণিত বলছে শেষ উত্তরটা সরাসরি
% 9দিয়ে পাওয়া যায়। loop সেই সত্যকে ধীরে ধীরে আবিষ্কার করছে মাত্র। - O(1)-এর হাতছানি: যদি সূত্রটা জানা থাকে, পুরো ব্যাপারটা এক ধাপে (একটা
%আর একটা যোগ) হয়ে যায় — কোনো loop ছাড়াই। interview-তে এই "আমি pattern-টা চিনি" দেখানোটাই মূল্যবান।
মূল শিক্ষা ধীরগতি নয় — একটা গাণিতিক pattern (mod 9) চিনতে পারলে loop পুরো বাদ দেওয়া যায়। চলো সেই জাদুটা দেখি।
9. Better thinking¶
মূল observation: digit sum নিলে সংখ্যার % 9 মান বদলায় না, তাই বারবার digit sum-এর শেষ ফলটা আসলে % 9-এরই একটা রূপ। সরাসরি সূত্র:
কেন 1 + (n - 1) % 9, সোজা n % 9 নয়? কারণ n % 9 9-এর গুণিতকে 0 দেয় (যেমন 18 % 9 = 0), অথচ digital root চায় 9। (n - 1) % 9 দিয়ে রেঞ্জটা 0-8-এ আনি, তারপর +1 করে 1-9-এ তুলি — ফলে 9-এর গুণিতকেও ঠিক 9 আসে। এটা O(1) — কোনো loop নেই।
10. Thinking tweak¶
আসল "আহা!" মুহূর্ত এক বাক্যে: digit যোগ করলে সংখ্যার % 9 মান বদলায় না — তাই বারবার digit যোগ করে এক-অঙ্কে নামার শেষ ফলটা আসলে % 9-ই (9-এর গুণিতকে 9, n = 0 হলে 0)।
কেন % 9 বদলায় না, এক ঝলক: 10 % 9 = 1, তাই 1234 = 1×1000 + 2×100 + 3×10 + 4 — mod 9-এ প্রতিটা place value 1-এর সমান, মানে 1234 % 9 = (1+2+3+4) % 9। সংখ্যা আর তার digit sum, mod 9-এ যমজ! এই একটা insight loop-টাকে এক লাইনে নামিয়ে দেয়। এই % 9-এর জাদু Level 2 (modular arithmetic)-এ পুরোপুরি খুলে দেখবে — আজ শুধু পরিচয়।
11. Step-by-step dry run¶
n = 9875। আগে brute force (যাতে চোখে দেখা যায়), পরে shortcut।
Brute force — বাইরের loop-এর প্রতি ধাপে ভেতরের digit sum:
| বাইরের ধাপ | শুরুর n | digit sum | নতুন n | n >= 10? |
|---|---|---|---|---|
| 1 | 9875 | 9+8+7+5 = 29 | 29 | হ্যাঁ, চালাও |
| 2 | 29 | 2+9 = 11 | 11 | হ্যাঁ, চালাও |
| 3 | 11 | 1+1 = 2 | 2 | না, থামো |
শেষে n = 2 → digital root = 2।
Shortcut — এক লাইনে:
দুটোই 2 — একই সত্য, কিন্তু shortcut সোজা গন্তব্যে। নিজে খাতায় 38 দিয়ে দুই পথ মেলাও (loop: 11 → 2; formula: 1 + 37 % 9 = 1 + 1 = 2)।
12. Python solution¶
def digital_root(n):
"""n-এর digital root, O(1) mod-9 সূত্রে।"""
n = abs(n)
if n == 0:
return 0
return 1 + (n - 1) % 9
def digital_root_loop(n):
"""একই উত্তর, বারবার digit sum (002-এর loop-এর উপর)।"""
n = abs(n)
while n >= 10:
s = 0
while n > 0:
s += n % 10
n //= 10
n = s
return n
# --- ছোট test গুলো (সব pass করা উচিত) ---
assert digital_root(9875) == 2 # 9875 -> 29 -> 11 -> 2
assert digital_root(38) == 2 # 38 -> 11 -> 2
assert digital_root(0) == 0 # একমাত্র 0-তে উত্তর 0
assert digital_root(9) == 9 # এক digit
assert digital_root(18) == 9 # 9-এর গুণিতক -> 9
assert digital_root(99) == 9 # 99 -> 18 -> 9
assert digital_root(123) == 6 # 1+2+3
# দুই পদ্ধতি সব ছোট সংখ্যায় মেলে কিনা যাচাই:
for x in range(0, 1000):
assert digital_root(x) == digital_root_loop(x)
print(digital_root(9875)) # 2
print(digital_root(99)) # 9
print(digital_root(0)) # 0
print("সব test pass!")
13. Line-by-line explanation¶
চিহ্ন সরিয়ে দিই, যাতে সূত্র আর loop দুটোই non-negative সংখ্যার উপর ঠিকমতো চলে।
একমাত্র বিশেষ ক্ষেত্র। 0-এর digital root 0; সূত্র 1 + (n-1) % 9 0-তে ভুল দিত (1 + (-1) % 9 = 1 + 8 = 9), তাই আগেই আলাদা করে 0 ফেরাই।
এই problem-এর জাদুকরী লাইন। (n - 1) % 9 ফলটাকে 0-8 রেঞ্জে আনে, +1 সেটাকে 1-9-এ তোলে — তাই 9-এর গুণিতকেও সঠিক 9 আসে (সোজা n % 9 দিলে 0 আসত)।
বাইরের while n >= 10 চালায় যতক্ষণ একাধিক digit; ভেতরের loop হুবহু 002-এর digit sum; n = s যোগফলকে নতুন n বানায়। এক digit-এ নামলে থামে। এটা ধীরে ধীরে সেই mod-9 মানেই পৌঁছায়।
শেষের for x in range(0, 1000) loop দুই পদ্ধতির উত্তর প্রতিটা সংখ্যায় মিলিয়ে দেখে — সূত্রটা সত্যিই loop-এর সমান কিনা তার পাকা প্রমাণ।
14. Output walkthrough¶
চালালে ছাপবে:
তিনটা মান তিনটা print থেকে: digital_root(9875) → 2, digital_root(99) → 9 (9-এর গুণিতক), digital_root(0) → 0 (বিশেষ ক্ষেত্র)। তার আগে সব assert চুপচাপ pass করেছে — যার মধ্যে 0 থেকে 999 পর্যন্ত প্রতিটা সংখ্যায় দুই পদ্ধতির মিল যাচাইও আছে। শেষ লাইন সব test pass! মানে সূত্র আর loop হুবহু মেলে, সব ঠিক।
15. Time complexity¶
- সূত্র version: O(1) — একটা বিয়োগ, একটা
%, একটা যোগ; সংখ্যা যত বড়ই হোক একই কাজ। - loop version: O(log n) কার্যত আরও কম — প্রথম digit sum-ই সংখ্যাটাকে নাটকীয়ভাবে ছোট করে দেয় (যেমন 9875 → 29), তারপর মাত্র কয়েক ধাপ। তবু upper bound হিসেবে প্রথম sum-এর জন্য O(log n) ধরা হয়।
interview-তে সূত্রটা বলতে পারলে O(1) — সেটাই চমক।
16. Space complexity¶
O(1) — দুই version-এই গুটিকয় variable (n, s), সংখ্যার আকারের সাথে বাড়ে না। কোনো list বা string লাগছে না।
17. Common mistakes¶
- সোজা
n % 9ফেরানো — সবচেয়ে কমন ভুল।18 % 9 = 0, কিন্তু digital root 9। তাই1 + (n - 1) % 9লাগে, খালিn % 9নয়। n = 0ভুলে যাওয়া — সূত্র 0-তে ভুল (1 + (-1) % 9 = 9) দেয়; আগেif n == 0: return 0বসাও।- একবার মাত্র digit sum নেওয়া — digital root চায় বারবার যোগ যতক্ষণ এক digit; 9875-এর উত্তর 2, 29 নয়। বাইরের loop (বা সূত্র) ছাড়া ভুল হবে।
while n >= 10-এর জায়গায়while n > 10— তাহলে n = 10-এ থেমে যাবে (অথচ 10 → 1 হওয়া উচিত)। সীমা>= 10হতে হবে।- ভেতরের loop-এ
n //= 10ভুলে যাওয়া — তাহলে digit sum loop কখনো থামবে না → infinite loop।
18. Harder problems that inherit this idea¶
- LeetCode — Add Digits (https://leetcode.com/problems/add-digits/) — এটাই digital root, হুবহু; follow-up-এ O(1) (loop ছাড়া) চায় — মানে এই
% 9সূত্রটাই উত্তর। - 006 — Armstrong Number (এই repo) আর LeetCode — Happy Number (https://leetcode.com/problems/happy-number/) — "digit নিয়ে কিছু করে বারবার চালানো" — একই repeated-digit-process পরিবার।
- Level 2 — Modular arithmetic (এই repo-র পরের দিকে) —
% 9-এর এই জাদু সেখানে পুরোপুরি খোলে: divisibility rule (3, 9 দিয়ে ভাগ), checksum, casting out nines — সবের মূলে এই idea।
19. Pattern learned¶
Repeated digit sum = mod 9। digit যোগ করা সংখ্যার % 9 বদলায় না, তাই বারবার digit sum-এর শেষ ফল সরাসরি 1 + (n-1) % 9 (n = 0 হলে 0)। বড় শিক্ষা: একটা গাণিতিক ধর্ম (এখানে mod 9 invariance) চিনতে পারলে গোটা loop এক লাইনে নামানো যায়। এই "loop-এর পেছনের সূত্র খোঁজা" চোখটাই পরের অনেক optimization-এর চাবি।
20. Final summary¶
ভবিষ্যতের আমাকে এক লাইনে: "digital root = বারবার digit sum যতক্ষণ এক digit; কিন্তু shortcut 1 + (n-1) % 9 (n = 0 হলে 0), কারণ digit sum সংখ্যার % 9 বদলায় না। 9-এর গুণিতকে উত্তর 9 — তাই সোজা n % 9 নয়।"
আগের ধাপ → 002 — Sum of Digits (ভেতরের digit sum loop)। শিকড়ে → 001 — Even or Odd। সম্পর্কিত → 006 — Armstrong Number।
ফিরে দেখা: এই level-এর map → ../README.md · এই note-টা যে ছকে লেখা → ../../../templates/math-problem-note-template.md।
মনে রাখার নিয়ম (legal): সব নিজের ভাষায় লেখা; problem statement copy করা হয়নি; official link শুধু attribution-এর জন্য; "asked by Google/Amazon" এমন দাবি করা হয়নি — বরং "common interview pattern" লেখা।