লোয়েস্ট কমন অ্যানসেস্টর

লোয়েস্ট কমন অ্যানসেস্টর জিনিসটা শুনতে একটু কঠিন মনে হলেও জিনিসটা সহজ আর খুবই কাজের। বেশ কিছু ধরণের প্রবলেম সলভ করে ফেলা যায় এই অ্যালগোরিদম দিয়ে। আমরা প্রথমে লোয়েস্ট কমন অ্যানসেস্টর বের করার ব্রুটফোর্স অ্যালগোরিদম দেখবো, তারপর স্পার্স টেবিল নামের নতুন একটা ডাটা স্ট্রাকচার শিখে কমপ্লেক্সিটি লগ এ নামিয়ে আনবো।

প্রথমেই আমরা জেনে নেই লোয়েস্ট কমন অ্যানসেস্টর বা এল.সি.(LCA) কি সেটা। নিচের ছবিটা দেখ:

 

ছবিতে k আর n নোডের প্যারেন্ট ধরে পিছে যেতে থাকলে তারা i নোডে এসে মিলবে। i হলো k,n এর লোয়েস্ট কমন অ্যানসেস্টর। ‘a’ ও দুইজনের কমন অ্যানসেস্টর কিন্তু i হলো ‘লোয়েস্ট’ বা সবথেকে কাছাকাছি।

মনে করো ট্রি টা দিয়ে বুঝানো হচ্ছে একটা অফিসে কে কার সুপারভাইজর বা বস। a হলো অফিসের হেড, b এবং i এর বস হলো a, আবার c এবং f এর বস হলো b ইত্যাদি। তাহলে লোয়েস্ট কমন অ্যানসেস্টর অ্যালগোরিদম দিয়ে দুইজন এমপ্লয়ির লোয়েস্ট/সব থেকে কাছের কমন বস খুজে বের করে ফেলতে পারি সহজেই।

প্রথমে আমরা একদম নেইভ একটা উপায় দেখি যেটা খুব বড় ইনপুটের জন্য কাজ করবেনা। মনে করি আমরা k,n এর এল.সি.এ বের করতে চাই, প্রথমেই k এর প্যারেন্ট ধরে ধরে উপরে উঠে সবগুলো অ্যানসেস্টরের লিস্ট করে ফেলি:

k এর অ্যানসেস্টর:

j

i

a

n এর অ্যানসেস্টর:

m

i

a

 

 

দুইটা লিস্টেই বাম থেকে ডানে যেতে থাকলে প্রথম যে দুইটা নোড কমন পাবো সেটাই হবে এল.সি.! এই উদাহরণে প্রথমে কমন পাবো i, তাই সেটাই হবে এল.সি.এ। ‘a’ ও একটা কমন এনসেস্টর কিন্তু সেটা লোয়েস্টনা। একটা ভিজিটেড অ্যারে বা ফ্ল্যাগ ব্যবহার করে খুব সহজেই আমরা এই অ্যালগোরিদম ইমপ্লিমেন্ট করতে পারি। প্রথমে k এর লিস্টে যারা আছে সেগুলোর ইনডেক্সের ফ্ল্যাগ অন করে দিবো, তারপর n এর লিস্টে লুপ চালিয়ে দেখবো ফ্ল্যাগ অন আছে কোন নোডের।

প্রতিটা লিস্ট বানাতে আমাদের সর্বোচ্চ n টা নোড ভিজিট করা লাগতে পারে, তাই কমপ্লেক্সিটি O(n)

O(n) কমপ্লিক্সিটি খারাপ না যদি আমাদের মাত্র এক জোড়া নোডের এল.সি.এ বের করতে হয়। যদি আমাদের m জোড়া নোড দিয়ে বলে প্রতিটা পেয়ারের এল.সি.এ বের করতে তাহলে কমপ্লেক্সিটি হয়ে যাবে O(m*n) । এটাকে আমরা চাইলে O(m*log(n)) এ সলভ করতে পারি, অর্থাৎ প্রতি কুয়েরি কাজ করবে O(logn) এ। তবে তার আগে কিছু প্রিপ্রসেসিং করে নিতে হবে।

প্রতিটা নোডের জন্য আমরা সেভ করে রাখবো নোডটার:

  • প্রথম প্যারেন্ট বা ২^০ তম প্যারেন্ট

  • ২য় প্যারেন্ট বা ২^১ তম প্যারেন্ট

  • ৪র্থ প্যারেন্ট বা ২^২ তম প্যারেন্ট

  • ৮ম প্যারেন্ট বা ২^৩ তম প্যারেন্ট

যদি K-তম প্যারেন্ট বলতে কিছু না থাকে তাহলে আমরা ১ রেখে দিবো। যেমন রুট নোডের প্রথম প্যারেন্ট বলতে কিছু নেই, তাই প্রথম প্যারেন্ট হবে ১।

 

 

 

তাহলে উপরের ট্রি এর জন্য নিচের মতো একটা ২ডিমেনশনাল টেবিল তৈরি হবে:

 

নোড

^০ তম প্যারেন্ট

^১ তম প্যারেন্ট

^২ তম প্যারেন্ট

 

তাহলে উপরের ট্রি এর জন্য নিচের মতো একটা ২ডিমেনশনাল টেবিল তৈরি হবে:

এই টেবিলটাকে বলে স্পার্স টেবিল। এই টেবিলটা তৈরি করে কি লাভ হলো? এখন তুমি কোনো নোডের k তম প্যারেন্ট খুব সহজেই খুজে বের করতে পারবে। যেমন কারো ২৫ তম প্যারেন্ট দরকার হলে প্রথমে ২৫ এর নিচে ২ এর সবথেকে বড় পাওয়ার ২^=১৬ তম প্যারেন্ট খুজে বের করবে। তারপর ১৬তম প্যারেন্টের (২৫১৬)=৯তম প্যারেন্ট খুজে বের করবে একই পদ্ধতিতে! প্রতিটা সংখ্যাকে ২ এর পাওয়ারের যোগফল হিসাবে প্রকাশ করা যায় সেটারই সুবিধা নিচ্ছি আমরা। এভাবে করে O(logn) কমপ্লেক্সিটিতে আমরা কারো k তম প্যারেন্ট বের করতে পারবো যেটা পরবর্তীতে LCA বের করতে কাজে লাগবে।

 

এখন কথা হলো উপরের টেবিলটা বানাবো কিভাবে? ^০ তম প্যারেন্ট বের করা সহজ, ট্রি এর উপর একটা ডেপথফার্স্টসার্চ বা ব্রেথডফার্স্টসার্চ চালিয়ে একটা অ্যারেতে সেভ করে রাখবো কার প্যারেন্ট কে। মনে করি অ্যারেটা হলো T। তাহলে উপরের ট্রি এর জন্য পাবো T[]=-, T[]=,T[]=৫ ইত্যাদি।

এবার আমরা কলামবাইকলাম টেবিলটা ভরাট করবো। প্রথমেই প্রথম কলাম ভরাট করবো যেটা আসলে T অ্যারের সমান।

টেবিলটা নাম P হলে প্রথম কলাম ভরাট করবো এভাবে,

এখন চিন্তা করে দেখো একটা নোডের ২^=১৬ তম প্যারেন্ট হলো নোডটার ২^=৮ম প্যারেন্টের ২^=৮ম প্যারেন্ট। তাহলে কোন নোডের 2^j তম প্যারেন্ট হবে নোডটার 2^(j-1) তম প্যারেন্টের 2^(j-1) তম প্যারেন্ট! তাহলে যেকোন নোড i এর জন্য:

P[i][j]=P[P[i][j – 1]][j – 1] যেখানে P[i][j – 1] হলো j-1 তম প্যারেন্ট।

তাহলে পুরো টেবিলটা ভরাট করে ফেলতে পারি এভাবে:

2^j এর মান যদি N এর ছোট না হয় তাহলে নতুন কোনো প্যারেন্ট পাবার সম্ভাবনা নেই, তাই লুপ চলবে (1<<j) বা 2^j যতক্ষণ N এর ছোট হবে।

এখন আমরা দুটি নোডের এল.সি.এ বের করবো। আমাদের আরেকটা অতিরিক্ত অ্যারে লাগবে L[] যেখানে থাকবে প্রতিটা নোডের ডেপথ বা লেভেল। রুট নোডের লেভেল হবে ০।

 

 

এখন আমরা হলুদ নোড দুইটার এল.সি.এ বের করতে চাই:

প্রথম কাজ হবে এদেরকে একই লেভেলে নিয়ে আসা। p এর লেভেল থেকে ২ এর পাওয়ার বিয়োগ করতে থাকবো এবং একই সাথে স্পার্স টেবিল ব্যবহার করে উপরে উঠাতে থাকবো যতক্ষণনা q এর লেভেলের সমান হয়।

এখন দুইজন সমান লেভেলে আসলো। এবার কাজ হবে দুইটা নোডকেও টেনে একসাথে উপরে উঠানো যতক্ষনা নোড দুটির প্যারেন্ট একসাথে এসে মিলে:

 

কোডটা হবে এরকম:

আমরা এমন জায়গায় p,q কে আনছি যেখানে তাদের দুইজনেরই প্যারেন্ট একই। তাহলে এখন p বা q এর প্যারেন্টই হবে এল.সি.এ।

 

 

আমরা যদি লুপের মধ্যেই ২ আর ৮ নাম্বার নোডকে p,q তে না এনে ডিরেক্ট ১ নম্বর নোডে আনার চেষ্টা করতাম তাহলে কি ঘটতে পারতো? P[p][i] != P[q][i] এই শর্তটা না থাকলে কি হত? এটা চিন্তা করা তোমার কাজ 🙂

 

কমপ্লেক্সিটি:

স্পার্স টেবিলে রো থাকবে n টা, প্রতিটা রো তে logn টা কলাম থাকবে, প্রিপ্রোসেসিং কমপ্লেক্সিটি O(nlogn)

প্রতি কুয়েরিতে কমপ্লেক্সিটি O(logn)

রিলেটেড প্রবলেম:

১. একটা ওয়েটেড ট্রি দেয়া আছে। দুটি নোড দিয়ে বলা হলো তাদের মধ্যের দূরত্ব বের করতে হবে। (Spoj QTree)

২. একটা ওয়েটেড ট্রি দেয়া আছে। দুটি নোড দিয়ে বলা হলো তাদের মধ্যকার পাথের সর্বোচ্চ ওয়েটের এজ এর মান বলতে হবে। (LOJ A Secret Mission)

সলিউশন ১: শুরুতে ডিএফস চালিয়ে রুট থেকে প্রতিটা নোডের দূরত্ব সেভ করে রাখো। এখন নোড দুইটা a,b এবং তাদের কমন অ্যানসেস্টর L হলে সলিউশন হবে dist(root,a)+dist(root,b)-2*dist(root,L)।

সলিউশন ২: স্পার্স টেবিলে অতিরিক্ত একটা ইনফরমেশন রাখতে হবে।  প্রতিটা নোড থেকে 2^0, 2^1, 2^2 ইত্যাদি তম প্যারেন্টের জন্য, প্যারেন্ট থেকে ওই নোডের মধ্যকার পাথের সর্বোচ্চ ওয়েটের এজের মান সেভ করে রাখতে হবে। বিস্তারিত লিখলাম, এটা সলভ করা তোমার কাজ।

আরো কিছু প্রবলেম:

TJU Closest Common Ancestors
UVA Flea Circus

LightOJ LCA Category

রিসোর্স:

এল.সি.এ বের করার আরো কিছু পদ্ধতি আছে যেগুলো পাবে এখানে। এগুলো ছাড়াও টারজানের একটা অফলাইন অ্যালগোরিদম আছে যেটা আগে সব কুয়েরি ইনপুট নিয়ে একটি ডিএফস চালিয়ে সবগুলো পেয়ারের এল.সি.এ বের করে দেয়।

সম্পূর্ণ কোড:

ভেক্টর g তে ট্রি সেভ করতে হবে। প্রথমে dfs ফাংশন কল করে লেভেল,প্যারেন্ট ফিক্সড করে তারপর lca_init কল করে প্রিপ্রসেসিং করতে হবে।

হ্যাপি কোডিং!

Print Friendly, PDF & Email

ফেসবুকে মন্তব্য

comments

Powered by Facebook Comments

42,872 times read (exlcuding bots)

13 thoughts on “লোয়েস্ট কমন অ্যানসেস্টর

  1. (Complexity description এর ঠিক আগের অংশ)

    “আমরা যদি লুপের মধ্যেই ২ আর ৮ নাম্বার নোডকে p,q কে না এনে ডিরেক্ট ১ নম্বর নোডে আনার চেষ্টা করতাম তাহলে কি ঘটতে পারতো? এটা চিন্তা করা তোমার কাজ 🙂 । ”

    ভাইয়া, এই অংশটুকু আরেকবার বুঝিয়ে বলবেন? 🙂

    1. প্রশ্ন হল আমরা লুপের মধ‍্যেই কেন কুয়েরি নোড দুটিকে টেনে উঠাতে উঠাতে lca পর্যন্ত উঠাইনি? P[p][i] != P[q][i] এই শর্তটা মুছে দিলে কি হবে?

    1. আমার পরামর্শ হবে সম্ভব হলে যতরকম টেকনিক জানা যায় সবই শিখে রাখা, তাতে তুমি অনেক ভাবে প্রবলেমকে অ্যাটাক করতে পারবে, কোন টেকনিক কখন কাজে লেগে যাবে সেটা আগে থেকে বলা যায় না।

  2. ” যেমন কারো ২৫ তম প্যারেন্ট দরকার হলে প্রথমে ২৫ এর নিচে ২ এর সবথেকে বড় পাওয়ার ২^৪=১৬ তম প্যারেন্ট খুজে বের করবে। তারপর ১৬তম প্যারেন্টের (২৫–১৬)=৯তম প্যারেন্ট খুজে বের করবে একই পদ্ধতিতে! ” –
    ভাইয়া এই ব্যাপার টা একটা চিত্রের মাধ্যমে বুঝিয়ে দিলে বেশ ভালো হয় । এমনিতে পরিষ্কার হচ্ছে উদাহরণ না ।

  3. অসাধারণ ভাইয়া… 67 লাইনে dfs ফাংশনকে dfs(-1,0,0) দিয়ে কল
    করা উচিত।তাছাড়া স্পার্স টেবিলের মান কিছু কিছু জায়গায় -1 স্থলে 0 আসে।

Leave a Reply

Your email address will not be published. Required fields are marked *