ऑब्जेक्ट ओरिएंटेड पायथन - उन्नत सुविधाएँ

इसमें हम कुछ उन्नत सुविधाओं पर ध्यान देंगे जो पायथन प्रदान करती हैं

हमारी कक्षा डिजाइन में कोर सिंटैक्स

इसमें हम इस बात पर ध्यान देंगे कि पायथन हमें अपनी कक्षाओं में ऑपरेटरों का लाभ उठाने की अनुमति कैसे देता है। अजगर बड़े पैमाने पर वस्तुओं और तरीकों से वस्तुओं पर कॉल करता है और यह तब भी चलता है जब इसके कुछ सुविधाजनक सिंटैक्स द्वारा छिपाया जाता है।

>>> 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!