সমস্যাঃ একটি লিস্টে কিছু সংখ্যা থাকবে, সংখ্যাগুলোর সবগুলো বিন্যাস (permutation) বের করতে হবে।
উদাহরণ: লিস্টে যদি [1, 2, 3] থাকে, তাহলে বিন্যাসগুলো হবে –
[1, 2, 3] [1, 3, 2] [2, 1, 3] [2, 3, 1] [3, 2, 1] [3, 1, 2]
সমাধানঃ এটি মোটামুটি সহজ একটি সমস্যা। তবে ইন্টারভিউয়ের সময় শুরুতেই একটি প্রশ্ন করে নিতে হবে – সংখ্যাগুলো অনন্য (unique) কী না। এটি না জেনেই হুট করে কোড শুরু করা যাবে না। যদি অনন্য হয়, তাহলে খুব সহজেই রিকার্শন ব্যবহার করে সমাধান করা যায়। আর যদি তা না হয়, তাহলে একটু সতর্ক থাকতে হবে, যেন একটি বিন্যাস একাধিকবার প্রিন্ট করা না হয়।
এরকম সহজ সমস্যা দেওয়ার কারণ হচ্ছে বেসিক রিকার্শন স্কিল পরীক্ষা করা। তারপরও আমার পরিচিত একজন ইন্টারভিউয়ারের মতে, ৮০% প্রার্থী এই সমস্যাটির সমাধান করতে ব্যর্থ হয়। যাই হোক, যারা এর সঙ্গে পরিচিত নয়, তারা শাফায়েতের ব্লগের ব্যাকট্র্যাকিং: পারমুটেশন জেনারেটর আর্টিকেলটি পড়তে পারে।
আমি পাইথন কোড দিচ্ছি –
def permute(i, n): if i == n: print(r) return for j in range(n): if visited[j]: continue visited[j] = True r[i] = a[j] permute(i+1, n) visited[j] = False if __name__ == "__main__": a = [1, 2, 3, 4] r = [0] * len(a) n = len(a) visited = [False] * n permute(0, n)
এখন, এখানে আমি অতিরিক্ত একটি অ্যারে বা লিস্ট ব্যবহার করেছি। যদি সেটি করতে না চাই, তাহলে আরেকভাবে সমস্যাটির সমাধান করা যায়। সেক্ষেত্রে প্রতিটি অবস্থানের জন্য বাকী অবস্থানে যেসব সংখ্যা আছে, তাদের মধ্যে অদল-বদল করতে হবে। কোড হবে এরকম –
def permute2(l, r): if l == r: print(a) return for i in range(l, r+1): a[l], a[i] = a[i], a[l] permute2(l+1, r) a[l], a[i] = a[i], a[l] if __name__ == "__main__": a = [1, 2, 3, 4] r = [0] * len(a) n = len(a) visited = [False] * n permute2(0, n-1)
এখন, যদি বলা হয়, লিস্টের সংখ্যাগুলো অনন্য নয়, একই সংখ্যা একাধিকবার থাকতে পারে, কিন্তু একই বিন্যাস একাধিকবার প্রিন্ট করা যাবে না, তাহলে আমরা দুটি পদ্ধতি প্রয়োগ করতে পারি। প্রথমত একটি সেট ব্যবহার করবো। প্রতিটি বিন্যাস পাওয়ার পরে সেটি সেটে আছে কী না পরীক্ষা করবো, যদি না থাকে, তাহলেই কেবল প্রিন্ট করব এবং সেটে যোগ করে নিব।
def permute3(i, n): if i == n: tpl = tuple(r) if tpl in s: return s.add(tpl) print(r) return for j in range(n): if visited[j]: continue visited[j] = True r[i] = a[j] permute3(i+1, n) visited[j] = False if __name__ == "__main__": a = [1, 2, 3, 1] n = len(a) r = [0] * n visited = [False] * n s = set() permute3(0, n)
এখন আমরা যদি সেট ব্যবহার করতে না চাই, তাহলে দ্বিতীয় প্রোগ্রামের কোড একটু পরিবর্তন করে কাজটি করা যায় –
def permute4(l, r): if l == r: print(a) return for i in range(l, r+1): if a[i] in a[l:i]: continue a[l], a[i] = a[i], a[l] permute4(l+1, r) a[l], a[i] = a[i], a[l] if __name__ == "__main__": a = [1, 2, 3, 1] n = len(a) r = [0] * n visited = [False] * n
ওপরের প্রোগ্রামটি কীভাবে কাজ করে, সেটি বুঝতে হলে একটু খাতাকলম নিয়ে বসতে হবে। পরিশ্রম ছাড়া এমনি এমনি প্রোগ্রামিং শেখা যায় না।