Comment modifier les propriétés des objets?
J'ai récemment lu un excellent article sur les tests unitaires . Il y a eu un exemple de mauvaise méthode qui n'est pas bien conçue. Ça ressemble à ça
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";
}
L'auteur a souligné certaines choses car ce sont des anti-modèles:
- Il est étroitement lié à la source de données concrète. (il lit le datetime actuel de la machine sur laquelle il s'exécute)
- Il viole le principe de responsabilité unique (SRP).
- Il ment sur les informations nécessaires pour faire son travail. Les développeurs doivent lire chaque ligne du code source réel pour comprendre quelles entrées cachées sont utilisées et d'où elles proviennent. La signature de la méthode seule ne suffit pas pour comprendre le comportement de la méthode.
Je code principalement en Python et après cet article, j'ai l'impression que l'utilisation self
dans la plupart des cas viole également ces points.
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
a une entrée masquée:self.power
- La signature de la méthode seule ne suffit pas pour comprendre le comportement de la méthode. Il y a une sortie cachée (?)
self.speed
C'est une petite méthode et c'est facile à lire, mais qu'en est-il des méthodes avec des centaines de lignes qui lisent et attribuent à self
de nombreux endroits? Si ceux-ci ne sont pas nommés correctement, le développeur aura de gros problèmes pour comprendre ce qu'il fait et même si ceux-ci sont correctement nommés, le développeur doit lire l'intégralité de l'implémentation pour savoir si elle modifie certaines self
choses ou si un contexte supplémentaire est injecté self
.
D'un autre côté, quand je vais essayer de coder chaque méthode sans utiliser self
, avec entrée (arguments) et sortie (valeur de retour), je finirai par passer une variable à travers de nombreuses méthodes et je me répéterai.
Alors, comment faire face self
et comment l'utiliser correctement? Comment préciser quelle méthode utilise comme entrée et ce qu'elle modifie (sortie)?
Réponses
Eeh, il vaut mieux ne pas devenir trop extrême. Oui, il est vrai que les petites fonctions pures sans flux de données explicites sont beaucoup plus faciles à tester que les opérations de mutation qui entraînent une action à distance. Mais dans la raison, la mutabilité, l'impureté et les dépendances ne sont pas un problème. Ils rendent certaines choses beaucoup plus pratiques.
En règle générale: plus un code est proche de la logique métier de certains logiciels, plus il devrait devenir pur, immuable, fonctionnel, explicite et testable. Plus le code est proche des couches externes de l'application, moins il y a de fonctionnalités qui valent la peine d'être testées avec soin, et donc les conceptions moins testables sont acceptables. Par exemple, le code qui encapsule juste une API externe ne peut pas être raisonnablement testé unitaire.
À titre d'exemple pour les problèmes d'impureté, de nombreuses introductions de programmation vous demandent de créer des objets de domaine qui produisent directement une sortie:
class Cat(Animal):
def make_noise(self):
print("meow")
Ce n'est pas une bonne conception, car la sortie est étroitement couplée au sys.stdout
flux. Plus testables conceptions comprendraient retournant une chaîne au lieu d'imprimer directement comme
def noise(self): return "meow"
ou le passage dans un fichier qui peut être imprimé à:
def make_noise(self, stream): print("meow", file=stream)
.
Dans votre exemple, vous avez une opération de mutation car.accelerate(t)
. Ce n'est pas un problème! Cette opération ne menace pas la testabilité car le résultat peut être facilement affirmé:
car = Car(10)
assert car.speed == 0
car.accelerate(5)
assert car.speed == 50
Le nom accelerate()
indique également suffisamment clairement qu'il s'agit d'une opération en mutation. D'autres langages l'encodent également dans le système de types (par exemple fn accelerate(&mut self)
dans Rust) ou dans la convention de dénomination (par exemple accelerate!
dans Ruby). Garder une distinction entre les commandes mutantes et les requêtes pures a tendance à être utile, même si cela ne fonctionne pas toujours dans la pratique.
S'il y a un problème dans votre code, ce n'est pas que la méthode accelerate () assigne self
, mais la self.calculate_acceleration(time, self.power)
méthode. Cette méthode reçoit les données de self
deux fois: une fois comme objet de la méthode sur laquelle elle est invoquée, une autre fois via le deuxième paramètre. Cela rend les flux de données non transparents - il n'y a aucune raison pour que ce soit une méthode à moins qu'elle self
ne soit mutée dans la méthode. Changer la conception de cette manière peut être utile:
def calculate_acceleration(time, power):
...
class Car:
def __init__(self, power):
...
def accelerate(self, acceleration_time):
self.speed = calculate_acceleration(acceleration_time, self.power)
Dans ce cas particulier, il n'y a pas d'impact réel sur la testabilité, mais dans d'autres cas, il peut maintenant être possible de tester le calcul directement, sans avoir à passer par l'interface de l'objet. Alors que dans d'autres langages, les méthodes d'assistance statiques privées sont normales, ce n'est pas une approche appropriée pour Python - utilisez simplement une fonction libre.
Une critique possible des méthodes est qu'il n'est pas clair quels champs sont consommés. Par exemple, ce type de flux de données serait dingue même s'il est sans doute conforme au «Clean Code»:
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())
Mais le WTF dans ce code est celui qui z
est utilisé pour communiquer les résultats internes, ou c'est use_x
un champ auquel il devrait probablement être un argument de mot-clé facultatif some_operation()
.
Ce qui n'est pas un problème, c'est qu'il some_operation()
consomme des champs de l'objet sur lequel il a été appelé. C'est comme… le point entier. Tant que les données de cet objet sont raisonnablement petites et gérables, de telles opérations conviennent. Si vous voulez avoir de la fantaisie, vous pouvez appeler cela une instance du «principe de ségrégation d'interface». Les problèmes se posent principalement pour les objets divins vraiment difficiles à manier qui ont des dizaines de champs.
La question ne devrait pas être de savoir si l'appelant externe de la méthode sait quels champs de l'objet seront utilisés. L'appelant ne devrait pas avoir à le savoir, l'objet devrait être une chose encapsulée. Une question plus importante est de savoir si ces dépendances et relations sont claires de l'intérieur de l'objet. Avoir de nombreux champs implique de nombreuses opportunités de désynchronisation des choses.
Tout d'abord, il convient de noter que l'exemple de l'article est quelque peu artificiel (pour des raisons pratiques), et que le contexte compte quand il s'agit de ces choses. Par exemple, si vous écrivez un petit outil unique, il y a peu de raisons de trop vous soucier de la conception. Mais disons que cela fait partie d'un projet à plus long terme, et que vous pouvez raisonnablement vous attendre à ce que ce code bénéficie de certaines modifications de conception (ou que vous avez déjà dû implémenter des modifications qui entrent en conflit avec la conception actuelle), et examinons dans ce contexte.
Voici le code pour référence:
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";
}
En C #, le static
mot - clé signifie essentiellement qu'il s'agit d'une fonction libre (c'est-à-dire pas d'une méthode d'instance sur un objet). Ceci est pertinent dans le contexte de votre question, puisque vous demandez comment ces considérations s'appliquent aux objets .
L'auteur de l'article soulève plusieurs points; permettez-moi d'abord d'aborder 1. (étroitement lié au service de fourniture de dates - la DateTime
classe) et 3. (induit en erreur sur les dépendances). Le problème que cela crée est que, bien que la fonction fonctionne bien dans les circonstances pour lesquelles elle a été créée à l'origine, elle n'est pas utilisable dans d' autres contextes .
Par exemple, que se passe-t-il si j'ai besoin de prendre en charge une interface utilisateur qui permet aux utilisateurs de voir la catégorie "heure du jour" pour une date future (encore une fois, cet exemple "Matin / Après-midi / Soir / Nuit" est artificiel, mais supposons qu'il renvoie des affaires- catégorie pertinente à la place, quelque chose qui intéresse les utilisateurs).
Un autre contexte de ce type est, bien sûr, le test, où vous voulez pouvoir brancher des valeurs prédéfinies (actuellement impossible) et vérifier les résultats (du point de vue d'un test, la fonction est non déterministe - vous ne pouvez pas dire À quoi s'attendre).
Ceci est facilement résolu en faisant de la date-heure un paramètre:
public static string GetTimeOfDay(DateTime dateTime)
{
// same code, except that it uses the dateTime param...
}
Maintenant, concernant la violation de SRP (point 2) - le problème est qu'il n'est pas très significatif d'en parler en termes abstraits. Ce que je veux dire par là, c'est qu'il n'est pas très significatif de simplement regarder le code de manière isolée et d'envisager un tas de scénarios «et si». Bien sûr, il y a des choses générales que vous pouvez dire à propos de SRP de cette manière, mais si vous ne considérez pas comment votre code change réellement, et les besoins réels de conception, vous vous retrouverez avec un butin d'efforts gaspillés et avec trop code compliqué (lire «sur-ingénierie»).
Cela signifie que même si vous pouvez et devriez appliquer le SRP initialement sur la base de quelques suppositions éclairées et d'hypothèses raisonnables, vous devrez reconsidérer votre conception sur plusieurs itérations / sprints à mesure que votre compréhension des responsabilités réelles et des modèles de changement augmente, au fur et à mesure que vous travaillez sur ce code.
Or, l'auteur dit que la fonction "consomme l'information et la traite également". C'est trop vague pour être utile, on pourrait dire cela de n'importe quelle fonction. Et même si une fonction délègue le traitement au code de niveau inférieur, à la fin de la chaîne, il doit y avoir quelque chose qui "consomme l'information et la traite également".
Le fait est que si cette partie de la base de code change très rarement (ou jamais), vous n'avez pas vraiment besoin de considérer SRP. Vous pouvez trouver un certain nombre de raisons différentes pour changer, mais si ces changements ne se produisent jamais, vous avez payé les coûts de conception sans obtenir aucun avantage. Par exemple, peut-être que les chaînes retournées devraient être disponibles dans différentes langues (peut-être que la fonction devrait renvoyer une clé à un dictionnaire pour prendre en charge la localisation). Ou peut-être que les valeurs de seuil pour différents moments de la journée peuvent varier - peut-être devraient-elles être lues à partir d'une base de données. Ou peut-être que ces valeurs changent tout au long de l'année. Ou peut-être que toute cette logique n'est pas universelle, alors peut-être qu'une sorte de stratégie devrait être injectée dans la fonction (le modèle de stratégie). Qu'en est-il d'une conception qui doit prendre en charge tout ce qui précède?
Voyez ce que je veux dire par un tas de scénarios «et si»? Ce que vous devriez faire à la place, c'est développer une compréhension du domaine du problème et de la base de code, et appliquer SRP afin que les axes de changement les plus importants (types de changements, responsabilités) soient bien pris en charge.
Le concept d'une couture
Ainsi, lorsque vous concevez des fonctions ou des classes (ou des bibliothèques et des frameworks, d'ailleurs), vous fournissez souvent des points d'extensibilité - des endroits où le code client peut brancher quelque chose, ou paramétrer autrement le comportement fourni. Michael Feathers (dans Working Effectively with Legacy Code ) appelle ces «coutures» - une couture est un endroit où vous pouvez joindre deux composants logiciels ensemble. Faire du datetime un paramètre est une couture très simple. L'injection de dépendance est également un moyen de créer des coutures. Par exemple, vous pouvez également injecter une fonction ou un objet qui peut renvoyer une instance de datetime (cela peut ou non être une surpuissance dans le contexte de cet exemple particulier).
Et les objets?
Jusqu'à présent, nous avons envisagé les choses au niveau d'une fonction libre; les objets fournissent un autre niveau d'organisation. Il faut donc maintenant considérer l'objet dans son ensemble, car les objets ont leurs propres mécanismes pour introduire des coutures.
La manière typique de le faire est par injection de constructeur (car cela se traduit par un objet prêt à l'emploi) 1 . Une classe (Python) équivalente à l'exemple de code ci-dessus serait:
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"
Cela pose les mêmes problèmes, mais le problème maintenant n'est pas la méthode elle-même, c'est le fait que le constructeur de classe crée la dépendance datetime en interne, et il n'offre pas un moyen explicite de brancher autre chose. Il n'y a pas de couture intégrée à cet effet. Il n'est pas facile de réutiliser la classe dans un scénario différent.
Voici la même classe, mais maintenant le constructeur prend un "fournisseur datetime":
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()
Vous pouvez maintenant brancher différentes choses, tant que la chose qui joue le rôle de datetimeProvider
satisfait l'interface requise (qui, dans ce cas, se compose uniquement de la méthode now () qui renvoie une instance datetime). Par exemple:
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"
Cela répond aux préoccupations 1. et 3. d'avant (avec les mêmes considérations concernant la préoccupation 2. (SRP)). Donc, vous voyez, l'utilisation de self
n'est pas le problème en soi, elle a plus à voir avec la conception de la classe. Comme d'autres réponses l'ont mentionné, lorsque vous utilisez une classe (ou plus précisément un objet), vous savez ce que cet objet représente conceptuellement, et il n'est pas surprenant pour vous, le programmeur, que la classe ait et utilise son état interne.
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)
D'après ma compréhension de la classe Car, de la dénomination de la méthode, et peut-être de la documentation, il n'est pas surprenant pour moi que cela accelerate
change l'état de l'instance. Ce n'est pas quelque chose d'inattendu pour les objets.
Ce qui pose problème, c'est si la classe a des dépendances cachées qui sont en quelque sorte pertinentes pour votre travail, ce qui rend les choses plus difficiles pour vous.
Cela dit, ce qui peut prêter à confusion (à la lumière de ce qui précède), c'est que les méthodes d'instance doivent souvent prendre leurs propres paramètres. Considérez-les comme une acceptation d'informations contextuelles supplémentaires qui ne sont pas directement liées à la responsabilité principale de la classe. Par exemple, ce n'est pas quelque chose que vous pouvez transmettre une fois au constructeur, mais quelque chose qui peut changer à chaque appel. Un exemple de jouet classique est celui des formes (cercles, triangles, rectangles) qui peuvent se dessiner (ou, au lieu de formes, il peut s'agir d'éléments d'interface utilisateur (boutons, étiquettes, etc.) ou d'entités de jeu (par exemple, des sprites 2D)). Une façon de le faire est d'avoir une méthode draw () sans paramètre, qui effectue tout le dessin en interne. Mais que se passe-t-il si vous souhaitez dessiner la même chose dans une partie complètement différente d'une interface utilisateur, sur une surface de dessin distincte? Ou sur un autre tampon pour pouvoir faire des effets spéciaux comme des portails ou des miroirs? L'alternative la plus flexible consiste à passer la surface de dessin (ou une sorte d'objet graphique) en tant que paramètre de la méthode de dessin.
mais qu'en est-il des méthodes avec des centaines de lignes qui se lit et s'assigne à soi en de nombreux endroits?
Prenez ce code et brûlez-le avec le feu.
Si ceux-ci ne sont pas nommés correctement, le développeur aura de gros problèmes pour comprendre ce qu'il fait et même si ceux-ci sont nommés correctement, le développeur doit lire toute l'implémentation pour savoir si elle modifie certaines choses personnelles, ou si un contexte supplémentaire est injecté avec soi-même.
Ouais. Exactement. N'écrivez pas de méthodes avec des centaines de lignes de code.
Maintenant, sur une note plus sérieuse, parfois, vous vous retrouverez avec de grandes méthodes. Mais la plupart du temps, essayez de décomposer votre code en méthodes plus petites et en petites classes.
Si vous avez une grande méthode comme celle que vous décrivez, une méthode dont vous ne pouvez pas faire la différence, cette méthode souffre de toutes sortes de problèmes de conception que vous n'allez pas résoudre en modifiant sa signature. Il ne s'agit pas self
ou du paramètre requis - cette méthode pose de plus gros problèmes . Vous devez le refactoriser, trouver des éléments généralisables et le décomposer en morceaux plus petits, plus compréhensibles et plus fiables (méthodes que vous n'avez pas à examiner pour comprendre la méthode qui les appelle). Vous pouvez même finir par mettre ces morceaux dans des classes complètement différentes.
D'un autre côté, quand je vais essayer de coder chaque méthode sans utiliser self, avec entrée (arguments) et sortie (valeur de retour), je finirai par passer une variable à travers de nombreuses méthodes et je me répéterai.
Eh bien, n'allez pas à l'un ou l'autre extrême. Écrivez des classes relativement petites, essayez de trouver des abstractions utiles et réfléchissez à ce que vous transmettez en tant que paramètre / dépendance de l'objet lui-même et à ce que vous souhaitez fournir en tant qu'informations contextuelles aux méthodes individuelles. Déterminez si les instances de votre classe doivent apparaître dans des scénarios autres que celui que vous aviez initialement prévu et voyez si votre conception peut les accueillir.
Comment préciser quelle méthode utilise comme entrée et ce qu'elle modifie (sortie)?
Encore une fois, en ce qui concerne les objets, ce que vous voulez faire est de clarifier ce que l’objet lui-même représente. Pour les dépendances au niveau objet, utilisez (de préférence) l'injection de constructeur et indiquez clairement ce que la classe représente conceptuellement, ce qu'elle fait et comment elle est censée être utilisée. Par exemple, les méthodes, utilisez une bonne dénomination, décrivez ce qu'elles font et utilisez des paramètres contextuels si nécessaire. En ce qui concerne les méthodes de classe et les méthodes statiques, menacez-les davantage comme des fonctions libres qui sont en quelque sorte étroitement liées au concept représenté par la classe contenant (ce sont souvent des choses comme des méthodes d'assistance et des usines).
1 Parfois, l'injection de constructeur n'est pas faisable (par exemple, un framework peut nécessiter un constructeur sans paramètre), donc les dépendances sont injectées via des méthodes ou des propriétés à la place, mais c'est moins idéal.
On peut généralement répondre à ces types de questions en examinant le code à l'aide de la méthode.
acceleration_time = 5000 # in milliseconds
car.accelerate(acceleration_time)
print(car.speed) # <-- what do you as a programmer expect the speed to be?
Alors que nous voulons écrire du code testable, nous faisons le code utilisation en dehors des tests unitaires. Les tests unitaires vérifient le comportement du public. Le comportement interne d'une classe n'est pas quelque chose qu'un test unitaire doit vérifier explicitement .
Quand je vois le mot «accélérer», je m'attends à ce que quelque chose soit plus rapide une fois l'accélération terminée. Cela implique une modification de la valeur d'exécution de self.speed
.
Comparez cela avec une physique de modélisation de classe, comme VehicleAccelerationPhysics
. Je m'attendrais à ce qu'une calculate_acceleration
méthode renvoie une valeur et non la modification d'une valeur. Mais une accelerate
méthode sur un Car
ne me surprendrait pas si elle était car.speed
modifiée - je m'attendrais à ce qu'elle soit modifiée.
Par conséquent, votre code ne viole aucune des meilleures pratiques en ce qui concerne les tests unitaires.
accelerate
a une entrée masquée:self.power
La valeur actuelle de self.power
est un détail d'implémentation et non une "entrée masquée". Si au contraire vous souhaitez accélérer à une vitesse spécifique, votre Car
classe a besoin d'une accelerate_to_speed
méthode qui calcule le temps d'accélération approprié en fonction de la puissance actuelle de la voiture.
La signature de la méthode seule ne suffit pas pour comprendre le comportement de la méthode.
Semble me trouver. Une voiture peut accélérer. Après l'accélération, la vitesse est supérieure à ce qu'elle était auparavant. C'est tout ce que j'ai besoin de savoir.
L'approche de base consiste à mettre autant de logique que possible dans des fonctions qui vivent en dehors de la classe (ou qui sont statiques), puis à les appeler de manière concise dans les méthodes qui dépendent d'un état. (Ces appels doivent encore techniquement masquer la propriété transmise de leur signature, mais c'est un peu le but de la POO, pour avoir un état persistant séparé de tout ce dont les méthodes ont besoin; ce ne sont pas seulement des fonctions in-a-empty. ) L'autre point principal que je veux souligner est qu'il y a d'autres problèmes que nous devrions aborder en premier.
Avec votre premier exemple, il est utile de le modifier d'abord pour répondre à un autre problème, qu'il est difficile de tester unitaire. Idéalement, nous voulons quelque chose comme
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";
}
Cette approche va toujours à l'encontre de la critique de couplage étroit. Mais nous pouvons résoudre ce problème en donnant GetTimeOfDay
un argument, que j'ai rendu facultatif dans l'exemple ci-dessous:
public static string GetTimeOfDay(DateTime now=DateTime.Now) => get_time_of_day(now.Hour);
Dans votre deuxième exemple, je vais changer votre power
terminologie. La accelerate
méthode est étrange en ce qu'elle passe une propriété de l'instance de classe à une méthode qui, parce qu'elle vit de manière non statique dans la classe, peut appeler cette propriété de toute façon, comme s'il s'agissait d'un hybride entre cacher deux de ces appels et ne cacher aucun d'eux . Il peut être modifié comme suit:
class Car:
def __init__(self, acceleration):
self.acceleration = acceleration
self.speed = 0
def accelerate(self, acceleration_time):
self.speed += acceleration_time*self.acceleration
C'est facile à tester, par exemple
car = Car(3)
car.accelerate(4)
assert car.speed == 12
(n'hésitez pas à reformater comme vous le souhaitez). Mais cela dépend toujours de self.acceleration
, donc vous préférerez peut-être par exemple
def accelerate(self, acceleration_time):
self.speed += delta_speed(self.acceleration, acceleration_time)
def delta_speed(acceleration, acceleration_time): return acceleration*acceleration_time
La note delta_speed
est au même niveau d'indentation que Car
parce qu'elle ne vit pas dans une classe, donc elle n'a aucun des paramètres cachés qui vous dérangent. (En tant qu'exercice, vous pouvez réécrire cette approche pour l'utiliser à la =
place de +=
; cela n'a pas de rapport avec le point soulevé ici.)
Certaines de vos observations (sinon la plupart) sont valables, mais les conclusions que vous en tirez sont trop extrêmes.
- Il est étroitement lié à la source de données concrète. (il lit le datetime actuel de la machine sur laquelle il s'exécute)
Correct. La valeur de date doit être passée en paramètre ou une dépendance de type horloge doit être injectée.
Notez que l'injection de dépendances nécessite une classe et une méthode non statiques. Plus à ce sujet plus tard.
Prenez note de cette dernière suggestion (injecter une dépendance). Votre question va à l'encontre de cette idée, et c'est là que votre observation déraille. Plus à ce sujet plus tard.
- Il viole le principe de responsabilité unique (SRP).
Je ne vois pas comment cela fonctionne, et vous n'avez pas non plus expliqué pourquoi vous pensez que c'est le cas. Cette méthode fait une chose. SRP ne se concentre pas sur l'injection de dépendances, SRP se concentre sur la logique contenue dans la classe. Cette classe a un objectif strictement défini: générer une étiquette conviviale pour l'heure actuelle.
Juste pour être clair: le code peut être amélioré, mais SRP n'est pas ce qui vient à l'esprit comme une violation ici.
L'argument selon lequel la récupération de la valeur datetime est une responsabilité discrète est un argument ardu. Toute responsabilité peut être subdivisée en responsabilités plus petites, mais il y a une limite entre ce qui est raisonnable et ce qui est excessif. En supposant que la méthode indique que l' heure actuelle de la journée est évaluée, il ne s'agit pas d'une violation SRP.
- Il ment sur les informations nécessaires pour faire son travail. Les développeurs doivent lire chaque ligne du code source réel pour comprendre quelles entrées cachées sont utilisées ...
C'est discutable. Quand je vois GetTimeOfDay
et qu'il ne prend pas clairement une valeur datetime (que ce soit en tant que paramètre de méthode ou en tant que dépendance), alors l'inférence logique est que l'heure actuelle est utilisée.
Même sémantiquement, «obtenir l'heure» suggère que vous obtenez l' heure actuelle , donc je ne vois pas vraiment de problème ici avec le nom.
... et d'où ils viennent. ...
Ceci, je suis d'accord. Vous ne savez pas si cela dépend de l'horloge système, ou d'une API basée sur le cloud ou ... Ceci est résolu lorsque vous l'injectez en tant que dépendance ou l'ajoutez en tant que paramètre de méthode.
La signature de la méthode seule ne suffit pas pour comprendre le comportement de la méthode.
La plupart des principes de POO (SOLID entre autres) se concentrent sur les classes , pas sur les méthodes. Vous ne devez pas observer les méthodes par elles-mêmes, vous devez les voir comme des opérations sur une classe, et plus spécifiquement sur une instance connue de cette classe.
En ce qui concerne la lisibilité du code, vous pouvez supposer que quiconque appelle une méthode de classe sur une instance (objet) de cette classe sait également comment cet objet a été construit en premier lieu. Ce n'est pas toujours le cas, mais quand ce n'est pas le cas, cela signifie que l'appelant a consenti à déléguer la construction de l'objet.
Ce n'est pas votre responsabilité (vous = le concepteur de la classe consommée). Vous ne pouvez pas et ne devez pas essayer de gérer la façon dont vos consommateurs délèguent leur propre travail en interne.
Lorsque la source de la valeur datetime a été refactorisée pour être une dépendance injectée ou un paramètre de méthode, le problème signalé dans votre troisième puce est nul et non avenu.
Alors, comment gérer
self
...?
"traiter" implique qu'il s'agit d'un problème ou d'un élément indésirable. Votre discours sur self
et les problèmes allégués avec celui-ci comporte une nuance d'aversion pour le concept d'état orienté objet.
Si c'est ce que vous ressentez et que vous ne voulez pas changer votre façon de penser, ça va aussi. La programmation est un concept abstrait de l'esprit et différentes approches existent pour résoudre le même problème. Dans ce cas, vous devriez envisager de passer à la programmation fonctionnelle au lieu de la programmation orientée objet, pour une raison majeure:
self
est au cœur de la POO .
Les objets suivent l'état. C'est ce qu'ils font. Si ce n'est pas le cas, alors votre base de code n'existe que de méthodes, et toutes ces méthodes pourraient être rendues statiques.
self
est le mot-clé qui vous permet d'accéder à l'état actuel de l'objet. Sans self
, vous êtes effectivement incapable de stocker et de récupérer l'état des objets, et nous reviendrions donc à un système où tout n'est qu'une collection de méthodes statiques.
Remarque: dans votre question, vous avez indiqué que vous jugez chaque méthode individuellement. Cela correspond en fait à la façon dont vous travaillez avec les méthodes statiques, mais ce n'est pas compatible avec la façon dont vous devriez penser au code orienté objet.
... et comment l'utiliser correctement?
Cela nous ramène à la partie où j'ai dit qu'il fallait observer les choses au niveau de la classe , pas au niveau de la méthode.
La façon la plus simple d'y penser est que l'état stocké dans un objet (c'est-à-dire via self
, généralement effectué via le constructeur) a été configuré une fois et est réutilisable par toutes les méthodes de cette classe. Par exemple:
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;
}
}
Remarquez que je n'avais qu'à dire à SundayChecker
quelle horloge elle devait utiliser une seule fois , mais je suis alors en mesure de vérifier à plusieurs reprises l'heure actuelle et de confirmer si c'est dimanche ou non.
Ceci n'est qu'un exemple simple, mais il présente la nature de base de la POO.
Remarque: il y a beaucoup plus d'arguments en faveur de l'utilisation de l'état d'objet, mais c'est le plus facile à saisir afin de déplacer votre esprit dans un cadre compatible OOP.
Ceci est beaucoup trop large pour une explication approfondie de la POO et de la façon dont elle doit être utilisée. Je vous suggère de rechercher des didacticiels et des exercices de POO qui vous apprennent à utiliser (et à savoir comment tirer parti) du code orienté objet.
C'est une petite méthode et c'est facile à lire, mais qu'en est-il des méthodes avec des centaines de lignes qui lisent et attribuent à
self
de nombreux endroits?
Tout peut être surpuissant. Ce n'est pas parce que la POO a son utilité qu'elle ne peut pas être abusée ou mal écrite.
- POO ou pas, les méthodes avec des centaines de lignes sont intrinsèquement un problème de qualité de code.
- Bien que l'état de l'objet puisse être manipulé et que ce ne soit pas intrinsèquement une mauvaise idée, avoir un objet dont l'état est constamment modifié au point de ne plus pouvoir garder une trace de son état est également un problème de qualité du code.
Mais ce ne sont pas des arguments contre l'utilisation de la POO comme règle générale. C'est comme dire que personne ne devrait jamais utiliser un marteau parce que vous avez vu votre père frapper son pouce avec un marteau.
Des erreurs se produisent mais l'existence d'erreurs ne réfute pas le concept dans son ensemble.
Il est mauvais d'appeler time of day "now" dans une méthode qui calcule également quelque chose comme la chaîne time of day comme vous l'avez montré. Ceci est dû au fait,
si vous voulez connaître la chaîne d'heure d'une autre heure que maintenant, vous ne pouvez tout simplement pas utiliser cette méthode - cela rend cette méthode beaucoup moins utile et vous devez répéter sa logique pour utiliser cette logique d'une autre manière.
si vous voulez connaître la chaîne de l'heure mais aussi l'heure actuelle, vous finissez par appeler l'heure de la journée maintenant deux fois, et les deux appels distincts à "maintenant" pourraient facilement être des valeurs différentes, où les auteurs du code s'attendent probablement à ce qu'ils correspondent exactement.
Idéalement, si vous avez besoin de l'heure "maintenant", celle-ci n'est obtenue qu'une seule fois (quel que soit le cas) et passée en paramètre à tout code traitant de l'heure "actuelle".