रूबी - ऑब्जेक्ट ओरिएंटेड

रूबी एक शुद्ध वस्तु-उन्मुख भाषा है और रूबी को एक वस्तु के रूप में सब कुछ दिखाई देता है। रूबी में हर मूल्य एक वस्तु है, यहां तक ​​कि सबसे आदिम चीजें: तार, संख्या और यहां तक ​​कि सच और गलत। यहां तक ​​कि एक वर्ग ही एक वस्तु है जो वर्ग वर्ग का एक उदाहरण है । यह अध्याय आपको ऑब्जेक्ट ओरिएंटेड रूबी से संबंधित सभी प्रमुख कार्यात्मकताओं के माध्यम से ले जाएगा।

किसी ऑब्जेक्ट के रूप को निर्दिष्ट करने के लिए एक वर्ग का उपयोग किया जाता है और यह उस डेटा को एक साफ पैकेज में हेरफेर करने के लिए डेटा प्रतिनिधित्व और विधियों को जोड़ता है। किसी वर्ग के भीतर के डेटा और विधियों को कक्षा के सदस्य कहा जाता है।

रूबी क्लास परिभाषा

जब आप एक वर्ग को परिभाषित करते हैं, तो आप डेटा प्रकार के लिए एक खाका परिभाषित करते हैं। यह वास्तव में किसी भी डेटा को परिभाषित नहीं करता है, लेकिन यह परिभाषित करता है कि वर्ग नाम का अर्थ क्या है, अर्थात, कक्षा की कोई वस्तु क्या होगी और इस तरह के ऑब्जेक्ट पर कौन से ऑपरेशन किए जा सकते हैं।

कीवर्ड के साथ एक वर्ग परिभाषा शुरू होती है class इसके बाद class name और एक के साथ सीमांकित है end। उदाहरण के लिए, हमने बॉक्स क्लास को कीवर्ड क्लास का उपयोग करते हुए परिभाषित किया है -

class Box
   code
end

नाम को एक कैपिटल लेटर से शुरू करना चाहिए और कन्वेंशन के नाम जिसमें एक से अधिक शब्द होते हैं, को एक-एक शब्द के साथ-साथ चलाया जाता है और कोई अलग अक्षर (CamelCase) नहीं होता है।

रूबी वस्तुओं को परिभाषित करें

एक वर्ग वस्तुओं के लिए ब्लूप्रिंट प्रदान करता है, इसलिए मूल रूप से एक वर्ग से एक वस्तु बनाई जाती है। हम एक वर्ग की वस्तुओं की घोषणा करते हैंnewकीवर्ड। निम्नलिखित कथन कक्षा बॉक्स की दो वस्तुओं की घोषणा करते हैं -

box1 = Box.new
box2 = Box.new

प्रारंभिक विधि

initialize method एक मानक रूबी वर्ग विधि है और लगभग उसी तरह से काम करती है constructorअन्य वस्तु उन्मुख प्रोग्रामिंग भाषाओं में काम करता है। प्रारंभिक विधि तब उपयोगी होती है जब आप ऑब्जेक्ट निर्माण के समय कुछ वर्ग चर को इनिशियलाइज़ करना चाहते हैं। यह विधि मापदंडों की एक सूची ले सकती है और किसी भी अन्य रूबी विधि की तरह इसके द्वारा पूर्ववर्ती होगीdef नीचे दिया गया कीवर्ड -

class Box
   def initialize(w,h)
      @width, @height = w, h
   end
end

चर चर

instance variablesएक तरह के वर्ग गुण होते हैं और वे वस्तुओं के गुण बन जाते हैं जब एक बार कक्षा का उपयोग करके वस्तुएं बनाई जाती हैं। प्रत्येक ऑब्जेक्ट की विशेषताओं को व्यक्तिगत रूप से सौंपा गया है और अन्य वस्तुओं के साथ कोई मूल्य साझा नहीं करता है। उन्हें कक्षा के भीतर @ ऑपरेटर का उपयोग करके एक्सेस किया जाता है लेकिन हम जिस कक्षा का उपयोग करते हैं, उसके बाहर उन्हें एक्सेस करने के लिएpublic विधियाँ, जिन्हें कहा जाता है accessor methods। यदि हम उपरोक्त परिभाषित वर्ग लेते हैंBox तब @ Recharge और @height क्लास बॉक्स के लिए उदाहरण चर हैं।

class Box
   def initialize(w,h)
      # assign instance variables
      @width, @height = w, h
   end
end

एक्सेसर और सेटर मेथड्स

वर्ग के बाहर से उपलब्ध चर बनाने के लिए, उन्हें भीतर परिभाषित किया जाना चाहिए accessor methods, इन अभिगम विधियों को गेट्टर विधियों के रूप में भी जाना जाता है। निम्नलिखित उदाहरण एक्सेसर विधियों के उपयोग को दर्शाता है -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def printWidth
      @width
   end

   def printHeight
      @height
   end
end

# create an object
box = Box.new(10, 20)

# use accessor methods
x = box.printWidth()
y = box.printHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Width of the box is : 10
Height of the box is : 20

एक्सेसर विधियों के समान, जिनका उपयोग चर के मूल्य तक पहुंचने के लिए किया जाता है, रूबी कक्षा के बाहर से उन चर के मूल्यों को निर्धारित करने का एक तरीका प्रदान करता है setter methods, जो नीचे के रूप में परिभाषित किए गए हैं -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end

   # setter methods
   def setWidth=(value)
      @width = value
   end
   def setHeight=(value)
      @height = value
   end
end

# create an object
box = Box.new(10, 20)

# use setter methods
box.setWidth = 30
box.setHeight = 50

# use accessor methods
x = box.getWidth()
y = box.getHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Width of the box is : 30
Height of the box is : 50

उदाहरण के तरीके

instance methods हम उसी तरह से परिभाषित होते हैं जैसे हम किसी अन्य विधि का उपयोग करके परिभाषित करते हैं defकीवर्ड और उन्हें क्लास उदाहरण का उपयोग करके केवल नीचे दिखाए अनुसार उपयोग किया जा सकता है। उनकी कार्यक्षमता उदाहरण चर तक पहुँचने के लिए सीमित नहीं है, लेकिन यह भी वे अपनी आवश्यकता के अनुसार बहुत अधिक कर सकते हैं।

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Area of the box is : 200

कक्षा के तरीके और चर

class variablesएक चर है, जो एक वर्ग के सभी उदाहरणों के बीच साझा किया जाता है। दूसरे शब्दों में, चर का एक उदाहरण है और इसे ऑब्जेक्ट इंस्टेंस द्वारा एक्सेस किया जाता है। वर्ग चर दो @ वर्ण (@@) के साथ उपसर्ग किए जाते हैं। एक वर्ग चर को वर्ग परिभाषा के भीतर आरंभीकृत किया जाना चाहिए जैसा कि नीचे दिखाया गया है।

एक वर्ग विधि का उपयोग करके परिभाषित किया गया है def self.methodname(), जो अंत परिसीमन के साथ समाप्त होता है और वर्ग नाम का उपयोग करके बुलाया जाएगा classname.methodname जैसा कि निम्नलिखित उदाहरण में दिखाया गया है -

#!/usr/bin/ruby -w

class Box
   # Initialize our class variables
   @@count = 0
   def initialize(w,h)
      # assign instance avriables
      @width, @height = w, h

      @@count += 1
   end

   def self.printCount()
      puts "Box count is : #@@count"
   end
end

# create two object
box1 = Box.new(10, 20)
box2 = Box.new(30, 100)

# call class method to print box count
Box.printCount()

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Box count is : 2

To_s विधि

आपके द्वारा परिभाषित किसी भी वर्ग में एक होना चाहिए to_sवस्तु की एक स्ट्रिंग प्रतिनिधित्व वापस करने के लिए उदाहरण विधि। चौड़ाई और ऊंचाई के संदर्भ में एक बॉक्स ऑब्जेक्ट का प्रतिनिधित्व करने के लिए एक सरल उदाहरण निम्नलिखित है -

#!/usr/bin/ruby -w

class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # define to_s method
   def to_s
      "(w:#@width,h:#@height)"  # string formatting of the object.
   end
end

# create an object
box = Box.new(10, 20)

# to_s method will be called in reference of string automatically.
puts "String representation of box is : #{box}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

String representation of box is : (w:10,h:20)

पहुँच नियंत्रण

रूबी आपको उदाहरण के स्तर के स्तर पर सुरक्षा के तीन स्तर देता है, जो हो सकता है public, private, or protected। रूबी उदाहरण और वर्ग चर पर कोई अभिगम नियंत्रण लागू नहीं करती है।

  • Public Methods- सार्वजनिक तरीकों से किसी को भी बुलाया जा सकता है। प्रारंभिक रूप को छोड़कर, विधि डिफ़ॉल्ट रूप से सार्वजनिक होती है, जो हमेशा निजी होती है।

  • Private Methods- निजी तरीकों को एक्सेस नहीं किया जा सकता है, या क्लास के बाहर से भी देखा जा सकता है। केवल कक्षा के तरीके निजी सदस्यों तक पहुंच सकते हैं।

  • Protected Methods- एक संरक्षित पद्धति को केवल परिभाषित वर्ग और उसके उपवर्गों की वस्तुओं द्वारा ही लागू किया जा सकता है। परिवार के भीतर पहुंच रखी जाती है।

तीनों पहुँच संशोधक के वाक्य विन्यास को दिखाने के लिए एक सरल उदाहरण निम्नलिखित है -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # instance method by default it is public
   def getArea
      getWidth() * getHeight
   end

   # define private accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end
   # make them private
   private :getWidth, :getHeight

   # instance method to print area
   def printArea
      @area = getWidth() * getHeight
      puts "Big box area is : #@area"
   end
   # make it protected
   protected :printArea
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"

# try to call protected or methods
box.printArea()

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है। यहां, पहली विधि को सफलतापूर्वक कहा जाता है लेकिन दूसरी विधि ने एक समस्या दी।

Area of the box is : 200
test.rb:42: protected method `printArea' called for #
<Box:0xb7f11280 @height = 20, @width = 10> (NoMethodError)

कक्षा में प्रवेश

ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग में सबसे महत्वपूर्ण अवधारणाओं में से एक विरासत की है। वंशानुक्रम हमें एक वर्ग को दूसरी कक्षा के संदर्भ में परिभाषित करने की अनुमति देता है, जिससे एप्लिकेशन बनाना और उसे बनाए रखना आसान हो जाता है।

वंशानुक्रम भी कोड कार्यक्षमता और तेजी से कार्यान्वयन के समय का पुन: उपयोग करने का अवसर प्रदान करता है लेकिन दुर्भाग्य से रूबी विरासत के कई स्तरों का समर्थन नहीं करता है लेकिन रूबी का समर्थन करता है mixins। एक मिक्सिन मल्टीपल इनहेरिटेंस के एक विशेष कार्यान्वयन की तरह है जिसमें केवल इंटरफ़ेस भाग विरासत में मिला है।

कक्षा बनाते समय, पूरी तरह से नए डेटा सदस्यों और सदस्य कार्यों को लिखने के बजाय, प्रोग्रामर यह नामित कर सकता है कि नए वर्ग को किसी मौजूदा वर्ग के सदस्यों को विरासत में प्राप्त करना चाहिए। इस मौजूदा वर्ग को कहा जाता हैbase class or superclass, और नई कक्षा को इस रूप में संदर्भित किया जाता है derived class or sub-class

रूबी उपवर्ग की अवधारणा का भी समर्थन करता है, अर्थात, विरासत और उदाहरण के बाद अवधारणा की व्याख्या करता है। एक वर्ग का विस्तार करने के लिए वाक्य रचना सरल है। बस अपने वर्ग विवरण में एक <चरित्र और सुपरक्लास का नाम जोड़ें। उदाहरण के लिए, निम्नलिखित बॉक्स के उपवर्ग के रूप में एक वर्ग बिगबॉक्स को परिभाषित करता है -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# define a subclass
class BigBox < Box

   # add a new instance method
   def printArea
      @area = @width * @height
      puts "Big box area is : #@area"
   end
end

# create an object
box = BigBox.new(10, 20)

# print the area
box.printArea()

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Big box area is : 200

तरीके ओवरराइडिंग

यद्यपि आप एक व्युत्पन्न वर्ग में नई कार्यक्षमता जोड़ सकते हैं, लेकिन कभी-कभी आप मूल श्रेणी में पहले से परिभाषित पद्धति के व्यवहार को बदलना चाहेंगे। आप विधि नाम को समान रखकर और उदाहरण में नीचे दिखाए गए तरीके की कार्यक्षमता को ओवरराइड करके ऐसा कर सकते हैं -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# define a subclass
class BigBox < Box

   # change existing getArea method as follows
   def getArea
      @area = @width * @height
      puts "Big box area is : #@area"
   end
end

# create an object
box = BigBox.new(10, 20)

# print the area using overriden method.
box.getArea()

ऑपरेटर ओवरलोडिंग

हम + ऑपरेटर का उपयोग करके दो बॉक्स ऑब्जेक्ट्स के वेक्टर जोड़ के प्रदर्शन के लिए +, * ऑपरेटर को एक बॉक्स चौड़ाई और ऊंचाई को स्केलर द्वारा गुणा करने के लिए, और एकरी - ऑपरेटर बॉक्स की चौड़ाई और ऊंचाई को नकारने के लिए चाहेंगे। यहाँ गणितीय संचालकों के साथ बॉक्स वर्ग का एक संस्करण परिभाषित किया गया है -

class Box
   def initialize(w,h)     # Initialize the width and height
      @width,@height = w, h
   end

   def +(other)       # Define + to do vector addition
      Box.new(@width + other.width, @height + other.height)
   end

   def -@           # Define unary minus to negate width and height
      Box.new(-@width, -@height)
   end

   def *(scalar)           # To perform scalar multiplication
      Box.new(@width*scalar, @height*scalar)
   end
end

बर्फ़ीली वस्तुएँ

कभी-कभी, हम किसी ऑब्जेक्ट को परिवर्तित होने से रोकना चाहते हैं। ऑब्जेक्ट में फ्रीज विधि हमें ऐसा करने की अनुमति देती है, प्रभावी रूप से एक वस्तु को स्थिर में बदल देती है। किसी भी वस्तु को तपाने से जम सकता हैObject.freeze। एक जमे हुए ऑब्जेक्ट को संशोधित नहीं किया जा सकता है: आप इसके उदाहरण चर नहीं बदल सकते।

आप जांच सकते हैं कि कोई दी गई वस्तु पहले से जमी हुई है या नहीं Object.frozen?विधि, जो वस्तु के जम जाने पर सही हो जाती है अन्यथा एक गलत मान वापस आ जाता है। निम्नलिखित उदाहरण अवधारणा को साफ करता है -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end

   # setter methods
   def setWidth=(value)
      @width = value
   end
   def setHeight=(value)
      @height = value
   end
end

# create an object
box = Box.new(10, 20)

# let us freez this object
box.freeze
if( box.frozen? )
   puts "Box object is frozen object"
else
   puts "Box object is normal object"
end

# now try using setter methods
box.setWidth = 30
box.setHeight = 50

# use accessor methods
x = box.getWidth()
y = box.getHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Box object is frozen object
test.rb:20:in `setWidth=': can't modify frozen object (TypeError)
   from test.rb:39

क्लास कांस्टेंट

आप एक चर के लिए एक प्रत्यक्ष संख्यात्मक या स्ट्रिंग मान निर्दिष्ट करके एक वर्ग के अंदर एक निरंतर परिभाषित कर सकते हैं, जो कि @ या @@ का उपयोग किए बिना परिभाषित किया गया है। अधिवेशन के द्वारा, हम ऊपरी मामले में निरंतर नाम रखते हैं।

एक बार एक स्थिरांक परिभाषित हो जाने के बाद, आप इसका मान नहीं बदल सकते हैं लेकिन आप किसी वर्ग जैसे चर के अंदर एक निरंतर का उपयोग कर सकते हैं लेकिन यदि आप कक्षा के बाहर किसी स्थिरांक तक पहुँचना चाहते हैं तो आपको इसका उपयोग करना होगा classname::constant जैसा कि नीचे दिए गए उदाहरण में दिखाया गया है।

#!/usr/bin/ruby -w

# define a class
class Box
   BOX_COMPANY = "TATA Inc"
   BOXWEIGHT = 10
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"
puts Box::BOX_COMPANY
puts "Box weight is: #{Box::BOXWEIGHT}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Area of the box is : 200
TATA Inc
Box weight is: 10

क्लास के स्थिरांक विरासत में मिले हैं और उदाहरण के तरीकों की तरह इसे ओवरराइड किया जा सकता है।

एलाट का उपयोग करके ऑब्जेक्ट बनाएं

ऐसी स्थिति हो सकती है जब आप किसी वस्तु को उसके निर्माता को बुलाए बिना बनाना चाहते हैं initializeअर्थात नई विधि का उपयोग करते हुए, ऐसे मामले में आप आवंटन को कॉल कर सकते हैं , जो आपके लिए एक असंगठित वस्तु बनाएगा जैसा कि निम्नलिखित उदाहरण में है -

#!/usr/bin/ruby -w

# define a class
class Box
   attr_accessor :width, :height

   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # instance method
   def getArea
      @width * @height
   end
end

# create an object using new
box1 = Box.new(10, 20)

# create another object using allocate
box2 = Box.allocate

# call instance method using box1
a = box1.getArea()
puts "Area of the box is : #{a}"

# call instance method using box2
a = box2.getArea()
puts "Area of the box is : #{a}"

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Area of the box is : 200
test.rb:14: warning: instance variable @width not initialized
test.rb:14: warning: instance variable @height not initialized
test.rb:14:in `getArea': undefined method `*' 
   for nil:NilClass (NoMethodError) from test.rb:29

कक्षा की जानकारी

यदि कक्षा की परिभाषाएं निष्पादन योग्य कोड हैं, तो इसका मतलब है कि वे किसी वस्तु के संदर्भ में निष्पादित करते हैं: स्वयं को कुछ का संदर्भ देना चाहिए। आइए जानें कि यह क्या है।

#!/usr/bin/ruby -w

class Box
   # print class information
   puts "Type of self = #{self.type}"
   puts "Name of self = #{self.name}"
end

जब उपरोक्त कोड निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Type of self = Class
Name of self = Box

इसका मतलब है कि एक वर्ग परिभाषा को वर्तमान वस्तु के रूप में उस वर्ग के साथ निष्पादित किया जाता है। इसका मतलब यह है कि विधि परिभाषा के निष्पादन के दौरान मेटाक्लास और उसके सुपरक्लास में विधियां उपलब्ध होंगी।