Eu tenho uma base de código em que o programador costumava encerrar as coisas em áreas que não fazem sentido. Por exemplo, dado um registro de erros, você pode fazer logon via
ErrorLog.Log(ex, "friendly message");
Ele acrescentou vários outros meios para realizar exatamente a mesma tarefa. POR EXEMPLO
SomeClass.Log(ex, "friendly message");
O que simplesmente se vira e chama o primeiro método. Isso adiciona níveis de complexidade sem nenhum benefício adicional. Existe um anti-padrão para descrever isso?
design-patterns
anti-patterns
P.Brian.Mackey
fonte
fonte
Respostas:
Só vale a pena chamar algum hábito ruim de codificação de Antipadrão, se for razoavelmente disseminado.
O resto chamamos de "código de lixo" ...
Se eu sugerisse um nome para esse mau hábito em particular, seria "Transtorno Obsessivo de Abstração" :-)
fonte
Não, não é um anti-padrão, mas eu levantaria os seguintes problemas.
Violação do princípio de responsabilidade única
O SRP diz que uma classe deve ter um motivo para mudar. Adicionar um método de logger significa que a classe também precisará ser alterada se a lógica de log for alterada.
Violação de um
is-a
relacionamentoAs classes base não são caixas de ferramentas nas quais você pode adicionar vários métodos de conveniência.
Fazer isso de maneira eficaz combina todas as classes herdadas às implementações que a classe base possui.
Compare isso com a composição.
fonte
Não que isso o torne aceitável, mas isso poderia ser apenas uma refatoração inacabada. Talvez o SomeClass.log () tenha sua própria lógica e tenha sido implementado primeiro. E depois eles perceberam que deveriam estar usando ErrorLog.Log (). Então, em vez de alterar 100 lugares onde SomeClass.Log () é chamado, eles apenas delegaram para o ErrorLog.Log (). Pessoalmente, eu mudaria todas as referências a ErrorLog.Log () ou, pelo menos, comente SomeClass.log () para dizer por que ele delega da maneira que faz.
Apenas algo a considerar.
fonte
O termo "Código de Lasanha" me vem à mente, embora aparentemente signifique coisas diferentes para pessoas diferentes . Minha interpretação sempre foi "estratificada por causa de ser estratificada".
fonte
Não tenho certeza de descrever isso como uma ofensa ao princípio de responsabilidade única, porque se houver uma alteração na assinatura do método ErrorLog (o método chamado por SomeClass), todo código de cliente que chamar esse método falhará.
Obviamente, há uma maneira de usar herança (ou criar classes que precisam de log implementando uma interface de log) lá.
fonte
Ou poderíamos ver isso como um "momento de aprendizado" :)
Seu desenvolvedor pode estar no meio de uma boa ideia. Suponha que você queira ter o requisito de que todos os seus objetos de negócios sejam capazes de log inteligente. Você pode definir:
Agora interno aos seus objetos, você pode usar o objeto ErrorLog para fazer o log, enquanto o código SomeClass adiciona valor específico ao objeto na mensagem de log. Você pode estender ainda mais suas APIs de log para implementar a funcionalidade de log com base em arquivo, db ou com base em mensagens sem tocar em objetos de negócios.
fonte
Não tenho certeza se isso é automaticamente mau.
Se você está chamando SomeClass.Log, certamente é ruim, mas se o Log estiver sendo usado apenas no WITHIN SomeClass, está cortando o acoplamento e eu chamaria isso de aceitável.
fonte
Na verdade, isso é bastante comum em certos estilos de codificação, e o básico da linha de pensamento não é, por si só, um antipadrão.
Provavelmente, é o resultado de alguém codificando por necessidade, sem conhecimento da base de código mais ampla: "OK, esse código precisa registrar um erro, mas essa função não é o objetivo principal do código. Portanto, preciso de um método / classe que faça isso para mim". Essa é uma boa maneira de pensar; mas, sem saber que o ErrorLog existia, eles criaram SomeClass. Eles então encontraram o ErrorLog em algum momento posterior e, em vez de substituir todos os usos do método que eles colocaram, fizeram o método chamar ErrorLog. É aí que isso se torna um problema.
fonte
Complexidade acidental é a complexidade que surge em programas de computador ou em seu processo de desenvolvimento que não é essencial para o problema a ser resolvido. Embora a complexidade essencial seja inerente e inevitável, a complexidade acidental é causada pela abordagem escolhida para resolver o problema.
http://en.wikipedia.org/wiki/Accidental_complexity
fonte
Pode ser um abuso / mal-entendido do Padrão de Fachada . Vi coisas semelhantes em uma base de código com a qual trabalhei. Os desenvolvedores passaram por uma fase em que eram loucos por padrões de design sem entender os princípios gerais do design de OO.
Um uso incorreto do padrão Fachada resulta em um borrão dos níveis de abstração nas camadas do aplicativo.
fonte
O que esse programador está fazendo é agrupar algum código, para que ele não tenha uma dependência direta do módulo de log. Sem informações mais específicas, não é possível fornecer um padrão mais específico. (Que esses invólucros não são úteis é a sua opinião que é impossível concordar ou discordar sem muito mais informações.)
Os padrões nos quais isso pode se encaixar são Proxy , Delegado , Decorador , Composto ou outros que tenham a ver com agrupar, ocultar ou distribuir chamadas. Pode ser uma dessas coisas em um estado incompleto de desenvolvimento.
fonte
Eu poderia imaginar que seja uma diferença cultural. Talvez haja um bom motivo para ter funcionalidade duplicada nessa base de código específica. Eu poderia imaginar a facilidade de escrever por esse motivo. O programador Python em mim ainda diria que é ruim, já que " Deve haver um-- e de preferência apenas uma maneira --obvious para fazê-lo. ", Mas alguns caras Perl pode estar acostumado ao fato de que " Há mais de um maneira de fazer isso ".
fonte