কম্পিউটার

মেথড_মিসিং সহ পূর্বপুরুষ চেইন উপরে

আপনার ক্যারি-অন লাগেজটি ক্লাচ করুন, কারণ আজ আমরা পূর্বপুরুষের চেইন পর্যন্ত সমস্ত পথ ভ্রমণ করব। আমরা একটি মেথড কল ফলো করব এবং দেখব কিভাবে এটি চেইনের উপরে যায় এবং সেইসাথে মেথডটি অনুপস্থিত থাকলে কি হবে তা খুঁজে বের করব। এবং যেহেতু আমরা আগুন নিয়ে খেলতে ভালবাসি, তাই আমরা সেখানে থামব না বরং আগুন নিয়ে খেলা চালিয়ে যাব BasicObject#method_missing ওভাররাইড করছে . আপনি যদি মনোযোগ দেন, তাহলে আমরা এটিকে ব্যবহারিক উদাহরণেও ব্যবহার করতে পারি। যদিও কোন গ্যারান্টি নেই। চলুন!

পূর্বপুরুষ চেইন

রুবিতে পূর্বপুরুষ চেইনের মৌলিক নিয়মগুলি দিয়ে শুরু করা যাক:

  • রুবি শুধুমাত্র একক উত্তরাধিকার সমর্থন করে
  • এটি একটি বস্তুকে মডিউলের একটি সেট অন্তর্ভুক্ত করার অনুমতি দেয়

রুবিতে, পূর্বপুরুষ শৃঙ্খলটি একটি প্রদত্ত শ্রেণীর জন্য উত্তরাধিকারসূত্রে প্রাপ্ত সমস্ত শ্রেণী এবং মডিউলগুলির ট্রাভার্সালের সমন্বয়ে গঠিত।

রুবিতে পূর্বপুরুষের চেইন কীভাবে পরিচালনা করা হয় তা দেখানোর জন্য আসুন একটি উদাহরণ দেখি।

module Auth end
 
module Session end
 
module Iterable end
 
class Collection
  prepend Iterable
end
 
class Users < Collection
  prepend Session
  include Auth
end
 
p Users.ancestors

উত্পাদন করে:

[
  Session, Users, Auth,        # Users
  Iterable, Collection,        # Collection
  Object, Kernel, BasicObject  # Ruby Object Model
]

প্রথমে, আমরা ancestors বলি প্রদত্ত ক্লাসের পূর্বপুরুষ চেইন অ্যাক্সেস করার জন্য ক্লাস পদ্ধতি।

আমরা দেখতে পাচ্ছি যে Users.ancestors কে কল করা হয়েছে ক্রমানুসারে ক্লাস এবং মডিউলগুলির একটি অ্যারে প্রদান করে:

  • Users-এর পূর্বে লেখা মডিউল
  • Users ক্লাস
  • Users ক্লাস' অন্তর্ভুক্ত মডিউল
  • Collection ক্লাসের প্রিপেন্ডেড মডিউলগুলো—ব্যবহারকারীদের সরাসরি অভিভাবক হিসেবে
  • Collection ক্লাস
  • Collection ক্লাস’ অন্তর্ভুক্ত মডিউল—কোনটিই নয়
  • Object ক্লাস—‘যেকোনো শ্রেণীর ডিফল্ট উত্তরাধিকার
  • Kernel মডিউল—Object-এ অন্তর্ভুক্ত এবং হোল্ডিং মূল পদ্ধতি
  • BasicObject ক্লাস—রুবিতে রুট ক্লাস

সুতরাং, প্রদত্ত ট্রাভার্সড ক্লাস বা মডিউলের উপস্থিতির ক্রম সর্বদা নিম্নরূপ:

  • প্রিপেন্ডেড মডিউলগুলি
  • শ্রেণী বা মডিউল
  • এর অন্তর্ভুক্ত মডিউলগুলি

পূর্বপুরুষ চেইনটি প্রধানত রুবি দ্বারা অতিক্রম করা হয় যখন একটি পদ্ধতি একটি বস্তু বা একটি শ্রেণীতে ব্যবহার করা হয়।

মেথড লুকআপ পাথ

যখন একটি বার্তা পাঠানো হয়, রুবি বার্তা গ্রহণকারীর পূর্বপুরুষের চেইনটি অতিক্রম করে এবং তাদের মধ্যে কেউ প্রদত্ত বার্তায় সাড়া দেয় কিনা তা পরীক্ষা করে।

যদি পূর্বপুরুষ চেইনের একটি প্রদত্ত শ্রেণী বা মডিউল বার্তাটির প্রতিক্রিয়া জানায়, তাহলে এই বার্তাটির সাথে যুক্ত পদ্ধতিটি কার্যকর করা হয় এবং পূর্বপুরুষ চেইন ট্রাভার্সাল বন্ধ করা হয়৷

class Collection < Array
end
 
Collection.ancestors # => [Collection, Array, Enumerable, Object, Kernel, BasicObject]
 
collection = Collection.new([:a, :b, :c])
 
collection.each_with_index # => :a

এখানে, collection.each_with_index বার্তাটি গণনাযোগ্য মডিউল দ্বারা গৃহীত হয়। তারপর Enumerable#each_with_index এই বার্তার জন্য পদ্ধতি বলা হয়৷

এখানে, যখন collection.each_with_index বলা হয়, রুবি পরীক্ষা করে যদি:

  • সংগ্রহ each_with_index message-এর উত্তর দেয় => না
  • অ্যারে each_with_index message-এ সাড়া দেয় => না
  • অগণিত each_with_index message-এর উত্তর দেয় => হ্যাঁ

সুতরাং, এখান থেকে, রুবি পূর্বপুরুষ চেইন ট্রাভার্সাল বন্ধ করে এবং এই বার্তাটির সাথে যুক্ত পদ্ধতিটিকে কল করে। আমাদের ক্ষেত্রে, Enumerable#each_with_index পদ্ধতি।

রুবিতে, এই প্রক্রিয়াটিকে মেথড লুকআপ পাথ বলা হয় .

এখন, যদি প্রদত্ত রিসিভারের পূর্বপুরুষ চেইন রচনা করে এমন কোনো ক্লাস এবং মডিউল বার্তায় সাড়া না দিলে কি হবে?

BasicObject#method_missing

সুন্দর খেলার সাথে যথেষ্ট! আসুন জিনিসগুলি ভেঙে ফেলি, বিকাশকারী শৈলী:ব্যতিক্রমগুলি ছুঁড়ে দিয়ে। আমরা একটি Collection বাস্তবায়ন করব ক্লাস এবং কল করুন একটি অজানা পদ্ধতির একটি উদাহরণে।

class Collection
end
 
c = Collection.new
c.search('item1') # => NoMethodError: undefined method `search` for #<Collection:0x123456890>

এখানে, Collection ক্লাস একটি search বাস্তবায়ন করে না পদ্ধতি তাই একটি NoMethodError উত্থাপিত হয়. কিন্তু এই ত্রুটি উত্থাপন কোথা থেকে আসে?

ত্রুটিটি BasicObject#method_missing-এ উত্থাপিত হয়েছে পদ্ধতি এই পদ্ধতিটি বলা হয় যখন পদ্ধতি সন্ধান পাথ শেষ পর্যন্ত প্রদত্ত বার্তার সাথে সামঞ্জস্যপূর্ণ কোনো পদ্ধতি খুঁজে পায় না।

ঠিক আছে... কিন্তু এই পদ্ধতিটি শুধুমাত্র একটি NoMethodError উত্থাপন করে . তাই আমাদের Collection এর প্রসঙ্গে পদ্ধতিটিকে ওভাররাইড করতে সক্ষম হওয়া দুর্দান্ত হবে ক্লাস।

BasicObject#method_missing ওভাররাইড করা পদ্ধতি

অনুমান কি? method_missing ওভাররাইড করা সম্পূর্ণ সূক্ষ্ম যেহেতু এই পদ্ধতিটিও মেথড লুকআপ পাথ এর মেকানিজমের অধীন . একটি সাধারণ পদ্ধতির সাথে একমাত্র পার্থক্য হল যে আমরা নিশ্চিত যে এই পদ্ধতিটি অন্তত একবার মেথড লুকআপ পাথ দ্বারা পাওয়া যাবে .

প্রকৃতপক্ষে, BasicObject class ——যেটি রুবি’-এর যেকোনো শ্রেণীর রুট ক্লাস—এই পদ্ধতির একটি ন্যূনতম সংস্করণ সংজ্ঞায়িত করে। ক্লাসিক রুবি ম্যাজিক, আর কি?

তাহলে আসুন আমাদের Collection class-এ এই পদ্ধতিটি ওভাররাইড করি :

class Collection
  def initialize
    @collection = {}
  end
 
  def method_missing(method_id, *args)
    if method_id[-1] == '='
      key = method_id[0..-2]
      @collection[key.to_sym] = args.first
    else
      @collection[method_id]
    end
  end
end
 
collection = Collection.new
collection.obj1 = 'value1'
collection.obj2 = 'value2'
 
collection.obj1 # => 'value1'
collection.obj2 # => 'value2'

এখানে, Collection#method_missing @collection-এর প্রতিনিধি হিসাবে কাজ করে উদাহরণস্বরূপ পরিবর্তনশীল. প্রকৃতপক্ষে, রুবি মোটামুটিভাবে অবজেক্ট ডেলিগেশন পরিচালনা করে— س.এফ:delegate লাইব্রেরি।

অনুপস্থিত পদ্ধতি যদি সেটার পদ্ধতি হয় (collection.obj1 = 'value1' ), তারপর পদ্ধতির নাম (:obj1 ) কী এবং আর্গুমেন্ট ('value1') হিসাবে ব্যবহৃত হয় ) @collection এর মান হিসাবে হ্যাশ এন্ট্রি (@collection[:obj1] = 'value1' )।

একটি HTML ট্যাগ জেনারেটর

এখন আমরা জানি কিভাবে method_missing পদ্ধতি পর্দার আড়ালে কাজ করে, আসুন একটি পুনরুৎপাদনযোগ্য ব্যবহারের ক্ষেত্রে প্রয়োগ করি।

এখানে, লক্ষ্য হল নিম্নলিখিত ডিএসএলকে সংজ্ঞায়িত করা:

HTML.p    'hello world'             # => <p>hello world</p>
HTML.div  'hello world'             # => <div>hello world</div>
HTML.h1   'hello world'             # => <h1>hello world</h1>
HTML.h2   'hello world'             # => <h2>hello world</h2>
HTML.span 'hello world'             # => <span>hello world</span>
HTML.p    "hello #{HTML.b 'world'}" # => <p>hello <b>world</b></p>

এটি করতে, আমরা HTML.method_missing বাস্তবায়ন করতে যাচ্ছি পদ্ধতি যাতে প্রতিটি HTML ট্যাগের জন্য একটি পদ্ধতি সংজ্ঞায়িত করা না হয়।

প্রথমে, আমরা একটি HTML সংজ্ঞায়িত করি মডিউল তারপর আমরা একটি method_missing সংজ্ঞায়িত করি এই মডিউলে ক্লাস পদ্ধতি:

module HTML
  def HTML.method_missing(method_id, *args, &block)
    "<#{method_id}>#{args.first}</#{method_id}>"
  end
end

আমাদের পদ্ধতি অনুপস্থিত method_id ব্যবহার করে একটি HTML ট্যাগ তৈরি করবে ——:div HTML.div এ কল করার জন্য , উদাহরণস্বরূপ।

মনে রাখবেন যে ক্লাস পদ্ধতিগুলিও মেথড লুকআপ পাথের বিষয়।

আমরা আমাদের HTML ট্যাগ জেনারেটরকে এর দ্বারা উন্নত করতে পারি:

  • নেস্টেড ট্যাগগুলি পরিচালনা করতে ব্লক আর্গুমেন্ট ব্যবহার করে
  • একক ট্যাগ পরিচালনা করা——<br/> উদাহরণস্বরূপ

কিন্তু মনে রাখবেন যে কোডের কয়েকটি লাইন দিয়ে, আমরা প্রচুর পরিমাণে HTML ট্যাগ তৈরি করতে সক্ষম।

সুতরাং, সংক্ষেপে:

method_missing ডিএসএল তৈরি করার জন্য এটি একটি ভাল এন্ট্রি পয়েন্ট যেখানে বেশিরভাগ কমান্ড চিহ্নিত প্যাটার্নের একটি সেট ভাগ করবে।

উপসংহার

আমরা রুবিতে পূর্বপুরুষ শৃঙ্খল পর্যন্ত গিয়েছিলাম এবং BasicObject#method_missing এ ঢুকেছিলাম . BasicObject#method_missing এটি রুবি হুক পদ্ধতির অংশ . এটি তাদের জীবনচক্রের কিছু নির্দিষ্ট মুহুর্তে বস্তুর সাথে যোগাযোগ করতে ব্যবহৃত হয়। অন্যান্য রুবি হুক পদ্ধতির মতো , এই হুক পদ্ধতি সাবধানে ব্যবহার করতে হবে. এবং সাবধানতার সাথে, আমরা বলতে চাচ্ছি যে এটি কখনই রুবি অবজেক্ট মডেলের আচরণগুলিকে সংশোধন করবে না —যখন আপনি এটি নিয়ে খেলছেন বা এটিতে একটি ব্লগ পোস্ট লিখছেন;-)

ভয়লা!


  1. রুবি গ্রেপ পদ্ধতি কীভাবে ব্যবহার করবেন (উদাহরণ সহ)

  2. রুবি মানচিত্র পদ্ধতি কীভাবে ব্যবহার করবেন (উদাহরণ সহ)

  3. রুবি ট্রান্সপোজ পদ্ধতিতে সারিগুলিকে কলামে পরিণত করুন

  4. মেটাপ্রোগ্রামিং এর লুকানো খরচ