वस्तु गुणों को कैसे संशोधित करें?
मैंने हाल ही में यूनिट परीक्षण के बारे में एक महान लेख पढ़ा है । एक बुरी विधि का एक उदाहरण था जो अच्छी तरह से डिज़ाइन नहीं किया गया है। यह इस तरह दिख रहा है
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";
}
कुछ चीजें हैं जो लेखक ने बताई हैं कि वे प्रतिमान हैं:
- इसे ठोस डेटा स्रोत से कसकर जोड़ा जाता है। (यह मशीन से चालू डेटाइम पढ़ता है जिस पर यह चलता है)
- यह एकल जिम्मेदारी सिद्धांत (एसआरपी) का उल्लंघन करता है।
- यह अपना काम करवाने के लिए आवश्यक जानकारी के बारे में है। डेवलपर्स को वास्तविक स्रोत कोड की प्रत्येक पंक्ति को यह समझने के लिए पढ़ना चाहिए कि छिपे हुए इनपुट का क्या उपयोग किया जाता है और वे कहां से आते हैं। विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।
मैं मुख्य रूप से पायथन में कोड करता हूं और इस लेख के बाद मुझे लगता है कि 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
छिपा इनपुट है:self.power
- विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है। वहाँ छिपा हुआ उत्पादन (?) है
self.speed
यह छोटा तरीका है और इसे पढ़ना आसान है लेकिन सौ लाइनों वाली विधियों के बारे में क्या है जो self
कई स्थानों पर पढ़ती और असाइन करती हैं? यदि उन लोगों का नाम ठीक से नहीं लिया गया है, तो यह समझने के लिए कि यह क्या करता है और यहां तक कि अगर उन लोगों का नाम ठीक से डेवलपर के लिए है, तो यह जानने के लिए पूरे कार्यान्वयन को पढ़ना चाहिए कि क्या यह कुछ self
सामान को संशोधित करता है , या यदि अतिरिक्त संदर्भ के साथ इंजेक्ट किया जाता है, तो बड़ी परेशानी होगी self
।
दूसरी ओर जब मैं self
इनपुट (आर्ग्युमेंट्स) और आउटपुट (रिटर्न वैल्यू) के साथ बिना उपयोग किए हर विधि को कोड करने की कोशिश करूंगा , तब मैं कई तरीकों के माध्यम से एक चर को समाप्त करूंगा और खुद को दोहराऊंगा।
तो इससे कैसे निपटें self
और इसका सही इस्तेमाल कैसे करें? यह कैसे स्पष्ट करें कि इनपुट के रूप में कौन सी विधि का उपयोग किया जाता है और यह क्या संशोधित करता है (आउटपुट)?
जवाब
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()
उस वस्तु के खेतों का उपभोग किया जाता है जिस पर उसे बुलाया गया था। यह पूरे बिंदु की तरह है ... जब तक इस ऑब्जेक्ट में डेटा यथोचित रूप से छोटा और प्रबंधनीय है, तब तक ऐसे ऑपरेशन ठीक हैं। यदि आप कल्पना करना चाहते हैं, तो आप इसे "इंटरफ़ेस अलगाव सिद्धांत" का एक उदाहरण कह सकते हैं। समस्याएँ वास्तव में उन अधकचरी देव वस्तुओं के लिए पैदा होती हैं जिनमें दर्जनों क्षेत्र होते हैं।
सवाल यह नहीं होना चाहिए कि क्या विधि के बाहरी कॉलर को पता है कि ऑब्जेक्ट के किन क्षेत्रों का उपयोग किया जाएगा। कॉल करने वाले को यह पता नहीं होना चाहिए, ऑब्जेक्ट एक इनकैप्सुलेटेड चीज़ होना चाहिए। एक अधिक महत्वपूर्ण सवाल यह है कि क्या ये निर्भरताएं और संबंध वस्तु के भीतर से स्पष्ट हैं। कई क्षेत्रों के होने से चीजों को सिंक से बाहर निकलने के कई अवसर मिलते हैं।
सबसे पहले, यह ध्यान देने योग्य है कि लेख में उदाहरण कुछ हद तक (व्यावहारिक कारणों से) वंचित है, और यह संदर्भ तब मायने रखता है जब यह इन चीजों की बात आती है। उदाहरण के लिए, यदि आप एक छोटा, एक-बंद उपकरण लिख रहे हैं, तो डिज़ाइन के साथ बहुत अधिक परेशान करने का कोई कारण नहीं है। लेकिन मान लें कि यह कुछ लंबी अवधि की परियोजना का एक हिस्सा है, और आप यथोचित अपेक्षा कर सकते हैं कि यह कोड कुछ डिज़ाइन परिवर्तनों से लाभान्वित होगा (या आपको पहले से ही मौजूदा डिज़ाइन के साथ परिवर्तन को लागू करना होगा), और चलिए जाँच करते हैं इस संदर्भ में।
यहाँ संदर्भ के लिए कोड है:
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 कभी-कभी कंस्ट्रक्टर इंजेक्शन संभव नहीं है (उदाहरण के लिए एक फ्रेमवर्क को एक पैरामीटर रहित कंस्ट्रक्टर की आवश्यकता हो सकती है), इसलिए निर्भरताएं विधियों या गुणों के माध्यम से इंजेक्ट की जाती हैं, लेकिन यह कम आदर्श है।
इस प्रकार के प्रश्नों का उत्तर आमतौर पर विधि का उपयोग करके कोड को देखकर दिया जा सकता है।
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
विधि की आवश्यकता होती है जो कार की वर्तमान शक्ति के आधार पर उचित त्वरण समय की गणना करती है।
विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।
मुझे लगता है। एक कार में तेजी आ सकती है। त्वरण के बाद गति पहले की तुलना में अधिक है। मुझे बस इतना ही पता होना चाहिए।
मूल दृष्टिकोण यह है कि कक्षा के बाहर रहने वाले कार्यों में यथासंभव अधिक से अधिक तर्क रखें (या स्थिर हैं), फिर उन्हें उन तरीकों से कॉल करें जो एक राज्य पर निर्भर करते हैं। (इन कॉलों को अभी भी तकनीकी रूप से पारित संपत्ति को अपने हस्ताक्षर से छिपाने की आवश्यकता है, लेकिन यह ओओपी की तरह है, जो कुछ और तरीकों की जरूरत से अलग राज्य है, वे सिर्फ एक-इन-वैक्यूम कार्यों में नहीं हैं। ) मैं जो दूसरा मुख्य मुद्दा बनाना चाहता हूं, वह यह है कि अन्य मुद्दे हैं जिन्हें हमें पहले संबोधित करना चाहिए।
अपने पहले उदाहरण के साथ, यह पहले एक और चिंता को संबोधित करने के लिए इसे संपादित करने में मदद करता है, कि यह इकाई-परीक्षण के लिए कठिन है। आदर्श रूप से, हम कुछ ऐसा चाहते हैं
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
क्योंकि यह एक वर्ग में नहीं रहता है, इसलिए इसमें कोई भी छिपा हुआ पैरामीटर नहीं है जो आपको परेशान करता है। (एक अभ्यास के रूप में, आप =
इसके बजाय उपयोग करने के लिए इस दृष्टिकोण को फिर से लिख सकते हैं +=
; यह यहाँ बनाए गए बिंदु के लिए अप्रासंगिक है।)
आपकी टिप्पणियों में कुछ (यदि सबसे अधिक नहीं) की वैधता है, लेकिन आप जो निष्कर्ष निकालते हैं, वे बहुत चरम हैं।
- इसे ठोस डेटा स्रोत से कसकर जोड़ा जाता है। (यह मशीन से चालू डेटाइम पढ़ता है जिस पर यह चलता है)
सही बात। दिनांक मान को या तो एक पैरामीटर के रूप में पारित किया जाना चाहिए या एक घड़ी जैसी निर्भरता को इंजेक्ट किया जाना चाहिए।
ध्यान दें कि निर्भरता इंजेक्शन के लिए एक गैर-स्थिर वर्ग और विधि की आवश्यकता होती है। उस पर और बाद में।
बाद के सुझाव (एक निर्भरता को इंजेक्ट) पर ध्यान दें। आपका प्रश्न इस विचार के विरुद्ध है, और यहीं से आपका अवलोकन रेल से दूर जाता है। उस पर और बाद में।
- यह एकल जिम्मेदारी सिद्धांत (एसआरपी) का उल्लंघन करता है।
मैं यह नहीं देखता कि यह कैसे होता है, और आपने यह नहीं बताया कि आप ऐसा क्यों सोचते हैं। यह तरीका एक काम करता है। एसआरपी इस बात पर ध्यान केंद्रित नहीं करता है कि क्या निर्भरताएं इंजेक्शन हैं, एसआरपी वर्ग के भीतर निहित तर्क पर केंद्रित है। इस वर्ग का एक सख्ती से परिभाषित उद्देश्य है: दिन के वर्तमान समय के लिए एक मानव-अनुकूल लेबल उत्पन्न करना।
बस स्पष्ट होने के लिए: कोड में सुधार किया जा सकता है, लेकिन एसआरपी वह नहीं है जो यहां उल्लंघन के रूप में ध्यान में आता है।
तर्क जो डेटाइम मूल्य प्राप्त कर रहा है वह एक असतत जिम्मेदारी है। किसी भी जिम्मेदारी को छोटी जिम्मेदारियों में विभाजित किया जा सकता है - लेकिन क्या उचित है और क्या ओवरकिल है, इसके बीच एक रेखा खींची जाती है। यह मानते हुए कि विधि दिन के वर्तमान समय का मूल्यांकन करती है, यह SRP उल्लंघन नहीं है।
- यह अपना काम करवाने के लिए आवश्यक जानकारी के बारे में है। छिपे हुए इनपुट का उपयोग करने के लिए डेवलपर्स को वास्तविक स्रोत कोड की प्रत्येक पंक्ति को पढ़ना चाहिए ...
यह यकीनन है। जब मैं देखता हूं GetTimeOfDay
और यह स्पष्ट रूप से एक डेटाइम मूल्य (या तो विधि पैरामीटर या निर्भरता) में नहीं लेता है, तो तार्किक निष्कर्ष यह है कि वर्तमान समय का उपयोग किया जा रहा है।
शब्दार्थ, "दिन का समय प्राप्त करना" बताता है कि आपको वर्तमान समय मिल रहा है, इसलिए मैं नामकरण के साथ यहां एक मुद्दा नहीं देखता हूं।
... और वे कहाँ से आते हैं। ...
यह, मैं इस पर सहमत हूं। आपको पता नहीं है कि यह सिस्टम घड़ी पर निर्भर है, या क्लाउड-आधारित एपीआई या ... यह तब हल किया जाता है जब आप इसे एक निर्भरता के रूप में इंजेक्ट करते हैं या इसे एक विधि पैरामीटर के रूप में जोड़ते हैं।
विधि के व्यवहार को समझने के लिए केवल विधि हस्ताक्षर पर्याप्त नहीं है।
अधिकांश ओओपी सिद्धांत (दूसरों के बीच में ठोस) कक्षाओं पर ध्यान केंद्रित करते हैं , तरीकों पर नहीं। आपको खुद से तरीकों का पालन नहीं करना चाहिए, आपको उन्हें एक वर्ग पर संचालन के रूप में देखना चाहिए, और विशेष रूप से उस वर्ग के एक ज्ञात उदाहरण पर।
जहाँ तक कोड पठनीयता का संबंध है, आप यह मान सकते हैं कि जो कोई भी वर्ग विधि को उस वर्ग के उदाहरण (ऑब्जेक्ट) पर कॉल करता है, वह भी इस बात से अवगत है कि उस वस्तु का निर्माण पहली बार में कैसे किया गया था। यह हमेशा मामला नहीं होता है, लेकिन जब यह मामला नहीं होता है तो इसका मतलब है कि कॉलर ने ऑब्जेक्ट निर्माण को सौंपने के लिए सहमति दी है।
यह आपकी ज़िम्मेदारी नहीं है (आप = भस्म वर्ग के डिजाइनर)। आप अपने उपभोक्ताओं को आंतरिक रूप से कैसे काम सौंपते हैं, इसे प्रबंधित करने की कोशिश नहीं करनी चाहिए।
जब डेटाटाइम मान का स्रोत इंजेक्ट की गई निर्भरता या एक विधि पैरामीटर होने के लिए फिर से तैयार किया गया है, तो आपके तीसरे बुलेट बिंदु में इंगित किया गया मुद्दा शून्य और शून्य है।
तो कैसे निपटें
self
...?
"सौदा" से तात्पर्य है कि यह एक समस्या या अवांछित वस्तु है। आपका प्रवचन self
और इसके साथ कथित मुद्दों पर वस्तु-उन्मुख राज्य की अवधारणा के लिए नापसंद का एक उपक्रम किया जाता है।
यदि आप ऐसा महसूस करते हैं, और आप अपने सोचने के तरीके को बदलना नहीं चाहते हैं, तो यह ठीक है। प्रोग्रामिंग मन की एक अमूर्त अवधारणा है, और एक ही समस्या को हल करने के लिए विभिन्न दृष्टिकोण मौजूद हैं। उस स्थिति में, आपको एक प्रमुख कारण के लिए, ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग के बजाय कार्यात्मक प्रोग्रामिंग में जाने पर विचार करना चाहिए:
self
OOP के केंद्र में है ।
वस्तुएं राज्य को ट्रैक करती हैं। वो यही करते हैं। यदि वे नहीं करते, तो आपका कोडबेस केवल विधियों में ही मौजूद होता है, और फिर उन सभी तरीकों को स्थिर बनाया जा सकता है।
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 या नहीं, सैकड़ों लाइनों वाली विधियां स्वाभाविक रूप से एक कोड गुणवत्ता मुद्दा है।
- जबकि वस्तु स्थिति में हेरफेर किया जा सकता है और यह स्वाभाविक रूप से एक बुरा विचार नहीं है, एक ऐसी वस्तु का होना जिसका राज्य लगातार इस बिंदु पर बदल रहा है कि अब उसका राज्य क्या है, इस पर नज़र रखने में सक्षम नहीं है, एक कोड गुणवत्ता मुद्दा भी है।
लेकिन ये ओओपी को कंबल नियम के रूप में उपयोग करने के खिलाफ तर्क नहीं हैं। यह कहने जैसा है कि किसी को भी कभी भी हथौड़े का इस्तेमाल नहीं करना चाहिए क्योंकि आपने अपने पिताजी को हथौड़े से अपना अंगूठा मारते देखा है।
गलतियाँ होती हैं, लेकिन गलतियों का अस्तित्व एक पूरे के रूप में अवधारणा को बाधित नहीं करता है।
दिन के समय को "अब" कहना एक तरीका है, जो आपके द्वारा दिखाए गए दिन स्ट्रिंग के समय की तरह गणना करता है। यह है क्योंकि,
यदि आप अब से किसी अन्य समय से दिन स्ट्रिंग का समय जानना चाहते हैं, तो आप बस इस पद्धति का उपयोग नहीं कर सकते हैं - जो इस पद्धति को बहुत कम उपयोगी बनाता है और आपको उस तर्क को दूसरे तरीके से उपयोग करने के लिए अपने तर्क को दोहराना होगा।
यदि आप दिन का समय जानना चाहते हैं, लेकिन दिन का वास्तविक समय भी चाहते हैं, तो आप दिन का समय दो बार समाप्त करते हैं, और दो अलग-अलग कॉल "अब" आसानी से अलग-अलग मूल्य हो सकते हैं, जहां कोड के लेखक हैं सबसे अधिक संभावना है कि वे वास्तव में मेल खाते हैं।
आदर्श रूप से यदि आपको "अब" दिन के समय की आवश्यकता है, तो यह केवल एक बार (प्रति के अनुसार) प्राप्त किया जाता है और किसी भी कोड के पैरामीटर के रूप में पारित किया जाता है जो "वर्तमान" समय के साथ काम कर रहा है।