C# :ключевое слово 'is' и проверка на отсутствие
Вопрос
Это глупый вопрос, но вы можете использовать этот код, чтобы проверить, относится ли что-то к определенному типу...
if (child is IContainer) { //....
Есть ли более элегантный способ проверить наличие экземпляра "NOT"?
if (!(child is IContainer)) { //A little ugly... silly, yes I know...
//these don't work :)
if (child !is IContainer) {
if (child isnt IContainer) {
if (child aint IContainer) {
if (child isnotafreaking IContainer) {
Да, да...глупый вопрос....
Потому что есть какой-то вопрос судя по тому, как выглядит код, это всего лишь простой возврат в начале метода.
public void Update(DocumentPart part) {
part.Update();
if (!(DocumentPart is IContainer)) { return; }
foreach(DocumentPart child in ((IContainer)part).Children) {
//...etc...
Решение
if(!(child is IContainer))
- единственный оператор, который должен идти (нет оператора IsNot
).
Вы можете создать метод расширения, который это делает:
public static bool IsA<T>(this object obj) {
return obj is T;
}
, а затем используйте его для
if (!child.IsA<IContainer>())
И вы можете следить за своей темой:
public static bool IsNotAFreaking<T>(this object obj) {
return !(obj is T);
}
if (child.IsNotAFreaking<IContainer>()) { // ...
<Ч>
Обновление (с учетом фрагмента кода OP):
Поскольку вы на самом деле приводите значение позже, вы можете просто использовать вместо него as
:
public void Update(DocumentPart part) {
part.Update();
IContainer containerPart = part as IContainer;
if(containerPart == null) return;
foreach(DocumentPart child in containerPart.Children) { // omit the cast.
//...etc...
Другие советы
Вы можете сделать это следующим образом:
object a = new StreamWriter("c:\\temp\\test.txt");
if (a is TextReader == false)
{
Console.WriteLine("failed");
}
Почему бы просто не использовать другое?
if (child is IContainer)
{
//
}
else
{
// Do what you want here
}
Это аккуратно, это знакомо и просто?
У вас все хорошо, но вы могли бы создать набор методов расширения, чтобы сделать " более элегантный способ проверить наличие экземпляра 'NOT'. " ; р>
public static bool Is<T>(this object myObject)
{
return (myObject is T);
}
public static bool IsNot<T>(this object myObject)
{
return !(myObject is T);
}
Тогда вы могли бы написать:
if (child.IsNot<IContainer>())
{
// child is not an IContainer
}
Уродливый?Я не согласен.Единственный другой способ (лично я считаю, что это "уродливее"):
var obj = child as IContainer;
if(obj == null)
{
//child "aint" IContainer
}
Оператор is
оценивает логический результат, поэтому вы можете делать все, что в противном случае вы могли бы делать в bool. Чтобы отрицать это, используйте оператор !
. Зачем вам нужен другой оператор только для этого?
Метод расширения IsNot<T>
- это хороший способ расширить синтаксис. Имейте в виду,
var container = child as IContainer;
if(container != null)
{
// do something w/ contianer
}
работает лучше, чем что-то вроде
if(child is IContainer)
{
var container = child as IContainer;
// do something w/ container
}
В вашем случае это не имеет значения, поскольку вы возвращаетесь из метода. Другими словами, будьте осторожны, чтобы не выполнять как проверку типа, так и преобразование типа сразу после.
Хотя оператор IS обычно является наилучшим способом, есть альтернатива, которую вы можете использовать при некоторых обстоятельствах. Вы можете использовать оператор as и проверить на ноль.
MyClass mc = foo as MyClass;
if ( mc == null ) { }
else {}
Хотя это не устраняет проблему с круглыми скобками, ради людей, попадающих сюда через Google, следует отметить, что существует более новый синтаксис (начиная с C # 7), чтобы сделать остальную часть вашего кода немного чище: р>
if (!(DocumentPart is IContainer container)) { return; }
foreach(DocumentPart child in container.Children) {
...
Это позволяет избежать двойного приведения, проверки на нуль и наличия переменной, доступной в областях, где она может быть нулевой.
Это еще не было упомянуто. Это работает, и я думаю, что это выглядит лучше, чем при использовании !(child is IContainer)
if (part is IContainer is false)
{
return;
}
if (child is IContainer ? false : true)