264

我有一个类,我想用它来存储另一个类的“属性”。这些属性只有一个名称和一个值。理想情况下,我希望能够添加类型化属性,以便返回的“值”始终是我想要的类型。

类型应该始终是原始类型。这个类继承了一个抽象类,它基本上将名称和值存储为字符串。这个想法是这个子类将为基类添加一些类型安全性(以及为我节省一些转换)。

所以,我创建了一个(大致)这样的类:

public class TypedProperty<DataType> : Property
{
    public DataType TypedValue
    {
        get { // Having problems here! }
        set { base.Value = value.ToString();}
    }
}

所以问题是:

是否有一种“通用”方法可以将字符串转换回原语?

我似乎找不到任何通用接口来链接全面的转换(像ItryParsable这样的东西本来是理想的!)。

4

11 回答 11

422

我不确定我是否正确理解了您的意图,但让我们看看这是否有帮助。

public class TypedProperty<T> : Property where T : IConvertible
{
    public T TypedValue
    {
        get { return (T)Convert.ChangeType(base.Value, typeof(T)); }
        set { base.Value = value.ToString();}
    }
}
于 2008-08-12T09:24:33.203 回答
82

lubos hasko 的方法对 nullables 失败。下面的方法适用于可空值。不过,我没有想出来。我通过谷歌找到它:http: //web.archive.org/web/20101214042641/http : //dogaoztuzun.com/post/C-Generic-Type-Conversion.aspx 归功于“Tuna Toksoz”

用法先:

TConverter.ChangeType<T>(StringValue);  

课程如下。

public static class TConverter
{
    public static T ChangeType<T>(object value)
    {
        return (T)ChangeType(typeof(T), value);
    }

    public static object ChangeType(Type t, object value)
    {
        TypeConverter tc = TypeDescriptor.GetConverter(t);
        return tc.ConvertFrom(value);
    }

    public static void RegisterTypeConverter<T, TC>() where TC : TypeConverter
    {

        TypeDescriptor.AddAttributes(typeof(T), new TypeConverterAttribute(typeof(TC)));
    }
}
于 2009-12-02T14:11:47.603 回答
20

对于许多类型(整数、双精度、日期时间等),有一个静态 Parse 方法。您可以使用反射调用它:

MethodInfo m = typeof(T).GetMethod("Parse", new Type[] { typeof(string) } );

if (m != null)
{
    return m.Invoke(null, new object[] { base.Value });
}
于 2008-08-12T09:32:06.480 回答
12
TypeDescriptor.GetConverter(PropertyObject).ConvertFrom(Value)

TypeDescriptor是具有GetConvertor接受Type对象的方法的类,然后您可以调用ConvertFrom方法来转换该value指定对象的方法。

于 2013-07-09T11:26:13.877 回答
7

受Bob回答的启发,这些扩展还支持空值转换和所有原始转换。

public static class ConversionExtensions
{
        public static object Convert(this object value, Type t)
        {
            Type underlyingType = Nullable.GetUnderlyingType(t);

            if (underlyingType != null && value == null)
            {
                return null;
            }
            Type basetype = underlyingType == null ? t : underlyingType;
            return System.Convert.ChangeType(value, basetype);
        }

        public static T Convert<T>(this object value)
        {
            return (T)value.Convert(typeof(T));
        }
}

例子

            string stringValue = null;
            int? intResult = stringValue.Convert<int?>();

            int? intValue = null;
            var strResult = intValue.Convert<string>();
于 2019-03-13T08:00:42.350 回答
4

您可以使用诸如特征类之类的构造。这样,您将拥有一个参数化的帮助类,该类知道如何将字符串转换为其自己类型的值。那么你的吸气剂可能看起来像这样:

get { return StringConverter<DataType>.FromString(base.Value); }

现在,我必须指出,我对参数化类型的经验仅限于 C++ 及其模板,但我想有一些方法可以使用 C# 泛型来做同样的事情。

于 2008-08-12T09:23:51.737 回答
4

检查静电Nullable.GetUnderlyingType。- 如果基础类型为null,则模板参数不是Nullable,我们可以直接使用该类型 - 如果基础类型不为null,则在转换中使用基础类型。

似乎对我有用:

public object Get( string _toparse, Type _t )
{
    // Test for Nullable<T> and return the base type instead:
    Type undertype = Nullable.GetUnderlyingType(_t);
    Type basetype = undertype == null ? _t : undertype;
    return Convert.ChangeType(_toparse, basetype);
}

public T Get<T>(string _key)
{
    return (T)Get(_key, typeof(T));
}

public void test()
{
    int x = Get<int>("14");
    int? nx = Get<Nullable<int>>("14");
}
于 2015-08-07T17:53:00.530 回答
0
public class TypedProperty<T> : Property
{
    public T TypedValue
    {
        get { return (T)(object)base.Value; }
        set { base.Value = value.ToString();}
    }
}

我使用通过对象进行转换。这有点简单。

于 2010-12-17T14:51:24.167 回答
0

我使用了 lobos 答案,它有效。但是由于文化设置,我在转换双打时遇到了问题。所以我加了

return (T)Convert.ChangeType(base.Value, typeof(T), CultureInfo.InvariantCulture);
于 2015-06-03T07:53:13.430 回答
0

又一个变种。处理 Nullables,以及字符串为空且 T不可为空的情况。

public class TypedProperty<T> : Property where T : IConvertible
{
    public T TypedValue
    {
        get
        {
            if (base.Value == null) return default(T);
            var type = Nullable.GetUnderlyingType(typeof(T)) ?? typeof(T);
            return (T)Convert.ChangeType(base.Value, type);
        }
        set { base.Value = value.ToString(); }
    }
}
于 2017-08-07T21:26:54.250 回答
0

您可以在一行中完成,如下所示:

YourClass obj = (YourClass)Convert.ChangeType(YourValue, typeof(YourClass));

快乐编码;)

于 2020-03-06T13:28:16.957 回答