6

当类型本身未知时,我试图找出支持将整数类型(short/int/long)拆箱为其内在类型的语法。

这是一个完全人为的例子来演示这个概念:

 // Just a simple container that returns values as objects
 struct DataStruct
 {
  public short ShortVale;
  public int IntValue;
  public long LongValue;
  public object GetBoxedShortValue() { return ShortVale; }
  public object GetBoxedIntValue() { return IntValue; }
  public object GetBoxedLongValue() { return LongValue; }
 }

 static void Main( string[] args )
 {

  DataStruct data;

  // Initialize data - any value will do
  data.LongValue = data.IntValue = data.ShortVale = 42;

  DataStruct newData;

  // This works if you know the type you are expecting!
  newData.ShortVale = (short)data.GetBoxedShortValue();
  newData.IntValue = (int)data.GetBoxedIntValue();
  newData.LongValue = (long)data.GetBoxedLongValue();

  // But what about when you don't know?
  newData.ShortVale = data.GetBoxedShortValue(); // error
  newData.IntValue = data.GetBoxedIntValue(); // error
  newData.LongValue = data.GetBoxedLongValue(); // error
 }

在每种情况下,整数类型都是一致的,所以应该有某种形式的语法说“对象包含一个简单的 X 类型,将其返回为 X(即使我不知道 X 是什么)”。因为对象最终来自同一个来源,所以真的不可能有不匹配(短!=长)。

我为这个人为的例子道歉,这似乎是演示语法的最佳方式。

谢谢。

4

4 回答 4

2

好吧,object它本身就是框架知道的最通用的类​​型。is 是盒装值类型(包括原始值类型)还是其他东西都没有关系;如果您想获得更具体的信息,则必须进行类型转换,除非您仍然处于“松散类型”的世界中object(或者,在 C# 4 中,dynamic)。

但是请注意,您可以使用条件列表来实现您想要的:

object boxedValue = GetBoxedValue();
if (typeof(short) == boxedValue.GetType()) {
  newData.ShortValue = (short)boxedValue;
} else if (typeof(int) == boxedValue.GetType()) {
  newData.IntValue = (int)boxedValue;
} else if (typeof(long) == boxedValue.GetType()) {
  newData.LongValue = (long)boxedValue;
} else {
  // not one of those
}

编辑:一个通用的“盒子”也可以做你想做的事:

public class Box<T>: IConvertible where T: struct, IConvertible {
    public static implicit operator T(Box<T> boxed) {
        return boxed.Value;
    }

    public static explicit operator Box<T>(T value) {
        return new Box<T>(value);
    }

    private readonly T value;

    public Box(T value) {
        this.value = value;
    }

    public T Value {
        get {
            return value;
        }
    }

    public override bool Equals(object obj) {
        Box<T> boxed = obj as Box<T>;
        if (boxed != null) {
            return value.Equals(boxed.Value);
        }
        return value.Equals(obj);
    }

    public override int GetHashCode() {
        return value.GetHashCode();
    }

    public override string ToString() {
        return value.ToString();
    }

    bool IConvertible.ToBoolean(IFormatProvider provider) {
        return value.ToBoolean(provider);
    }

    char IConvertible.ToChar(IFormatProvider provider) {
        return value.ToChar(provider);
    }

    sbyte IConvertible.ToSByte(IFormatProvider provider) {
        return value.ToSByte(provider);
    }

    byte IConvertible.ToByte(IFormatProvider provider) {
        return value.ToByte(provider);
    }

    short IConvertible.ToInt16(IFormatProvider provider) {
        return value.ToInt16(provider);
    }

    ushort IConvertible.ToUInt16(IFormatProvider provider) {
        return value.ToUInt16(provider);
    }

    int IConvertible.ToInt32(IFormatProvider provider) {
        return value.ToInt32(provider);
    }

    uint IConvertible.ToUInt32(IFormatProvider provider) {
        return value.ToUInt32(provider);
    }

    long IConvertible.ToInt64(IFormatProvider provider) {
        return value.ToInt64(provider);
    }

    ulong IConvertible.ToUInt64(IFormatProvider provider) {
        return value.ToUInt64(provider);
    }

    float IConvertible.ToSingle(IFormatProvider provider) {
        return value.ToSingle(provider);
    }

    double IConvertible.ToDouble(IFormatProvider provider) {
        return value.ToDouble(provider);
    }

    decimal IConvertible.ToDecimal(IFormatProvider provider) {
        return value.ToDecimal(provider);
    }

    DateTime IConvertible.ToDateTime(IFormatProvider provider) {
        return value.ToDateTime(provider);
    }

    string IConvertible.ToString(IFormatProvider provider) {
        return value.ToString(provider);
    }

    object IConvertible.ToType(Type conversionType, IFormatProvider provider) {
        return value.ToType(conversionType, provider);
    }
}

然后可以使用它来代替object; 它仍然是一个对象引用,但它也是原始结构或原始类型的强类型。

于 2010-05-19T20:23:34.820 回答
2

我不完全确定你想用这个实现什么,但你的 DataStruct 类型是错误的。

我想,并不是它的所有方法都返回 LongValue。

struct DataStruct
{
    public short ShortVale;
    public int IntValue;
    public long LongValue;
    public object GetBoxedShortValue() { return ShortVale; }
    public object GetBoxedIntValue() { return IntValue; }
    public object GetBoxedLongValue() { return LongValue; }
}

否则,您始终可以使用 Convert 类尝试在不同类型之间进行转换。
例如:

Convert.ToInt32(SomeObject);

如果您的意思有所不同,请澄清您的帖子(只需点击编辑按钮并进行编辑)。

顺便说一句,转换 fromobject很容易出错,因为它是所有内容的基本类型。因此,anobject可以是任何东西,这意味着您不能总是安全地将 an 转换object为 int 或任何其他类型。

更多示例:

int value;
try
{
    value = Convert.ToInt32(someObject);
}
catch (FormatException)
{
    // the convertion is unsuccessful
}

这也很有用:

int myValue;
if (!int.TryParse(something, out myValue))
{
    //unsuccessful
}

我希望这有帮助。

于 2010-05-19T20:24:52.520 回答
1

您可以 return dynamic,然后可以将其强制转换为整数类型。

于 2010-05-19T20:22:37.317 回答
0

正如其他人所说,您的示例不起作用,因为您从每个方法返回 LongValue,因此您将在这里得到一个无效的转换异常(装箱长不能转换为短)。

newData.ShortVale = (short)data.GetBoxedShortValue();

但是,使用 C# 4 dynamic,这将起作用(注意对 GetBoxed 方法的修复,dynamic而不是object

// Just a simple container that returns values as objects
struct DataStruct
{
    public short ShortVale;
    public int IntValue;
    public long LongValue;
    public dynamic GetBoxedShortValue() { return ShortValue; }
    public dynamic GetBoxedIntValue() { return IntValue; }
    public dynamic GetBoxedLongValue() { return LongValue; }
}

static void Main( string[] args )
{
    DataStruct data;

    // Initialize data - any value will do
    data.LongValue = data.IntValue = data.ShortVale = 42;

    DataStruct newData;

    newData.ShortVale = (short)data.GetBoxedShortValue();
    newData.IntValue = (int)data.GetBoxedIntValue();
    newData.LongValue = (long)data.GetBoxedLongValue();

    newData.ShortVale = data.GetBoxedShortValue(); // ok
    newData.IntValue = data.GetBoxedIntValue(); // ok
    newData.LongValue = data.GetBoxedLongValue(); // ok
}

请注意,在最后三种情况下您不需要任何演员表。但是,还要注意,如果类型不对齐,例如 in GetBoxedShortValue() { return LongValue; },最后三行将导致无效的强制转换异常。(有趣的是,前三个不会,它们会正常工作,但是当你改dynamic回时object它们会抛出无效的强制转换异常。)

于 2010-05-19T20:30:36.383 回答