ট্রাভেলিং সেলসম্যান সমস্যা

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

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

মনেকরো তুমি একদিন যশোর থেকে মাগুরা বেড়াতে গেছো। সেখানে তোমার n জন বন্ধুর বাড়ি আছে। তুমি একে একে তাদের সবার বাড়ি যেতে চাও। তাদের সবার বাড়ির দূরত্ব তুমি জানো। তুমি প্রথমে তোমার সবচেয়ে ভালো বন্ধু ১ এর বাসায় গেছো, তারপর একে একে সবার বাসা ঘুরে আবারও তোমার জিগার বন্ধু ১ এর বাসায় ফিরে আসবে। সব থেকে কম মোট কত দূরত্ব অতিক্রম করে তুমি সবার বাসা ঘুরতে পারবে। এটিই হলো ট্রাভেলিং সেলসম্যান সমস্যা (Travelling Salesman Problem)।

এর আগে হয়তো তোমরা DP উপায়ে একপ্টি সমস্যা সমাধানের জন্য বড় একটি সমস্যাকে ছোট সমস্যা দিয়ে সমাধান করেছো। DP ছাড়া আরেকটি উওয়ায় আছে, আর তা হলো একই রকম জিনিস খুজে বের করা। যেমন একটী আগে বলা সমস্যার ক্ষেত্রে খেয়াল করো, তুমি  ১ – ২ -৩ – ৪, এভাবে ৪ জন বন্ধুর বাসায় ঘুরেছো, আর বাকি আছে ৫…n বন্ধুরা। এই বাকি বন্ধুদের বাসা ঘুরতে তোমার যেই সবচেয়ে কম খরচ হচ্ছে ১ -৩ – ২ – ৪ ঘোরার পর বাকি বন্ধুদের বাসা ঘুরে ফেলার জন্য সবচেয়ে কম খরচের সমান। অর্থাৎ, কোন এক সময় তোমাকে শুধু জানতে হবে তুমি কোন কোন বন্ধুর বাসা ঘুরে ফেলছো এবং এখন তুমি কোথায় আছো।

বিভিবনভাবে আমরা একই দশা বা স্টেট (state) এ আসতে পারি, যেমন উপরের উদাহরণে আমরা ৪ জন বন্ধুর বাসা দুভাবে ঘুরে এখন ৪ এর বাসায় আছি। অর্থাৎ তোমার স্টেট হলো তুমি কার কার বাসায় ঘুরে ফেলছো (১,২,৩,৪) আর এখন কোথায় আছো (৪)। এখন কোথায় আছো সেটি শুধু একটি সংখ্যা, কিন্তু তুমি কোথায় কোথায় ঘুরে ফেলেছো এই জিনিস অনেকগুলো সংখ্যার সেট।

আমরা DP এর সময় স্টেটকে অ্যারের প্যারামিটার হিসেবে লিখি। এক্ষেত্রে আমরা একটি সেটকে কিভাবে সংখ্যা আকারে লিখতে পারি? আচ্ছা, এখটূ খেয়াল করে দেখো, আমাদের মোট n জন বন্ধু আছে, কার কার বাসায় গিয়েছি তাদের ১ আর কার কার বাসায় এখনো যাওয়া হয়নি তাদের ০ দিয়ে লিখতে পারি। তাহলে n টি ০-১ দিয়ে আমরা কার কার বাসায় গিয়েছি সেটি বানিয়ে ফেলতে পারি। কিন্তু তুমি যদি অ্যারের মাত্রা বা ডাইমেনশন (Dimension) n নিতে চাও তাহলে নিশ্চয়ই কোড করা খুব একটা সুখকর হবে না?

এখানে একটি মজার চালাকি আছে, আর তা হলো তুমি এই ০-১ সংখাকে বাইনারি ফর্মে ভাবতে পারো। যেমনঃ তোমার যদি ১,২,৪ নাম্বার বন্ধুর বাসা ঘোরা হয়ে থাকে তাহলে তোমার নাম্বার হবেঃ ০০০০…১০১১ = ৭.  এখন এই সংখ্যা কত বড় হতে পারে? 2n কারন একটি বন্ধু থাকতে পারে নাও পারে। যেহেতু আমাদের মোট ন জন বন্ধু তাই এই সংখ্যা 2n রকম হতে পারে। তাহলে আমাদের পুরো স্টেট কত বড়? nx2n এবং প্রতি স্টেটে গিয়ে তুমি অন্যান্য সবার বাসায় যাওয়ার চেষ্টা করবে (n ভাবে)। সুতরাং আমাদের টাইম কমপ্লেক্সিটি হবে O(n22n).

এতক্ষণ যা জানলে তা উপরে ছবিটা দেখে আরো বেশি স্পস্ট হওয়ার কথা, কিন্তু যদি ন হয় তবে ইমপ্লিমেন্টেশন দেখলে পুরোপুরি ক্লিয়ার হয়ে যাবে।

যেহেতু শুরুতেই বলেছি যে ডায়ানামিক প্রোগ্রামিং বেশ কঠিন এবং বেশি বেশি অনুশীলন না করলে ভালো করা যায় না, বলা যায় বোঝাও যায় না।

আমার বিশ্বাস ট্রাভেলিং সেলসম্যান সমস্যা এখন আর আগের মত কঠিন লাগবে না। সবার জন্য শুভ কামনা।।