Обработка цепочек свойств, которые могут содержать нули - PullRequest
4 голосов
/ 22 ноября 2010

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

Например:

var value = prop1.prop2.prop3.prop4;

В порядкеЧтобы обработать возможность нулевого значения в prop1, я должен написать:

var value = prop1 == null ? null : prop1.prop2.prop3.prop4;

Чтобы обработать возможность нулевого значения в prop1 и prop2, я должен написать:

var value = prop1 == null 
            ? null 
            : prop1.prop2 == null ? null : prop1.prop2.prop3.prop4;

или

var value = prop1 != null && prop1.prop2 != null 
            ? prop1.prop2.prop3.prop4 
            : null;

Если я хочу обработать возможность нулевого значения в prop1, prop2 и prop3 или даже в более длинных цепочках свойств, тогда код начинает становиться довольно сумасшедшим.1018 * Должен быть лучший способ сделать это.

Как я могу обработать цепочки свойств, чтобы при обнаружении нулевого значения возвращалось нулевое значение?

Что-то вроде??Оператор был бы великолепен.

Ответы [ 2 ]

7 голосов
/ 26 ноября 2010

Обновление

Начиная с C # 6, решение теперь запекается в языке с помощью нуль-условного оператора ; ?. для свойств и ?[n] для индексаторов.

Нулевой условный оператор позволяет получить доступ к элементам и элементам. только если получатель не равен NULL, в противном случае выдается нулевой результат:

int? length = customers?.Length; // null if customers is null
Customer first = customers?[0];  // null if customers is null

Старый ответ

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

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

Я создал следующий класс на основе этого решения

public static class NullHandling
{
    /// <summary>
    /// Returns the value specified by the expression or Null or the default value of the expression's type if any of the items in the expression
    /// return null. Use this method for handling long property chains where checking each intermdiate value for a null would be necessary.
    /// </summary>
    /// <typeparam name="TObject"></typeparam>
    /// <typeparam name="TResult"></typeparam>
    /// <param name="instance"></param>
    /// <param name="expression"></param>
    /// <returns></returns>
    public static TResult GetValueOrDefault<TObject, TResult>(this TObject instance, Expression<Func<TObject, TResult>> expression) 
        where TObject : class
    {
        var result = GetValue(instance, expression.Body);

        return result == null ? default(TResult) : (TResult) result;
    }

    private static object GetValue(object value, Expression expression)
    {
        object result;

        if (value == null) return null;

        switch (expression.NodeType)
        {
            case ExpressionType.Parameter:
                return value;

            case ExpressionType.MemberAccess:
                var memberExpression = (MemberExpression)expression;
                result = GetValue(value, memberExpression.Expression);

                return result == null ? null : GetValue(result, memberExpression.Member);

            case ExpressionType.Call:
                var methodCallExpression = (MethodCallExpression)expression;

                if (!SupportsMethod(methodCallExpression))
                    throw new NotSupportedException(methodCallExpression.Method + " is not supported");

                result = GetValue(value, methodCallExpression.Method.IsStatic
                                             ? methodCallExpression.Arguments[0]
                                             : methodCallExpression.Object);
                return result == null
                           ? null
                           : GetValue(result, methodCallExpression.Method);

            case ExpressionType.Convert:
                var unaryExpression = (UnaryExpression) expression;

                return Convert(GetValue(value, unaryExpression.Operand), unaryExpression.Type);

            default:
                throw new NotSupportedException("{0} not supported".FormatWith(expression.GetType()));
        }
    }

    private static object Convert(object value, Type type)
    {
        return Expression.Lambda(Expression.Convert(Expression.Constant(value), type)).Compile().DynamicInvoke();
    }

    private static object GetValue(object instance, MemberInfo memberInfo)
    {
        switch (memberInfo.MemberType)
        {
            case MemberTypes.Field:
                return ((FieldInfo)memberInfo).GetValue(instance);
            case MemberTypes.Method:
                return GetValue(instance, (MethodBase)memberInfo);
            case MemberTypes.Property:
                return GetValue(instance, (PropertyInfo)memberInfo);
            default:
                throw new NotSupportedException("{0} not supported".FormatWith(memberInfo.MemberType));
        }
    }

    private static object GetValue(object instance, PropertyInfo propertyInfo)
    {
        return propertyInfo.GetGetMethod(true).Invoke(instance, null);
    }

    private static object GetValue(object instance, MethodBase method)
    {
        return method.IsStatic
                   ? method.Invoke(null, new[] { instance })
                   : method.Invoke(instance, null);
    }

    private static bool SupportsMethod(MethodCallExpression methodCallExpression)
    {
        return (methodCallExpression.Method.IsStatic && methodCallExpression.Arguments.Count == 1) || (methodCallExpression.Arguments.Count == 0);
    }
}

Это позволяет мне написать следующее:

var x = scholarshipApplication.GetValueOrDefault(sa => sa.Scholarship.CostScholarship.OfficialCurrentWorldRanking);

x будет содержать значение scholarshipApplication.Scholarship.CostScholarship.OfficialCurrentWorldRanking или null, если какое-либо из свойств в цепях будет возвращать ноль по пути.

2 голосов
/ 22 ноября 2010

Я использую метод расширения IfNotNull , чтобы справиться с этим.

Это не самая красивая вещь в мире (я бы немного испугалась, если бы увидела, что она идет на 4 слоя глубиной)) но это работает для небольших случаев.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...