Есть ли случай, когда проверка параметров может считаться избыточной?
Первое, что я делаю в публичном методе, - это проверяю каждый параметр, прежде чем они получат какой-либо шанс привыкнуть, обойтись или сослаться на него, а затем вызвать исключение, если какой-либо из них нарушит контракт. Я считаю, что это очень хорошая практика, так как она позволяет вам поймать нарушителя в момент совершения нарушения, но затем довольно часто я пишу очень простой метод получения / индексации, такой как этот:
private List<Item> m_items = ...;
public Item GetItemByIdx( int idx )
{
if( (idx < 0) || (idx >= m_items.Count) )
{
throw new ArgumentOutOfRangeException( "idx", "Invalid index" );
}
return m_items[ idx ];
}
В этом случае параметр index напрямую связан с индексами в списке, и я точно знаю (например, документацию), что сам список будет делать то же самое и будет выдавать то же исключение. Должен ли я удалить эту проверку или мне лучше оставить ее в покое?
Я хотел знать, что вы, ребята, думаете, так как сейчас я нахожусь в процессе рефакторинга большого проекта, и я обнаружил много случаев, подобных описанному выше.
Заранее спасибо.
3 ответа
Это не просто вопрос вкуса, рассмотрим
if (!File.Exists(fileName)) throw new ArgumentException("...");
var s = File.OpenText(fileName);
Это похоже на ваш пример, но есть несколько причин (параллелизм, права доступа), почему OpenText()
метод все еще может потерпеть неудачу, даже с ошибкой FileNotFound. Таким образом, проверка Exists просто дает ложное чувство безопасности и контроля.
Это умонастроение, когда вы пишете метод GetItemByIdx, он, вероятно, выглядит весьма разумным. Но если вы просматриваете случайный фрагмент кода, обычно есть много предположений, которые вы можете проверить, прежде чем продолжить. Просто не практично проверять их все снова и снова. Мы должны быть избирательными.
Поэтому в простом методе передачи, таком как GetItemByIdx, я бы поспорил с избыточными проверками. Но как только функция добавляет больше функциональности или если есть очень явная спецификация, которая говорит что-то о idx, этот аргумент оборачивается.
Как правило, должно быть выброшено исключение, когда четко определенное условие нарушено и это условие актуально на текущем уровне. Если условие принадлежит более низкому уровню, то пусть этот уровень обрабатывает его.
Я бы проверил параметры только тогда, когда это привело бы к некоторому улучшению поведения кода. Поскольку в этом случае вы знаете, что проверка будет выполняться самим списком, то ваша собственная проверка является излишней и не дает никаких дополнительных значений, поэтому я бы не стал беспокоиться.
Это правда, что, возможно, вы дублировали работу, которая уже была выполнена в API, но сейчас она есть. Если ваша структура обработки ошибок работает и надежна, и не вызывает проблем с производительностью ( профилирование IYF), то я считаю, что оставлю ее и постепенно откажусь от нее, если у вас будет время. Это не похоже на главный приоритет!