वस्तु गुणों को कैसे संशोधित करें?

Aug 18 2020

मैंने हाल ही में यूनिट परीक्षण के बारे में एक महान लेख पढ़ा है । एक बुरी विधि का एक उदाहरण था जो अच्छी तरह से डिज़ाइन नहीं किया गया है। यह इस तरह दिख रहा है

public static string GetTimeOfDay()
{
    DateTime time = DateTime.Now;
    if (time.Hour >= 0 && time.Hour < 6)
    {
        return "Night";
    }
    if (time.Hour >= 6 && time.Hour < 12)
    {
        return "Morning";
    }
    if (time.Hour >= 12 && time.Hour < 18)
    {
        return "Afternoon";
    }
    return "Evening";
}

कुछ चीजें हैं जो लेखक ने बताई हैं कि वे प्रतिमान हैं:

  1. इसे ठोस डेटा स्रोत से कसकर जोड़ा जाता है। (यह मशीन से चालू डेटाइम पढ़ता है जिस पर यह चलता है)
  2. यह एकल जिम्मेदारी सिद्धांत (एसआरपी) का उल्लंघन करता है।
  3. यह अपना काम करवाने के लिए आवश्यक जानकारी के बारे में है। डेवलपर्स को वास्तविक स्रोत कोड की प्रत्येक पंक्ति को यह समझने के लिए पढ़ना चाहिए कि छिपे हुए इनपुट का क्या उपयोग किया जाता है और वे कहां से आते हैं। विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।

मैं मुख्य रूप से पायथन में कोड करता हूं और इस लेख के बाद मुझे लगता है कि selfज्यादातर मामलों में उपयोग करने से उन बिंदुओं का भी उल्लंघन होता है।

class Car:
    def __init__(self, power):
        self.power = power
        self.speed = 0
        
    def accelerate(self, acceleration_time):
        self.speed = self.calculate_acceleration(acceleration_time, self.power)
  1. accelerate छिपा इनपुट है: self.power
  2. विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है। वहाँ छिपा हुआ उत्पादन (?) हैself.speed

यह छोटा तरीका है और इसे पढ़ना आसान है लेकिन सौ लाइनों वाली विधियों के बारे में क्या है जो selfकई स्थानों पर पढ़ती और असाइन करती हैं? यदि उन लोगों का नाम ठीक से नहीं लिया गया है, तो यह समझने के लिए कि यह क्या करता है और यहां तक ​​कि अगर उन लोगों का नाम ठीक से डेवलपर के लिए है, तो यह जानने के लिए पूरे कार्यान्वयन को पढ़ना चाहिए कि क्या यह कुछ selfसामान को संशोधित करता है , या यदि अतिरिक्त संदर्भ के साथ इंजेक्ट किया जाता है, तो बड़ी परेशानी होगी self

दूसरी ओर जब मैं selfइनपुट (आर्ग्युमेंट्स) और आउटपुट (रिटर्न वैल्यू) के साथ बिना उपयोग किए हर विधि को कोड करने की कोशिश करूंगा , तब मैं कई तरीकों के माध्यम से एक चर को समाप्त करूंगा और खुद को दोहराऊंगा।

तो इससे कैसे निपटें selfऔर इसका सही इस्तेमाल कैसे करें? यह कैसे स्पष्ट करें कि इनपुट के रूप में कौन सी विधि का उपयोग किया जाता है और यह क्या संशोधित करता है (आउटपुट)?

जवाब

6 amon Aug 18 2020 at 18:16

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

अंगूठे के एक नियम के रूप में: कुछ कोड कुछ सॉफ़्टवेयर के व्यावसायिक तर्क के करीब है, अधिक शुद्ध, अपरिवर्तनीय, कार्यात्मक-ईश, स्पष्ट और परीक्षण योग्य होना चाहिए। करीब कुछ कोड अनुप्रयोग की बाहरी परतों के लिए है, कम कार्यक्षमता है जो कि इकाई परीक्षण के लायक है ध्यान से, और इसलिए कम परीक्षण योग्य डिजाइन ठीक है। उदाहरण के लिए, कोड जो बस कुछ बाहरी एपीआई को लपेटता है, यथोचित इकाई-परीक्षण नहीं किया जा सकता है।

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

class Cat(Animal):
  def make_noise(self):
    print("meow")

यह एक अच्छा डिज़ाइन नहीं है, क्योंकि आउटपुट sys.stdoutस्ट्रीम के साथ कसकर युग्मित है। अधिक परीक्षण योग्य डिजाइनों में इसे सीधे प्रिंट करने
def noise(self): return "meow"
या किसी फ़ाइल में पास करने के बजाय एक स्ट्रिंग लौटना शामिल होगा जिसे मुद्रित किया जा सकता है
def make_noise(self, stream): print("meow", file=stream):।

आपके उदाहरण में, आपके पास एक म्यूटेशन ऑपरेशन है car.accelerate(t)। यह एक समस्या नहीं है! इस ऑपरेशन से परीक्षण क्षमता को खतरा नहीं है क्योंकि परिणाम आसानी से मुखर हो सकता है:

car = Car(10)
assert car.speed == 0
car.accelerate(5)
assert car.speed == 50

नाम accelerate()यह भी स्पष्ट रूप से स्पष्ट करता है कि यह एक परिवर्तनशील संचालन है। अन्य भाषाएं भी इसे टाइप सिस्टम (उदाहरण fn accelerate(&mut self)के लिए रस्ट) या नामकरण सम्मेलन (उदाहरण accelerate!के लिए रूबी) में एनकोड करती हैं । उत्परिवर्तन आदेशों और शुद्ध प्रश्नों के बीच अंतर रखना उपयोगी साबित होता है, भले ही यह हमेशा अभ्यास में न हो।

यदि आपके कोड में कोई समस्या है, तो ऐसा नहीं है कि गति () विधि असाइन की गई है self, लेकिन self.calculate_acceleration(time, self.power)विधि। यह विधि selfदो बार से डेटा प्राप्त करती है : एक बार जिस विधि के रूप में इसे लागू किया जाता है, दूसरे पैरामीटर के माध्यम से दूसरी बार। यह डेटा प्रवाह को तीव्र बनाता है - इस विधि के होने का कोई कारण नहीं है जब तक selfकि विधि के भीतर उत्परिवर्तित नहीं किया जाएगा। इस तरह से डिजाइन बदलना मददगार हो सकता है:

def calculate_acceleration(time, power):
  ...

class Car:
  def __init__(self, power):
    ...
        
  def accelerate(self, acceleration_time):
    self.speed = calculate_acceleration(acceleration_time, self.power)

इस विशेष मामले में परीक्षण क्षमता पर कोई वास्तविक प्रभाव नहीं है, लेकिन अन्य मामलों में अब ऑब्जेक्ट के इंटरफ़ेस के माध्यम से जाने के बिना, गणना को सीधे परीक्षण करना संभव हो सकता है। जबकि अन्य भाषाओं में निजी स्थिर सहायक विधियां सामान्य हैं, यह पायथन के लिए एक उपयुक्त दृष्टिकोण नहीं है - बस एक निशुल्क फ़ंक्शन का उपयोग करें।

तरीकों की एक संभावित आलोचना यह है कि यह स्पष्ट नहीं है कि किन क्षेत्रों में खपत होती है। उदाहरण के लिए इस प्रकार का डेटा प्रवाह bonkers होगा यह है भले ही यकीनन शिकायत "स्वच्छ कोड":

class ReallyWeirdObject:
  def __init__(self, x, y):
    self.x = x
    self.y = y
    self.z = None
    self.use_x = False

  def _helper(self):
    self.z = self.x + self.y

  def some_operation(self):
    if self.use_x:
      return self.x
    else:
      self._helper()
      return 2 * self.z

weirdo = ReallyWeirdObject(1, 2)
weirdo.use_x = True
print(weirdo.some_operation())

लेकिन इस कोड में डब्ल्यूटीएफ zका उपयोग आंतरिक परिणामों को संप्रेषित करने के लिए किया जाता है, या यह use_xएक ऐसा क्षेत्र है जब इसे वैकल्पिक कीवर्ड तर्क होने की संभावना होनी चाहिए some_operation()

समस्या यह नहीं है कि some_operation()उस वस्तु के खेतों का उपभोग किया जाता है जिस पर उसे बुलाया गया था। यह पूरे बिंदु की तरह है ... जब तक इस ऑब्जेक्ट में डेटा यथोचित रूप से छोटा और प्रबंधनीय है, तब तक ऐसे ऑपरेशन ठीक हैं। यदि आप कल्पना करना चाहते हैं, तो आप इसे "इंटरफ़ेस अलगाव सिद्धांत" का एक उदाहरण कह सकते हैं। समस्याएँ वास्तव में उन अधकचरी देव वस्तुओं के लिए पैदा होती हैं जिनमें दर्जनों क्षेत्र होते हैं।

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

4 FilipMilovanović Aug 18 2020 at 22:24

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

यहाँ संदर्भ के लिए कोड है:

public static string GetTimeOfDay()
{
    DateTime time = DateTime.Now;
    if (time.Hour >= 0 && time.Hour < 6)
    {
        return "Night";
    }
    if (time.Hour >= 6 && time.Hour < 12)
    {
        return "Morning";
    }
    if (time.Hour >= 12 && time.Hour < 18)
    {
        return "Afternoon";
    }
    return "Evening";
}

C # में, staticकीवर्ड अनिवार्य रूप से इसका मतलब है कि यह एक नि: शुल्क फ़ंक्शन है (यानी, ऑब्जेक्ट पर इंस्टेंस विधि नहीं)। यह आपके प्रश्न के संदर्भ में प्रासंगिक है, क्योंकि आप पूछते हैं कि ये विचार वस्तुओं पर कैसे लागू होते हैं

लेख के लेखक ने कई बिंदु उठाए; मुझे पहला पता 1. (कसकर तारीख प्रदान करने वाली सेवा - DateTimeवर्ग) और 3. (निर्भरता के बारे में गलतियाँ) के लिए युग्मित करें । इससे जो समस्या पैदा होती है, वह यह है कि, जबकि फ़ंक्शन उन परिस्थितियों में अच्छी तरह से काम करता है, जो मूल रूप से इसके लिए बनाई गई थी, अन्य संदर्भों में उपयोग करने योग्य नहीं है

उदाहरण के लिए, क्या होगा यदि मुझे एक यूआई का समर्थन करने की आवश्यकता है जो उपयोगकर्ताओं को भविष्य की तारीख के लिए "दिन का समय" श्रेणी देखने की अनुमति देता है (फिर से, यह "सुबह / दोपहर / शाम / रात") उदाहरण से वंचित है, लेकिन मान लीजिए कि यह कुछ व्यवसाय देता है- इसके बजाय प्रासंगिक श्रेणी, उपयोगकर्ताओं के लिए कुछ हित)।

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

दिनांक-समय को एक पैरामीटर बनाकर यह आसानी से तय हो जाता है:

public static string GetTimeOfDay(DateTime dateTime)
{
    // same code, except that it uses the dateTime param...
}

अब, SRP उल्लंघन (2 बिंदु) के बारे में - समस्या यह है कि, सार शब्दों में इसके बारे में बात करना बहुत सार्थक नहीं है। मेरे कहने का मतलब यह है कि कोड को अलगाव में देखना और "क्या अगर" परिदृश्यों का एक गुच्छा पर विचार करना बहुत सार्थक नहीं है। निश्चित रूप से, इस तरह से एसआरपी के बारे में आप कह सकते हैं कि कुछ सामान्य चीजें हैं, लेकिन अगर आप यह नहीं मानते हैं कि आपका कोड वास्तव में कैसे बदल रहा है, और वास्तविक डिजाइन की जरूरत है, तो आप व्यर्थ प्रयास की लूट और अधिकता के साथ समाप्त करेंगे। जटिल (पढ़ें "अधिक इंजीनियर") कोड।

इसका मतलब यह है कि जब आप शुरुआत में शिक्षित अनुमानों और उचित मान्यताओं के एक जोड़े के आधार पर SRP को लागू कर सकते हैं, तो आपको अपनी जिम्मेदारियों / बदलावों के बारे में अपनी डिजाइन पर पुनर्विचार करना होगा क्योंकि आपकी वास्तविक ज़िम्मेदारियों और बदलाव के पैटर्न में आपकी समझ बढ़ती है। इस कोड पर।

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

बात यह है, अगर कोडबेस का यह हिस्सा बहुत कम (या कभी नहीं) बदलता है, तो आपको वास्तव में एसआरपी पर विचार करने की आवश्यकता नहीं है। आप कई अलग-अलग कारणों से बदलाव ला सकते हैं, लेकिन यदि वे परिवर्तन कभी नहीं होते हैं, तो आपने बिना किसी लाभ के डिजाइन लागत का भुगतान किया है। जैसे, शायद लौटे हुए तार अलग-अलग भाषाओं में उपलब्ध होने चाहिए (हो सकता है कि फ़ंक्शन स्थानीयकरण का समर्थन करने के लिए कुछ शब्दकोश में एक कुंजी लौटाए)। या हो सकता है कि दिन के अलग-अलग समय के लिए सीमा मूल्य भिन्न हो सकते हैं - शायद उन्हें डेटाबेस से पढ़ा जाना चाहिए। या हो सकता है कि ये मान पूरे वर्ष बदलते रहें। या हो सकता है कि यह पूरा तर्क सार्वभौमिक न हो, इसलिए हो सकता है कि किसी प्रकार की रणनीति को फ़ंक्शन (रणनीति पैटर्न) में इंजेक्ट किया जाए। उस डिज़ाइन के बारे में क्या जो उपरोक्त सभी का समर्थन करने की आवश्यकता है?

देखें कि "क्या होगा" परिदृश्यों के एक समूह से मेरा क्या मतलब है? इसके बजाय आपको क्या करना चाहिए समस्या डोमेन और कोडबेस की समझ विकसित करना है, और SRP लागू करना है ताकि सबसे प्रमुख परिवर्तन अक्षों (परिवर्तन, जिम्मेदारियों) का अच्छी तरह से समर्थन हो।

एक सीवन की अवधारणा

इसलिए, जब आप फ़ंक्शंस या क्लासेस (या लाइब्रेरीज़ और फ्रेमवर्क, उस मामले के लिए) डिज़ाइन करते हैं, तो आप अक्सर कुछ एक्स्टेंसिबिलिटी पॉइंट्स प्रदान करते हैं - ऐसे स्थान जहाँ क्लाइंट कोड किसी चीज़ को प्लग कर सकते हैं, या अन्यथा प्रदत्त व्यवहार को रोक सकते हैं। माइकल पंख ( लिगेसी कोड के साथ प्रभावी ढंग से काम करने में ) इन "सीम" को कहते हैं - एक सीम एक ऐसी जगह है जहां आप एक साथ दो सॉफ्टवेयर जोड़ सकते हैं। डेटाइम बनाना एक पैरामीटर है एक बहुत ही सरल सीम। निर्भरता इंजेक्शन भी सीम बनाने का एक तरीका है। उदाहरण के लिए, आप किसी फ़ंक्शन या ऑब्जेक्ट को इंजेक्ट कर सकते हैं जो डेटाइम इंस्टेंस को वापस कर सकता है (यह इस विशेष उदाहरण के संदर्भ में एक ओवरकिल नहीं हो सकता है)।

वस्तुओं के बारे में क्या?

अब तक, हम एक मुफ्त फ़ंक्शन के स्तर पर चीजों पर विचार कर रहे हैं; ऑब्जेक्ट एक और संगठनात्मक स्तर प्रदान करते हैं। इसलिए अब आपको ऑब्जेक्ट को समग्र रूप से विचार करना होगा, क्योंकि ऑब्जेक्ट्स के पास सीम पेश करने के लिए अपने स्वयं के तंत्र हैं।

ऐसा करने का विशिष्ट तरीका कंस्ट्रक्टर इंजेक्शन के माध्यम से है (जैसा कि एक तैयार-से-उपयोग की जाने वाली वस्तु में यह परिणाम है) 1 । एक (पायथन) वर्ग जो ऊपर दिए गए उदाहरण कोड के बराबर होगा:

class DateTimeServices:
  def __init__(self):
    self.datetime = datetime;    # from datetime import datetime

  def get_time_of_day(self):
    now = self.datetime.now()
    if 0 <= now.hour < 6:
      return "Night"
    if 6 <= now.hour < 12:
      return "Morning"
    if 12 <= now.hour < 18:
      return "Afternoon"
    return "Evening"

यह एक ही मुद्दा है, लेकिन समस्या अब विधि ही नहीं है, यह तथ्य यह है कि वर्ग निर्माता आंतरिक रूप से डेटाइम निर्भरता बनाता है, और यह कुछ और में प्लग करने का एक स्पष्ट तरीका प्रदान नहीं करता है। इस उद्देश्य के लिए सीम में निर्मित नहीं है। कक्षा को एक अलग परिदृश्य के लिए पुन: उपयोग करना आसान नहीं है।

यहाँ एक ही क्लास है, लेकिन अब कंस्ट्रक्टर "डेटाइम प्रदाता" लेता है:

class DateTimeServices:
  def __init__(self, datetimeProvider):
    self.datetimeProvider = datetimeProvider;

  def get_time_of_day(self):
    now = self.datetimeProvider.now()
    if 0 <= now.hour < 6:
      return "Night"
    if 6 <= now.hour < 12:
      return "Morning"
    if 12 <= now.hour < 18:
      return "Afternoon"
    return "Evening"

# elsewhere:
dts = DateTimeServices(datetime)
dts.get_time_of_day()

अब आप अलग-अलग चीजों में प्लग कर सकते हैं, जब तक कि वह चीज जो datetimeProviderआवश्यक इंटरफ़ेस को संतुष्ट करने में भूमिका निभाती है (जो, इस मामले में, केवल अब () पद्धति में एक डेटाइम इंस्टेंस लौटाता है)। जैसे:

class FakeDateTimeProvider:
  def __init__(self, year, month, day, hour, minute = 0, second = 0):
    self.datetime = datetime(year, month, day, hour, minute, second)

  def now(self):
    return self.datetime

# then:
dts = DateTimeServices(FakeDateTimeProvider(2020, 8, 18, 8))
dts.get_time_of_day()

# always returns "Morning"

यह चिंताओं को 1. और 3. से पहले (चिंता के संबंध में एक ही विचार के साथ 2.) (एसआरपी) को संबोधित करता है। तो, आप देखते हैं, अपने आप selfमें समस्या नहीं है, यह वर्ग के डिजाइन के साथ अधिक करना है। जैसा कि अन्य उत्तरों में उल्लेख किया गया है, जब आप एक वर्ग (या अधिक सटीक रूप से, एक ऑब्जेक्ट) का उपयोग करते हैं, तो आप जानते हैं कि वह वस्तु वैचारिक रूप से क्या दर्शाती है, और यह आपके लिए आश्चर्य की बात नहीं है, प्रोग्रामर, कि वर्ग के पास है और अपनी आंतरिक स्थिति का उपयोग करता है।

class Car:
    def __init__(self, power):
        self.power = power
        self.speed = 0
        
    def accelerate(self, acceleration_time):
        self.speed = self.calculate_acceleration(acceleration_time, self.power)

क्लास कार की मेरी समझ से, विधि के नामकरण से, और शायद प्रलेखन से, यह मेरे लिए आश्चर्यजनक नहीं है कि accelerateउदाहरण की स्थिति को बदल देता है। यह वस्तुओं के लिए कुछ अप्रत्याशित नहीं है।

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

उस ने कहा, जो भ्रामक हो सकता है (ऊपर के प्रकाश में) यह है कि अक्सर उदाहरण के तरीकों को अपने स्वयं के मापदंडों को लेने की आवश्यकता होती है। अतिरिक्त संदर्भ जानकारी को स्वीकार करने के बारे में सोचें जो सीधे वर्ग की मुख्य जिम्मेदारी से संबंधित नहीं है। उदाहरण के लिए, यह कुछ ऐसा नहीं है जिसे आप एक बार कंस्ट्रक्टर को पास कर सकते हैं, लेकिन कुछ ऐसा जो हर कॉल पर बदल सकता है। एक क्लासिक खिलौना उदाहरण आकार (मंडलियां, त्रिकोण, आयताकार) हैं जो खुद को आकर्षित कर सकते हैं (या, आकृतियों के बजाय, ये यूआई तत्व (बटन, लेबल, आदि), या गेम इकाइयां (कह सकते हैं, 2 डी स्प्राइट्स) हो सकते हैं)। इसे करने का एक तरीका पैरामीटर रहित ड्रा () विधि है, जो सभी ड्राइंग को आंतरिक रूप से करता है। लेकिन क्या होगा यदि आप एक यूआई के पूरी तरह से अलग हिस्से में, एक अलग ड्राइंग सतह पर एक ही चीज़ खींचना चाहते हैं? या एक अलग बफर पर ताकि आप पोर्टल्स या दर्पण जैसे विशेष प्रभाव कर सकें? अधिक लचीला विकल्प ड्रॉ विधि के पैरामीटर के रूप में ड्राइंग की सतह (या किसी प्रकार की ग्राफिक्स ऑब्जेक्ट) में पास करना है।

लेकिन कई लाइनों के साथ तरीकों के बारे में क्या है जो कई स्थानों पर स्वयं को पढ़ता है और असाइन करता है?

उस कोड को लें और आग से जला दें।

यदि उन लोगों का नाम ठीक से नहीं लिया गया है, तो यह समझने के लिए कि यह क्या करता है और यहां तक ​​कि अगर उन लोगों का नाम ठीक से डेवलपर के लिए है, तो यह जानने के लिए पूरे कार्यान्वयन को पढ़ना चाहिए कि क्या यह कुछ स्व-सामान को संशोधित करता है, या यदि अतिरिक्त संदर्भ स्वयं के साथ इंजेक्शन है

हाँ। बिल्कुल सही। कोड की सैकड़ों लाइनों के साथ तरीके न लिखें।

अब, अधिक गंभीर नोट पर, कभी-कभी, आप बड़े तरीकों के साथ समाप्त करेंगे। लेकिन ज्यादातर समय, अपने कोड को छोटे तरीकों और छोटी कक्षाओं में विघटित करने का प्रयास करते हैं।

यदि आपके पास एक बड़ी विधि है जैसे कि आप जो वर्णन कर रहे हैं, वह वह है जिसके आप सिर या पूंछ नहीं बना सकते हैं, तो यह विधियाँ उन सभी प्रकार की डिज़ाइन समस्याओं से ग्रस्त हैं जिन्हें आप अपने हस्ताक्षर को बदलकर हल नहीं करने जा रहे हैं । यह इस बारे selfमें या इसके पैरामीटर के बारे में क्या नहीं है - इस पद्धति में बड़े मुद्दे हैं । आपको इसे रिफ्लेक्टर करना होगा, ऐसी चीजें ढूंढनी होंगी जो सामान्य हो, और इसे छोटे, अधिक समझने योग्य और अधिक भरोसेमंद विखंडू में विभाजित करें (ऐसी विधियाँ जो आपको उस पद्धति को समझने के लिए नहीं दिखनी चाहिए, जो उन्हें कहता है)। तुम भी पूरी तरह से विभिन्न वर्गों में उन विखंडू को समाप्त कर सकते हैं।

दूसरी ओर जब मैं इनपुट (तर्कों) और आउटपुट (रिटर्न वैल्यू) के साथ स्वयं का उपयोग किए बिना हर विधि को कोड करने की कोशिश करूंगा, तब मैं कई तरीकों के माध्यम से एक चर को समाप्त करूंगा और खुद को दोहराऊंगा।

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

यह कैसे स्पष्ट करें कि इनपुट के रूप में कौन सी विधि का उपयोग किया जाता है और यह क्या संशोधित करता है (आउटपुट)?

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


1 कभी-कभी कंस्ट्रक्टर इंजेक्शन संभव नहीं है (उदाहरण के लिए एक फ्रेमवर्क को एक पैरामीटर रहित कंस्ट्रक्टर की आवश्यकता हो सकती है), इसलिए निर्भरताएं विधियों या गुणों के माध्यम से इंजेक्ट की जाती हैं, लेकिन यह कम आदर्श है।

3 GregBurghardt Aug 18 2020 at 18:06

इस प्रकार के प्रश्नों का उत्तर आमतौर पर विधि का उपयोग करके कोड को देखकर दिया जा सकता है।

acceleration_time = 5000 # in milliseconds
car.accelerate(acceleration_time)

print(car.speed) # <-- what do you as a programmer expect the speed to be?

जबकि हम परीक्षण योग्य कोड लिखना चाहते हैं, हम इकाई परीक्षणों के बाहर कोड का उपयोग करते हैं। यूनिट परीक्षण सार्वजनिक सामना करने वाले व्यवहार को सत्यापित करते हैं। एक कक्षा के आंतरिक व्यवहार को स्पष्ट रूप से सत्यापित करने के लिए एक इकाई परीक्षण की आवश्यकता नहीं है ।

जब मैं "गति" शब्द देखता हूं तो मुझे उम्मीद है कि त्वरण पूरा होने के बाद कुछ तेज होगा। इसका तात्पर्य रनटाइम वैल्यू में बदलाव से है self.speed

इसके विपरीत एक वर्ग मॉडलिंग भौतिकी के साथ, जैसे VehicleAccelerationPhysics। मैं calculate_accelerationएक मूल्य वापस करने के लिए एक विधि की उम्मीद करूंगा , एक मूल्य को संशोधित नहीं करूंगा । लेकिन एक accelerateविधि Carमुझे आश्चर्यचकित नहीं करेगी अगर वह car.speedबदल जाती है - मैं इसे बदलने की उम्मीद करूंगा।

इसलिए आपका कोड किसी भी सर्वोत्तम प्रथाओं का उल्लंघन नहीं कर रहा है जहां तक ​​इकाई परीक्षण का संबंध है।

accelerate छिपा इनपुट है: self.power

वर्तमान मूल्य self.powerएक कार्यान्वयन विवरण है, न कि "छिपा हुआ इनपुट"। यदि इसके बजाय आप एक विशिष्ट गति में तेजी लाने की इच्छा रखते हैं, तो आपकी Carकक्षा को एक ऐसी accelerate_to_speedविधि की आवश्यकता होती है जो कार की वर्तमान शक्ति के आधार पर उचित त्वरण समय की गणना करती है।

विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।

मुझे लगता है। एक कार में तेजी आ सकती है। त्वरण के बाद गति पहले की तुलना में अधिक है। मुझे बस इतना ही पता होना चाहिए।

1 J.G. Aug 18 2020 at 18:07

मूल दृष्टिकोण यह है कि कक्षा के बाहर रहने वाले कार्यों में यथासंभव अधिक से अधिक तर्क रखें (या स्थिर हैं), फिर उन्हें उन तरीकों से कॉल करें जो एक राज्य पर निर्भर करते हैं। (इन कॉलों को अभी भी तकनीकी रूप से पारित संपत्ति को अपने हस्ताक्षर से छिपाने की आवश्यकता है, लेकिन यह ओओपी की तरह है, जो कुछ और तरीकों की जरूरत से अलग राज्य है, वे सिर्फ एक-इन-वैक्यूम कार्यों में नहीं हैं। ) मैं जो दूसरा मुख्य मुद्दा बनाना चाहता हूं, वह यह है कि अन्य मुद्दे हैं जिन्हें हमें पहले संबोधित करना चाहिए।

अपने पहले उदाहरण के साथ, यह पहले एक और चिंता को संबोधित करने के लिए इसे संपादित करने में मदद करता है, कि यह इकाई-परीक्षण के लिए कठिन है। आदर्श रूप से, हम कुछ ऐसा चाहते हैं

public static string GetTimeOfDay() => get_time_of_day(DateTime.Now.Hour);

// Helper function that's easy to unit test, & can live outside a class
public static get_time_of_day(hour)
{
    if (hour >= 0 && hour < 6)
        return "Night";
    if (hour >= 6 && hour < 12)
        return "Morning";
    if (hour >= 12 && hour < 18)
        return "Afternoon";
    return "Evening";
}

यह दृष्टिकोण अभी भी तंग-युग्मन आलोचना से दूर है। लेकिन हम इसे GetTimeOfDayएक तर्क देकर ठीक कर सकते हैं , जिसे मैंने नीचे दिए उदाहरण में वैकल्पिक किया है:

 public static string GetTimeOfDay(DateTime now=DateTime.Now) => get_time_of_day(now.Hour);

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

class Car:
    def __init__(self, acceleration):
        self.acceleration = acceleration
        self.speed = 0
        
    def accelerate(self, acceleration_time):
        self.speed += acceleration_time*self.acceleration

यह परीक्षण करना आसान है, उदा

car = Car(3)
car.accelerate(4)
assert car.speed == 12

(हालांकि आप की तरह है कि reformat करने के लिए स्वतंत्र महसूस)। लेकिन यह अभी भी निर्भर करता है self.acceleration, इसलिए आप उदाहरण के लिए पसंद कर सकते हैं

    def accelerate(self, acceleration_time):
        self.speed += delta_speed(self.acceleration, acceleration_time)

def delta_speed(acceleration, acceleration_time): return acceleration*acceleration_time

नोट delta_speedसमान इंडेंटेशन स्तर पर है Carक्योंकि यह एक वर्ग में नहीं रहता है, इसलिए इसमें कोई भी छिपा हुआ पैरामीटर नहीं है जो आपको परेशान करता है। (एक अभ्यास के रूप में, आप =इसके बजाय उपयोग करने के लिए इस दृष्टिकोण को फिर से लिख सकते हैं +=; यह यहाँ बनाए गए बिंदु के लिए अप्रासंगिक है।)

1 Flater Aug 19 2020 at 10:11

आपकी टिप्पणियों में कुछ (यदि सबसे अधिक नहीं) की वैधता है, लेकिन आप जो निष्कर्ष निकालते हैं, वे बहुत चरम हैं।

  1. इसे ठोस डेटा स्रोत से कसकर जोड़ा जाता है। (यह मशीन से चालू डेटाइम पढ़ता है जिस पर यह चलता है)

सही बात। दिनांक मान को या तो एक पैरामीटर के रूप में पारित किया जाना चाहिए या एक घड़ी जैसी निर्भरता को इंजेक्ट किया जाना चाहिए।

ध्यान दें कि निर्भरता इंजेक्शन के लिए एक गैर-स्थिर वर्ग और विधि की आवश्यकता होती है। उस पर और बाद में।

बाद के सुझाव (एक निर्भरता को इंजेक्ट) पर ध्यान दें। आपका प्रश्न इस विचार के विरुद्ध है, और यहीं से आपका अवलोकन रेल से दूर जाता है। उस पर और बाद में।

  1. यह एकल जिम्मेदारी सिद्धांत (एसआरपी) का उल्लंघन करता है।

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

बस स्पष्ट होने के लिए: कोड में सुधार किया जा सकता है, लेकिन एसआरपी वह नहीं है जो यहां उल्लंघन के रूप में ध्यान में आता है।

तर्क जो डेटाइम मूल्य प्राप्त कर रहा है वह एक असतत जिम्मेदारी है। किसी भी जिम्मेदारी को छोटी जिम्मेदारियों में विभाजित किया जा सकता है - लेकिन क्या उचित है और क्या ओवरकिल है, इसके बीच एक रेखा खींची जाती है। यह मानते हुए कि विधि दिन के वर्तमान समय का मूल्यांकन करती है, यह SRP उल्लंघन नहीं है।

  1. यह अपना काम करवाने के लिए आवश्यक जानकारी के बारे में है। छिपे हुए इनपुट का उपयोग करने के लिए डेवलपर्स को वास्तविक स्रोत कोड की प्रत्येक पंक्ति को पढ़ना चाहिए ...

यह यकीनन है। जब मैं देखता हूं GetTimeOfDayऔर यह स्पष्ट रूप से एक डेटाइम मूल्य (या तो विधि पैरामीटर या निर्भरता) में नहीं लेता है, तो तार्किक निष्कर्ष यह है कि वर्तमान समय का उपयोग किया जा रहा है।
शब्दार्थ, "दिन का समय प्राप्त करना" बताता है कि आपको वर्तमान समय मिल रहा है, इसलिए मैं नामकरण के साथ यहां एक मुद्दा नहीं देखता हूं।

... और वे कहाँ से आते हैं। ...

यह, मैं इस पर सहमत हूं। आपको पता नहीं है कि यह सिस्टम घड़ी पर निर्भर है, या क्लाउड-आधारित एपीआई या ... यह तब हल किया जाता है जब आप इसे एक निर्भरता के रूप में इंजेक्ट करते हैं या इसे एक विधि पैरामीटर के रूप में जोड़ते हैं।

विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।

अधिकांश ओओपी सिद्धांत (दूसरों के बीच में ठोस) कक्षाओं पर ध्यान केंद्रित करते हैं , तरीकों पर नहीं। आपको खुद से तरीकों का पालन नहीं करना चाहिए, आपको उन्हें एक वर्ग पर संचालन के रूप में देखना चाहिए, और विशेष रूप से उस वर्ग के एक ज्ञात उदाहरण पर।

जहाँ तक कोड पठनीयता का संबंध है, आप यह मान सकते हैं कि जो कोई भी वर्ग विधि को उस वर्ग के उदाहरण (ऑब्जेक्ट) पर कॉल करता है, वह भी इस बात से अवगत है कि उस वस्तु का निर्माण पहली बार में कैसे किया गया था। यह हमेशा मामला नहीं होता है, लेकिन जब यह मामला नहीं होता है तो इसका मतलब है कि कॉलर ने ऑब्जेक्ट निर्माण को सौंपने के लिए सहमति दी है।

यह आपकी ज़िम्मेदारी नहीं है (आप = भस्म वर्ग के डिजाइनर)। आप अपने उपभोक्ताओं को आंतरिक रूप से कैसे काम सौंपते हैं, इसे प्रबंधित करने की कोशिश नहीं करनी चाहिए।

जब डेटाटाइम मान का स्रोत इंजेक्ट की गई निर्भरता या एक विधि पैरामीटर होने के लिए फिर से तैयार किया गया है, तो आपके तीसरे बुलेट बिंदु में इंगित किया गया मुद्दा शून्य और शून्य है।

तो कैसे निपटें self...?

"सौदा" से तात्पर्य है कि यह एक समस्या या अवांछित वस्तु है। आपका प्रवचन selfऔर इसके साथ कथित मुद्दों पर वस्तु-उन्मुख राज्य की अवधारणा के लिए नापसंद का एक उपक्रम किया जाता है।

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

selfOOP के केंद्र में है

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

selfवह कीवर्ड है जो आपको वर्तमान ऑब्जेक्ट की स्थिति तक पहुंचने की अनुमति देता है। बिना self, आप वास्तव में वस्तु स्थिति को स्टोर और पुनः प्राप्त करने में असमर्थ हैं, और इस तरह हम एक ऐसी प्रणाली में वापस लौट आएंगे जहाँ सब कुछ स्थैतिक विधियों का एक संग्रह है।

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

... और इसका सही इस्तेमाल कैसे करें?

यह उस हिस्से पर वापस जाता है जहां मैंने कहा था कि आपको क्लास स्तर पर चीजों को देखने की जरूरत है, न कि किसी विधि स्तर पर।

इसके बारे में सोचने का सबसे आसान तरीका यह है कि एक वस्तु में संग्रहीत स्थिति (अर्थात self, आमतौर पर निर्माणकर्ता के माध्यम से) एक बार कॉन्फ़िगर की गई है और उस वर्ग के सभी तरीकों से पुन: उपयोग योग्य है। उदाहरण के लिए:

public class Clock
{
    public DateTime GetDateTime()
    {
        return DateTime.Now;
    }
}

public class SundayChecker
{
    private Clock clock;

    public SundayChecker(Clock clock)
    {
        this.clock = clock;
    }

    public bool IsItSunday()
    {
        var now = this.clock.GetDateTime();
        return now.DayOfWeek == DayOfWeek.Sunday;
    }
}

ध्यान दें कि मुझे केवल एक बारSundayChecker किस घड़ी का उपयोग करना चाहिए , यह बताने के लिए है , लेकिन मैं तब वर्तमान समय की बार-बार जांच करने में सक्षम हूं और पुष्टि करता हूं कि यह रविवार है या नहीं।

यह सिर्फ एक सरल उदाहरण है, लेकिन यह OOP की मूल प्रकृति को दर्शाता है।

नोट: ऑब्जेक्ट स्टेट का उपयोग करने के पक्ष में कई और तर्क हैं, लेकिन यह आपके दिमाग को OOP- संगत फ्रेम में शिफ्ट करने के लिए सबसे आसान है।

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

यह छोटा तरीका है और इसे पढ़ना आसान है लेकिन सौ लाइनों वाली विधियों के बारे में क्या है जो selfकई स्थानों पर पढ़ती और असाइन करती हैं?

किसी भी चीज की अति हो सकती है। सिर्फ इसलिए कि OOP के उपयोग का मतलब यह नहीं है कि इसका दुरुपयोग या बुरी तरह से लिखा नहीं जा सकता है।

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

लेकिन ये ओओपी को कंबल नियम के रूप में उपयोग करने के खिलाफ तर्क नहीं हैं। यह कहने जैसा है कि किसी को भी कभी भी हथौड़े का इस्तेमाल नहीं करना चाहिए क्योंकि आपने अपने पिताजी को हथौड़े से अपना अंगूठा मारते देखा है।
गलतियाँ होती हैं, लेकिन गलतियों का अस्तित्व एक पूरे के रूप में अवधारणा को बाधित नहीं करता है।

ErikEidt Aug 19 2020 at 11:38

दिन के समय को "अब" कहना एक तरीका है, जो आपके द्वारा दिखाए गए दिन स्ट्रिंग के समय की तरह गणना करता है। यह है क्योंकि,

  • यदि आप अब से किसी अन्य समय से दिन स्ट्रिंग का समय जानना चाहते हैं, तो आप बस इस पद्धति का उपयोग नहीं कर सकते हैं - जो इस पद्धति को बहुत कम उपयोगी बनाता है और आपको उस तर्क को दूसरे तरीके से उपयोग करने के लिए अपने तर्क को दोहराना होगा।

  • यदि आप दिन का समय जानना चाहते हैं, लेकिन दिन का वास्तविक समय भी चाहते हैं, तो आप दिन का समय दो बार समाप्त करते हैं, और दो अलग-अलग कॉल "अब" आसानी से अलग-अलग मूल्य हो सकते हैं, जहां कोड के लेखक हैं सबसे अधिक संभावना है कि वे वास्तव में मेल खाते हैं।

आदर्श रूप से यदि आपको "अब" दिन के समय की आवश्यकता है, तो यह केवल एक बार (प्रति के अनुसार) प्राप्त किया जाता है और किसी भी कोड के पैरामीटर के रूप में पारित किया जाता है जो "वर्तमान" समय के साथ काम कर रहा है।