Подтвердить что ты не робот

Внедрение пользовательского типа в классы библиотеки .NET

Я хотел бы реализовать преобразование между двумя библиотечными классами Convert.ChangeType в С#. Я не могу изменить ни один из двух типов. Например, преобразование между Guid и byte [].

Guid g = new Guid();
object o1 = g;
byte[] b = (byte[]) Convert.ChangeType(o1, typeof(byte[])); // throws exception

Я знаю, что Guid предоставляет метод ToByteArray(), но я бы хотел, чтобы это вызвало, когда Guid был преобразован в byte []. Причина этого заключается в том, что преобразование также происходит в библиотечном коде (AseDataAdapter), который я не могу изменить. Можно ли определить правило преобразования между двумя типами без изменения исходного кода любого из двух классов?

Я экспериментировал с TypeConverter, но, похоже, не работает:

Guid g = new Guid();
TypeConverter tc = TypeDescriptor.GetConverter(typeof(Guid));
byte[] b2 = (byte[])tc.ConvertTo(g, typeof(byte[])); // throws exception

Переменная tc устанавливается в System.ComponentModel.GuidConverter, которая не поддерживает преобразования в байт []. Могу ли я иметь два TypeConverters для одного и того же класса? Даже если бы я мог, мне не нужно было бы добавлять атрибут к исходному коду класса для назначения TypeConverter?

Спасибо

4b9b3361

Ответ 1

Вы можете изменить зарегистрированный TypeConverter для чего-либо, используя TypeDescriptor.AddAttributes; это не совсем то же самое, что и Convert.ChangeType, но этого может быть достаточно:

using System;
using System.ComponentModel;
static class Program
{
    static void Main()
    {
        TypeDescriptor.AddAttributes(typeof(Guid), new TypeConverterAttribute(
            typeof(MyGuidConverter)));

        Guid guid = Guid.NewGuid();
        TypeConverter conv = TypeDescriptor.GetConverter(guid);
        byte[] data = (byte[])conv.ConvertTo(guid, typeof(byte[]));
        Guid newGuid = (Guid)conv.ConvertFrom(data);
    }
}

class MyGuidConverter : GuidConverter
{
    public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
    {
        return sourceType == typeof(byte[]) || base.CanConvertFrom(context, sourceType);
    }
    public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
    {
        return destinationType == typeof(byte[]) || base.CanConvertTo(context, destinationType);
    }
    public override object ConvertFrom(ITypeDescriptorContext context, System.Globalization.CultureInfo culture, object value)
    {
        if (value != null && value is byte[])
        {
            return new Guid((byte[])value);
        }
        return base.ConvertFrom(context, culture, value);
    }
    public override object ConvertTo(ITypeDescriptorContext context, System.Globalization.CultureInfo culture, object value, Type destinationType)
    {
        if (destinationType == typeof(byte[]))
        {
            return ((Guid)value).ToByteArray();
        }
        return base.ConvertTo(context, culture, value, destinationType);
    }
}

Ответ 2

Если код, выполняющий преобразование, поддерживает TypeConverter, вы можете использовать TypeConverterAttribute на уровне сборки.

Ответ 3

System.ComponentModel.ICustomTypeDescriptor

Да, это возможно. Прочтите документацию по MSDN для соответствующей информации, чтобы "ввести" это в запущенную программу. (TypeDescriptor предоставляет метод IIRC).

Ответ 4

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