Класс с одним методом - лучший подход? - PullRequest
161 голосов
/ 15 октября 2008

Скажем, у меня есть класс, предназначенный для выполнения одной функции. После выполнения функции ее можно уничтожить. Есть ли основания предпочитать один из этих подходов?

// Initialize arguments in constructor
MyClass myObject = new MyClass(arg1, arg2, arg3);
myObject.myMethod();

// Pass arguments to method
MyClass myObject = new MyClass();
myObject.myMethod(arg1, arg2, arg3);

// Pass arguments to static method
MyClass.myMethod(arg1, arg2, arg3);

Я намеренно размышлял о деталях, пытаясь найти рекомендации для разных ситуаций. Но я не имел в виду простые библиотечные функции, такие как Math.random (). Я думаю больше о классах, которые выполняют какую-то конкретную, сложную задачу, но для этого требуется только один (публичный) метод.

Ответы [ 14 ]

254 голосов
/ 16 октября 2008

Раньше мне нравились служебные классы, заполненные статическими методами. Они сделали большую консолидацию вспомогательных методов, которые в противном случае могли бы привести к избыточности и адскому обслуживанию. Они очень просты в использовании, нет инстанцирования, нет утилизации, просто fire'n'forget. Я думаю, это была моя первая невольная попытка создать сервис-ориентированную архитектуру - множество сервисов без сохранения состояния, которые просто сделали свою работу и ничего больше. Однако по мере развития системы появляются драконы.

Полиморфизм
Скажем, у нас есть метод UtilityClass.SomeMethod, который радостно гудит. Внезапно нам нужно немного изменить функциональность. Большая часть функциональности одинакова, но, тем не менее, мы должны изменить пару частей. Если бы это не был статический метод, мы могли бы создать производный класс и изменить содержимое метода по мере необходимости. Поскольку это статический метод, мы не можем. Конечно, если нам просто нужно добавить функциональность до или после старого метода, мы можем создать новый класс и вызвать внутри него старый класс - но это просто брутто.

Горе интерфейса
Статические методы не могут быть определены через интерфейсы по логическим причинам. И поскольку мы не можем переопределить статические методы, статические классы бесполезны, когда нам нужно передать их через их интерфейс. Это лишает нас возможности использовать статические классы как часть шаблона стратегии. Мы могли бы исправить некоторые проблемы путем передачи делегатов вместо интерфейсов .

Тестирование
Это в основном идет рука об руку с интерфейсом, упомянутым выше. Поскольку наша способность обмениваться реализациями очень ограничена, у нас также будут проблемы с заменой производственного кода тестовым кодом. Опять же, мы можем обернуть их, но для этого потребуется изменить большие части нашего кода, чтобы иметь возможность принимать обертки вместо реальных объектов.

Воспитывает капли
Поскольку статические методы обычно используются в качестве служебных методов, а служебные методы обычно имеют разные цели, мы быстро получим большой класс, заполненный непоследовательной функциональностью - в идеале, каждый класс должен иметь одну цель в системе. Я бы предпочел, чтобы занятия были в пять раз больше, если их цели четко определены.

Параметр ползучести
Начнем с того, что этот маленький симпатичный и невинный статический метод может принимать один параметр. По мере роста функциональности добавляется пара новых параметров. Вскоре добавляются дополнительные параметры, которые являются необязательными, поэтому мы создаем перегрузки метода (или просто добавляем значения по умолчанию на языках, которые их поддерживают). Вскоре у нас есть метод, который принимает 10 параметров. Только первые три действительно необходимы, параметры 4-7 являются необязательными. Но если указан параметр 6, необходимо заполнить 7-9 ... Если бы мы создали класс с единственной целью сделать то, что сделал этот статический метод, мы могли бы решить эту проблему, приняв необходимые параметры в конструктор и позволяющий пользователю устанавливать необязательные значения через свойства или методы для одновременного задания нескольких взаимозависимых значений. Кроме того, если метод вырос до такой степени сложности, он, скорее всего, в любом случае должен находиться в своем собственном классе.

Требование потребителей создать экземпляр класса без причины
Один из наиболее распространенных аргументов: зачем требовать, чтобы потребители нашего класса создавали экземпляр для вызова этого единственного метода, но впоследствии не использовали этот экземпляр? Создание экземпляра класса является очень дешевой операцией в большинстве языков, поэтому скорость не является проблемой. Добавление дополнительной строки кода для потребителя - это низкая стоимость, чтобы заложить основу гораздо более удобного в будущем решения. И, наконец, если вы хотите избежать создания экземпляров, просто создайте одноэлементную оболочку вашего класса, которая позволяет легко использовать повторно - хотя это делает требование, чтобы ваш класс не имел состояния. Если это не состояние, вы все равно можете создавать статические методы-обертки, которые обрабатывают все, но при этом дают вам все преимущества в долгосрочной перспективе. Наконец, вы также можете создать класс, который скрывает создание экземпляров, как если бы он был одиночным: MyWrapper.Instance - это свойство, которое просто возвращает новый MyClass ();

Только ситхи имеют дело с абсолютами
Конечно, есть исключения из моей неприязни к статическим методам. Истинные служебные классы, которые не представляют опасности для вздутия, являются отличными случаями для статических методов - System.Convert в качестве примера. Если ваш проект одноразовый без каких-либо требований для будущего обслуживания, общая архитектура на самом деле не очень важна - статическая или нестатическая, на самом деле не имеет значения - однако скорость разработки имеет значение.

Стандарты, стандарты, стандарты!
Использование методов экземпляра не мешает вам также использовать статические методы, и наоборот. До тех пор, пока есть различие между дифференциацией и стандартизацией. Нет ничего хуже, чем просматривать бизнес-уровень с различными методами реализации.

84 голосов
/ 15 октября 2008

Я предпочитаю статический способ. Поскольку класс не представляет объект, не имеет смысла создавать его экземпляр.

Классы, которые существуют только для их методов, должны оставаться статическими.

17 голосов
/ 15 октября 2008

Если нет причин создавать экземпляр класса для выполнения функции, используйте статическую реализацию. Зачем заставлять потребителей этого класса создавать экземпляр, когда он не нужен.

15 голосов
/ 15 октября 2008

Если вам не нужно сохранять состояние объекта, то нет необходимости создавать его экземпляр в первую очередь. Я бы пошел с единственным статическим методом, которому вы передаете параметры.

Я бы также предупредил против гигантского класса Utils, который имеет десятки несвязанных статических методов. Это может стать неорганизованным и громоздким в спешке. Лучше иметь много классов, каждый из которых имеет несколько связанных методов.

6 голосов
/ 15 октября 2008

Я бы сказал, что лучшим вариантом будет формат статического метода. И я бы тоже сделал класс статическим, чтобы вам не пришлось беспокоиться о случайном создании экземпляра класса.

5 голосов
/ 15 октября 2008

Я действительно не знаю, какова ситуация здесь, но я бы посмотрел на то, чтобы поместить его как метод в один из классов, к которым принадлежат arg1, arg2 или arg3 - Если вы можете семантически сказать, что один из этих классов будет владельцем метода.

4 голосов
/ 16 октября 2008

Я бы предположил, что ответить сложно, основываясь на предоставленной информации.

Мой инстинкт в том, что если у вас будет только один метод и вы собираетесь немедленно отбросить класс, то сделайте его статическим классом, который принимает все параметры.

Конечно, трудно сказать точно, почему вам нужно создать отдельный класс только для этого одного метода. Является ли это типичной ситуацией в "классе коммунальных услуг", как большинство предполагает? Или вы реализуете какой-то класс правил, которых может быть больше в будущем.

Например, этот класс должен быть подключаемым. Затем вы захотите создать интерфейс для вашего единственного метода, а затем вы захотите, чтобы все параметры передавались в интерфейс, а не в конструктор, но вы не хотели бы, чтобы он был статическим.

3 голосов
/ 15 января 2010

Для простых приложений и internal помощников я бы использовал статический метод. Для приложений с компонентами мне нравится Managed Extensibility Framework . Вот выдержка из документа, который я пишу, чтобы описать шаблоны, которые вы найдете в моих API.

  • Услуги
    • Определяется интерфейсом I[ServiceName]Service.
    • Экспортируется и импортируется по типу интерфейса.
    • Единственная реализация предоставляется хост-приложением и используется внутренне и / или расширениями.
    • Методы интерфейса службы являются поточно-ориентированными.

В качестве надуманного примера:

public interface ISettingsService
{
    string ReadSetting(string name);

    void WriteSetting(string name, string value);
}

[Export]
public class ObjectRequiringSettings
{
    [Import]
    private ISettingsService SettingsService
    {
        get;
        set;
    }

    private void Foo()
    {
        if (SettingsService.ReadSetting("PerformFooAction") == bool.TrueString)
        {
            // whatever
        }
    }
}
3 голосов
/ 15 октября 2008

Если этот метод не имеет состояния и вам не нужно передавать его, тогда имеет смысл определить его как статический. Если вам НЕОБХОДИМО обойти метод, вы можете рассмотреть возможность использования делегата вместо одного из предложенных вами подходов.

3 голосов
/ 15 октября 2008

Может ли ваш класс стать статичным?

Если это так, то я бы сделал это классом «Утилиты», в который я поместил бы все свои однофункциональные классы.

...