রুবি ম্যাজিকের আরেকটি পর্বের সময় এসেছে! এই সময়, আমরা রুবির সবচেয়ে জাদুকরী বৈশিষ্ট্যগুলির মধ্যে একটি দেখব, যেটি Array
এর মত রুবির গণনাযোগ্য ক্লাসগুলির সাথে কাজ করার সময় আপনি যে পদ্ধতিগুলি ব্যবহার করবেন তার বেশিরভাগই প্রদান করে৷ , Hash
এবং Range
. প্রক্রিয়ায়, আমরা শিখব আপনি গণনাযোগ্য বস্তুর সাথে কী করতে পারেন, কীভাবে গণনা কাজ করে এবং কীভাবে একটি একক পদ্ধতি প্রয়োগ করে একটি বস্তুকে গণনাযোগ্য করে তোলা যায়।
Enumerable
, #each
এবং Enumerator
গণনা বস্তুর উপর দিয়ে অতিক্রম করা বোঝায়। রুবিতে, আমরা একটি বস্তুকে গণনাযোগ্য বলি যখন এটি আইটেমগুলির একটি সেট এবং তাদের প্রতিটি লুপ করার একটি পদ্ধতি বর্ণনা করে।
অন্তর্নির্মিত গণনাযোগ্যগুলি Enumerable
অন্তর্ভুক্ত করে তাদের গণনার বৈশিষ্ট্যগুলি পায় মডিউল, যা #include?
এর মত পদ্ধতি প্রদান করে , #count
, #map
, #select
এবং #uniq
, অন্যদের মধ্যে. অ্যারে এবং হ্যাশের সাথে যুক্ত বেশিরভাগ পদ্ধতি আসলে এই ক্লাসগুলিতে প্রয়োগ করা হয় না, সেগুলি অন্তর্ভুক্ত করা হয়েছে৷
দ্রষ্টব্য :কিছু পদ্ধতি, যেমন #count
এবং #take
Array
-এ ক্লাস, আছে Enumerable
থেকে ব্যবহার না করে অ্যারের জন্য বিশেষভাবে প্রয়োগ করা হয়েছে মডিউল এটি সাধারণত দ্রুত অপারেশন করার জন্য করা হয়৷
Enumerable
মডিউল #each
নামের একটি পদ্ধতির উপর নির্ভর করে , যা এটি অন্তর্ভুক্ত যেকোন শ্রেণিতে প্রয়োগ করা প্রয়োজন। যখন অ্যারের একটি ব্লকের সাথে কল করা হয়, তখন #each
পদ্ধতি অ্যারের প্রতিটি উপাদানের জন্য ব্লক কার্যকর করবে।
irb> [1,2,3].each { |i| puts "* #{i}" }
* 1
* 2
* 3
=> [1,2,3]
যদি আমরা #each
কল করি একটি অ্যারের উপর পদ্ধতি ব্যতীত এর প্রতিটি উপাদানের জন্য কার্যকর করার জন্য একটি ব্লক পাস করলে, আমরা Enumerator
এর একটি উদাহরণ পাব .
irb> [1,2,3].each
=> #<Enumerator: [1, 2, 3]:each>
Enumerator
এর উদাহরণ একটি বস্তুর উপর কিভাবে পুনরাবৃত্তি করতে হয় তা বর্ণনা করুন। গণনাকারী ম্যানুয়ালি বস্তুর উপর পুনরাবৃত্তি করে এবং চেইন গণনা করে।
irb> %w(dog cat mouse).each.with_index { |a, i| puts "#{a} is at position #{i}" }
dog is at position 0
cat is at position 1
mouse is at position 2
=> ["dog", "cat", "mouse"]
#with_index
পদ্ধতি পরিবর্তিত গণনাকারীরা কীভাবে কাজ করে তার একটি ভাল উদাহরণ। এই উদাহরণে, #each
একটি গণনাকারী ফেরত দিতে অ্যারেতে বলা হয়। তারপর, #with_index
প্রতিটি উপাদানের সূচক প্রিন্ট করার অনুমতি দেওয়ার জন্য অ্যারের প্রতিটি উপাদানে সূচক যোগ করতে বলা হয়।
অবজেক্টকে গণনাযোগ্য করা
হুডের নিচে, #max
এর মত পদ্ধতি , #map
এবং #take
#each
-এর উপর নির্ভর করুন কাজ করার পদ্ধতি।
def max
max = nil
each do |item|
if !max || item > max
max = item
end
end
max
end
অভ্যন্তরীণভাবে, Enumerable
এর পদ্ধতিতে সি বাস্তবায়ন রয়েছে, কিন্তু উপরের উদাহরণটি মোটামুটিভাবে দেখায় কিভাবে #max
কাজ করে #each
ব্যবহার করে সমস্ত মান লুপ করতে এবং সর্বোচ্চ মনে রাখার জন্য, এটি সর্বোচ্চ মান প্রদান করে।
def map(&block)
new_list = []
each do |item|
new_list << block.call(item)
end
new_list
end
#map
ফাংশন প্রতিটি আইটেমের সাথে পাস করা ব্লককে কল করে এবং সমস্ত মান লুপ করার পরে ফিরে আসার জন্য ফলাফলটিকে একটি নতুন তালিকায় রাখে।
যেহেতু Enumerable
-এ সমস্ত পদ্ধতি #each
ব্যবহার করুন পদ্ধতি কিছুটা হলেও, একটি কাস্টম ক্লাস গণনাযোগ্য করার জন্য আমাদের প্রথম ধাপ হল #each
বাস্তবায়ন করা পদ্ধতি।
#each
প্রয়োগ করা
#each
বাস্তবায়ন করে ফাংশন এবং Enumerable
সহ একটি ক্লাসে মডিউল, এটি গণনাযোগ্য হয়ে ওঠে এবং #min
এর মত পদ্ধতি গ্রহণ করে , #take
এবং #inject
বিনামূল্যে।
যদিও বেশিরভাগ পরিস্থিতিতে অ্যারের মতো একটি বিদ্যমান বস্তুতে ফিরে যেতে এবং #each
কল করার অনুমতি দেয় এর উপর পদ্ধতি, আসুন একটি উদাহরণ দেখি যেখানে আমাদের এটিকে স্ক্র্যাচ থেকে লিখতে হবে। এই উদাহরণে, আমরা #each
প্রয়োগ করব একটি লিঙ্ক করা তালিকাতে এটিকে গণনাযোগ্য করতে।
লিঙ্ক করা তালিকা:অ্যারে ছাড়া তালিকা
একটি লিঙ্ক করা তালিকা হল ডেটা উপাদানগুলির একটি সংগ্রহ, যেখানে প্রতিটি উপাদান পরের দিকে নির্দেশ করে। তালিকার প্রতিটি উপাদানের দুটি মান রয়েছে, যার নাম head এবং লেজ . মাথাটি উপাদানটির মান ধরে রাখে এবং লেজটি তালিকার বাকি অংশের একটি লিঙ্ক৷
[42, [12, [73, nil]]
তিনটি মান (42, 12 এবং 73) সহ একটি লিঙ্কযুক্ত তালিকার জন্য, প্রথম উপাদানটির মাথাটি 42 এবং লেজটি দ্বিতীয় উপাদানটির একটি লিঙ্ক। দ্বিতীয় উপাদানটির মাথাটি 12টি, এবং লেজটি তৃতীয় উপাদানটি ধারণ করে। তৃতীয় উপাদানটির মাথা 73, এবং লেজটি হল nil
, যা তালিকার শেষ নির্দেশ করে।
রুবিতে, @head
নামে দুটি ইনস্ট্যান্স ভেরিয়েবল ধারণ করে এমন একটি ক্লাস ব্যবহার করে একটি লিঙ্ক করা তালিকা তৈরি করা যেতে পারে। এবং @tail
.
class LinkedList
def initialize(head, tail = nil)
@head, @tail = head, tail
end
def <<(item)
LinkedList.new(item, self)
end
def inspect
[@head, @tail].inspect
end
end
#<<
পদ্ধতিটি তালিকায় নতুন মান যোগ করার জন্য ব্যবহৃত হয়, যা হেড হিসাবে পাস করা মান সহ একটি নতুন তালিকা ফেরত দিয়ে কাজ করে এবং পূর্ববর্তী তালিকাটিকে লেজ হিসাবে দেয়।
এই উদাহরণে, #inspect
মেথড যোগ করা হয়েছে যাতে আমরা তালিকায় দেখতে পারি যে এতে কোন উপাদান রয়েছে।
irb> LinkedList.new(73) << 12 << 42
=> [42, [12, [73, nil]]]
এখন যেহেতু আমাদের একটি লিঙ্ক করা তালিকা আছে, চলুন #each
প্রয়োগ করি চালু কর. #each
ফাংশন একটি ব্লক নেয় এবং বস্তুর প্রতিটি মানের জন্য এটি কার্যকর করে। আমাদের লিঙ্ক করা তালিকায় এটি বাস্তবায়ন করার সময়, আমরা তালিকার @head
-এ পাস করা ব্লককে কল করে আমাদের সুবিধার জন্য তালিকার পুনরাবৃত্তিমূলক প্রকৃতি ব্যবহার করতে পারি। , এবং #each
কল করা হচ্ছে @tail
-এ , যদি এটি বিদ্যমান থাকে।
class LinkedList
def initialize(head, tail = nil)
@head, @tail = head, tail
end
def <<(item)
LinkedList.new(item, self)
end
def inspect
[@head, @tail].inspect
end
def each(&block)
block.call(@head)
@tail.each(&block) if @tail
end
end
#each
কল করার সময় আমাদের লিঙ্ক করা তালিকার একটি উদাহরণে, এটি বর্তমান @head
সহ পাস করা ব্লককে কল করে . তারপর, এটি @tail
-এ লিঙ্ক করা তালিকার প্রতিটিকে কল করে যদি না লেজটি nil
হয় .
irb> list = LinkedList.new(73) << 12 << 42
=> [42, [12, [73, nil]]]
irb> list.each { |item| puts item }
42
12
73
=> nil
এখন আমাদের লিঙ্ক করা তালিকা #each
-এ সাড়া দেয় , আমরা include Enumberable
করতে পারি আমাদের তালিকাকে গণনাযোগ্য করতে।
class LinkedList
include Enumerable
def initialize(head, tail = nil)
@head, @tail = head, tail
end
def <<(item)
LinkedList.new(item, self)
end
def inspect
[@head, @tail].inspect
end
def each(&block)
block.call(@head)
@tail.each(&block) if @tail
end
end
irb> list = LinkedList.new(73) << 12 << 42
=> [42, [12, [73, nil]]]
irb> list.count
=> 3
irb> list.max
=> 73
irb> list.map { |item| item * item }
=> [1764, 144, 5329]
irb> list.select(&:even?)
=> [42, 12]
প্রত্যাবর্তন Enumerator
দৃষ্টান্ত
আমরা এখন আমাদের লিঙ্ক করা তালিকার সমস্ত মান লুপ করতে পারি, তবে আমরা এখনও গণনাযোগ্য ফাংশনগুলিকে চেইন করতে পারি না। এটি করার জন্য, আমাদের একটি Enumerator
ফেরত দিতে হবে উদাহরণ যখন আমাদের #each
ব্লক ছাড়াই ফাংশন বলা হয়।
class LinkedList
include Enumerable
def initialize(head, tail = nil)
@head, @tail = head, tail
end
def <<(item)
LinkedList.new(item, self)
end
def inspect
[@head, @tail].inspect
end
def each(&block)
if block_given?
block.call(@head)
@tail.each(&block) if @tail
else
to_enum(:each)
end
end
end
একটি গণনাকারীতে একটি বস্তু মোড়ানোর জন্য, আমরা #to_enum
কল করি এর উপর পদ্ধতি। আমরা :each
পাস করি , যেহেতু এই পদ্ধতিটি গণনাকারীর অভ্যন্তরীণভাবে ব্যবহার করা উচিত।
এখন, আমাদের #each
কল করা হচ্ছে একটি ব্লক ছাড়া পদ্ধতি আমাদের চেইন গণনা করার অনুমতি দেবে৷
irb> list = LinkedList.new(73) << 12 << 42
=> [42, [12, [73, nil]]]
irb> list.each
=> #<Enumerator: [42, [12, [73, nil]]]:each>
irb> list.map.with_index.to_h
=> {42=>0, 12=>1, 73=>2}
কোডের নয়টি লাইন এবং একটি অন্তর্ভুক্ত
#each
প্রয়োগ করে Enumerable
ব্যবহার করে মডিউল এবং ফেরত Enumerator
আমাদের নিজস্ব বস্তু, আমরা নয়টি লাইন কোড এবং একটি অন্তর্ভুক্ত করে আমাদের লিঙ্ক করা তালিকাকে সুপারচার্জ করতে সক্ষম হয়েছি।
এটি রুবিতে গণনার আমাদের ওভারভিউ শেষ করে। আমরা জানতে চাই যে আপনি এই নিবন্ধটি সম্পর্কে কি ভেবেছেন, বা যদি আপনার কোন প্রশ্ন থাকে। আমরা সর্বদা তদন্ত এবং ব্যাখ্যা করার জন্য বিষয়গুলির সন্ধানে থাকি, তাই রুবিতে যদি যাদুকর কিছু থাকে যা আপনি পড়তে চান, তাহলে এখনই @AppSignal-এ আমাদের জানাতে দ্বিধা করবেন না!