ऑब्जेक्ट ओरिएंटेड पायथन - उन्नत सुविधाएँ
इसमें हम कुछ उन्नत सुविधाओं पर ध्यान देंगे जो पायथन प्रदान करती हैं
हमारी कक्षा डिजाइन में कोर सिंटैक्स
इसमें हम इस बात पर ध्यान देंगे कि पायथन हमें अपनी कक्षाओं में ऑपरेटरों का लाभ उठाने की अनुमति कैसे देता है। अजगर बड़े पैमाने पर वस्तुओं और तरीकों से वस्तुओं पर कॉल करता है और यह तब भी चलता है जब इसके कुछ सुविधाजनक सिंटैक्स द्वारा छिपाया जाता है।
>>> var1 = 'Hello'
>>> var2 = ' World!'
>>> var1 + var2
'Hello World!'
>>>
>>> var1.__add__(var2)
'Hello World!'
>>> num1 = 45
>>> num2 = 60
>>> num1.__add__(num2)
105
>>> var3 = ['a', 'b']
>>> var4 = ['hello', ' John']
>>> var3.__add__(var4)
['a', 'b', 'hello', ' John']
इसलिए अगर हमें अपनी कक्षाओं में जादू पद्धति __add__ को जोड़ना है, तो क्या हम भी ऐसा कर सकते हैं। आइए करने की कोशिश करते हैं।
हमारे पास एक वर्ग है जिसका नाम सुमलिस्ट है जिसमें एक कंस्ट्रक्टर __init__ है जो एक तर्क के रूप में सूची लेता है जिसे my_list कहा जाता है।
class SumList(object):
def __init__(self, my_list):
self.mylist = my_list
def __add__(self, other):
new_list = [ x + y for x, y in zip(self.mylist, other.mylist)]
return SumList(new_list)
def __repr__(self):
return str(self.mylist)
aa = SumList([3,6, 9, 12, 15])
bb = SumList([100, 200, 300, 400, 500])
cc = aa + bb # aa.__add__(bb)
print(cc) # should gives us a list ([103, 206, 309, 412, 515])
उत्पादन
[103, 206, 309, 412, 515]
लेकिन कई तरीके हैं जो आंतरिक रूप से दूसरों के जादू के तरीकों द्वारा प्रबंधित किए जाते हैं। नीचे उनमें से कुछ हैं,
'abc' in var # var.__contains__('abc')
var == 'abc' # var.__eq__('abc')
var[1] # var.__getitem__(1)
var[1:3] # var.__getslice__(1, 3)
len(var) # var.__len__()
print(var) # var.__repr__()
अंतर्निहित प्रकार से इनहेरिट करना
कक्षाएं अंतर्निहित प्रकारों से भी इनहेरिट की जा सकती हैं, इसका मतलब किसी भी अंतर्निहित से विरासत में मिला है और वहां पाई गई सभी कार्यक्षमता का लाभ उठाएं।
नीचे दिए गए उदाहरण में हम शब्दकोश से विरासत में प्राप्त कर रहे हैं लेकिन फिर हम इसकी एक विधि __setitem__ को लागू कर रहे हैं। जब हम शब्दकोश में कुंजी और मान सेट करते हैं तो यह (सेटिटेम) लागू किया जाता है। जैसा कि यह एक जादू की विधि है, इसे संक्षेप में कहा जाएगा।
class MyDict(dict):
def __setitem__(self, key, val):
print('setting a key and value!')
dict.__setitem__(self, key, val)
dd = MyDict()
dd['a'] = 10
dd['b'] = 20
for key in dd.keys():
print('{0} = {1}'.format(key, dd[key]))
उत्पादन
setting a key and value!
setting a key and value!
a = 10
b = 20
चलिए अपने पिछले उदाहरण को बढ़ाते हैं, नीचे हमने दो मैजिक विधियों को बुलाया है जिन्हें __getitem__ और __setitem__ बेहतर कहा जाता है जब हम सूची सूचकांक के साथ काम करते हैं।
# Mylist inherits from 'list' object but indexes from 1 instead for 0!
class Mylist(list): # inherits from list
def __getitem__(self, index):
if index == 0:
raise IndexError
if index > 0:
index = index - 1
return list.__getitem__(self, index) # this method is called when
# we access a value with subscript like x[1]
def __setitem__(self, index, value):
if index == 0:
raise IndexError
if index > 0:
index = index - 1
list.__setitem__(self, index, value)
x = Mylist(['a', 'b', 'c']) # __init__() inherited from builtin list
print(x) # __repr__() inherited from builtin list
x.append('HELLO'); # append() inherited from builtin list
print(x[1]) # 'a' (Mylist.__getitem__ cutomizes list superclass
# method. index is 1, but reflects 0!
print (x[4]) # 'HELLO' (index is 4 but reflects 3!
उत्पादन
['a', 'b', 'c']
a
HELLO
ऊपर के उदाहरण में, हमने Mylist में एक तीन आइटम सूची निर्धारित की है और अंतर्निहित __init__ विधि को कहा जाता है और जब हम तत्व x को प्रिंट करते हैं, तो हम तीन आइटम सूची (['ए', 'बी', 'सी']) प्राप्त करते हैं। फिर हम इस सूची में एक और तत्व जोड़ते हैं। बाद में हम सूचकांक 1 और सूचकांक 4 के लिए पूछते हैं। लेकिन यदि आप आउटपुट देखते हैं, तो हम (इंडेक्स -1) से वह तत्व प्राप्त कर रहे हैं जो हमने मांगा है। जैसा कि हम जानते हैं कि सूची अनुक्रमण 0 से शुरू होता है, लेकिन यहाँ अनुक्रमण 1 से शुरू होता है (इसीलिए हमें सूची का पहला आइटम मिल रहा है)।
नामकरण की परंपरा
इसमें हम उन नामों पर गौर करेंगे जिनका उपयोग हम विशेष रूप से निजी चर और सम्मेलनों के लिए करेंगे, जिसका उपयोग दुनिया भर में पायथन प्रोग्रामर्स द्वारा किया जाता है। हालांकि चर को निजी के रूप में नामित किया गया है, लेकिन पायथन में गोपनीयता नहीं है और यह डिजाइन द्वारा है। किसी भी अन्य अच्छी तरह से प्रलेखित भाषाओं की तरह, पायथन का नामकरण और शैली परंपराएं हैं जो इसे बढ़ावा देती हैं हालांकि यह उन्हें लागू नहीं करता है। वहाँ एक शैली गाइड द्वारा लिखित "Guido van Rossum” the originator of Python, that describe the best practices and use of name and is called PEP8. Here is the link for this, https://www.python.org/dev/peps/pep-0008/
PEP पायथन वृद्धि प्रस्ताव के लिए खड़ा है और प्रस्तावित परिवर्तनों पर चर्चा करने के लिए पायथन समुदाय के बीच वितरित प्रलेखन की एक श्रृंखला है। उदाहरण के लिए यह सभी की सिफारिश की है,
- मॉड्यूल नाम - all_lower_case
- वर्ग के नाम और अपवाद के नाम - CamelCase
- वैश्विक और स्थानीय नाम - all_lower_case
- कार्य और विधि के नाम - all_lower_case
- लगातार - ALL_UPPER_CASE
ये सिर्फ सिफारिश हैं, आप चाहें तो अलग-अलग हो सकते हैं। लेकिन जैसा कि अधिकांश डेवलपर्स इन अनुशंसाओं का पालन करते हैं, इसलिए हो सकता है कि मेरा कोड कम पठनीय हो।
सम्मेलन के अनुरूप क्यों?
हम पीईपी की सिफारिश का पालन कर सकते हैं, हम इसे प्राप्त करने की अनुमति देते हैं,
- डेवलपर्स के विशाल बहुमत के लिए अधिक परिचित
- आपके कोड के अधिकांश पाठकों के लिए क्लीयर।
- अन्य योगदानकर्ताओं की शैली से मेल खाएगा जो समान कोड आधार पर काम करते हैं।
- एक पेशेवर सॉफ्टवेयर डेवलपर्स का चिह्न
- हर कोई आपको स्वीकार करेगा।
परिवर्तनीय नामकरण - 'सार्वजनिक' और 'निजी'
पायथन में, जब हम मॉड्यूल और कक्षाओं के साथ काम कर रहे होते हैं, हम कुछ चर या विशेषता को निजी के रूप में नामित करते हैं। पायथन में, "निजी" उदाहरण चर का कोई अस्तित्व नहीं है जिसे किसी वस्तु के अंदर तक नहीं पहुँचा जा सकता है। निजी बस का मतलब है कि वे कोड के उपयोगकर्ताओं द्वारा उपयोग किए जाने के बजाय केवल आंतरिक रूप से उपयोग करने के लिए अभिप्रेत नहीं हैं। सामान्य तौर पर, अधिकांश पायथन डेवलपर्स द्वारा एक सम्मेलन का अनुसरण किया जा रहा है, उदाहरण के लिए एक अंडरस्कोर के साथ उपसर्ग नाम। _attrval (नीचे उदाहरण) को एपीआई या किसी पायथन कोड के गैर-सार्वजनिक हिस्से के रूप में माना जाना चाहिए, चाहे वह एक फ़ंक्शन, एक विधि या डेटा सदस्य हो। नीचे दिया गया नामकरण सम्मेलन है,
सार्वजनिक विशेषताएँ या चर (इस मॉड्यूल या इस वर्ग के उपयोगकर्ता के आयातक द्वारा उपयोग किए जाने का इरादा) -regular_lower_case
निजी विशेषताएँ या चर (मॉड्यूल या वर्ग द्वारा आंतरिक उपयोग) -_single_leading_underscore
निजी विशेषताएँ जिन्हें उप-वर्गित नहीं किया जाना चाहिए -__double_leading_underscore
जादू की विशेषताएँ -__double_underscores__(उनका उपयोग करें, उन्हें न बनाएं)
class GetSet(object):
instance_count = 0 # public
__mangled_name = 'no privacy!' # special variable
def __init__(self, value):
self._attrval = value # _attrval is for internal use only
GetSet.instance_count += 1
@property
def var(self):
print('Getting the "var" attribute')
return self._attrval
@var.setter
def var(self, value):
print('setting the "var" attribute')
self._attrval = value
@var.deleter
def var(self):
print('deleting the "var" attribute')
self._attrval = None
cc = GetSet(5)
cc.var = 10 # public name
print(cc._attrval)
print(cc._GetSet__mangled_name)
उत्पादन
setting the "var" attribute
10
no privacy!