रूबी - ऑब्जेक्ट ओरिएंटेड
रूबी एक शुद्ध वस्तु-उन्मुख भाषा है और रूबी को एक वस्तु के रूप में सब कुछ दिखाई देता है। रूबी में हर मूल्य एक वस्तु है, यहां तक कि सबसे आदिम चीजें: तार, संख्या और यहां तक कि सच और गलत। यहां तक कि एक वर्ग ही एक वस्तु है जो वर्ग वर्ग का एक उदाहरण है । यह अध्याय आपको ऑब्जेक्ट ओरिएंटेड रूबी से संबंधित सभी प्रमुख कार्यात्मकताओं के माध्यम से ले जाएगा।
किसी ऑब्जेक्ट के रूप को निर्दिष्ट करने के लिए एक वर्ग का उपयोग किया जाता है और यह उस डेटा को एक साफ पैकेज में हेरफेर करने के लिए डेटा प्रतिनिधित्व और विधियों को जोड़ता है। किसी वर्ग के भीतर के डेटा और विधियों को कक्षा के सदस्य कहा जाता है।
रूबी क्लास परिभाषा
जब आप एक वर्ग को परिभाषित करते हैं, तो आप डेटा प्रकार के लिए एक खाका परिभाषित करते हैं। यह वास्तव में किसी भी डेटा को परिभाषित नहीं करता है, लेकिन यह परिभाषित करता है कि वर्ग नाम का अर्थ क्या है, अर्थात, कक्षा की कोई वस्तु क्या होगी और इस तरह के ऑब्जेक्ट पर कौन से ऑपरेशन किए जा सकते हैं।
कीवर्ड के साथ एक वर्ग परिभाषा शुरू होती है 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
इसका मतलब है कि एक वर्ग परिभाषा को वर्तमान वस्तु के रूप में उस वर्ग के साथ निष्पादित किया जाता है। इसका मतलब यह है कि विधि परिभाषा के निष्पादन के दौरान मेटाक्लास और उसके सुपरक्लास में विधियां उपलब्ध होंगी।