如何使用BinaryWriter编写列表?

时间:2011-07-20 22:20:04

标签: c# binarywriter

我想使用通用的WriteList(List值)函数来使用BinaryWriter编写List。这是我正在使用的代码:

public void WriteList<T>(List<T> value)
{
    for (int i = 0; i < value.Count; i++)
    {
        _writer.Write(value[i]);
    }
}

我收到的错误是:

 Error  1   The best overloaded method match for 'System.IO.BinaryWriter.Write(bool)' has some invalid arguments    
Error   2   Argument 1: cannot convert from 'T' to 'bool'   

BinaryFormatter绝对是的选项。

3 个答案:

答案 0 :(得分:6)

我真的不认为你可以避免使用BinaryFormatter。因为类型T可以是任何复杂类型,T的每个实例都可以表示内存中变量的巨大图形。

因此,唯一的解决方案是将您的T实例转换为byte []格式,最简单的解决方案是:BinaryFormatter

实际上.Write()方法只接受原始类型的原因是它知道如何将它们直接转换为byte [](使用Convert.ToXXX())但是它无法猜测泛型类型T.

作为一种解决方法,您可以定义如下界面:

public interface IBinarySerializable
{
    byte[] GetBytes();
}

然后在你的课程中实现它:

public class MyClass: IBinarySerializable
{
    public int X {get;set;}
    public byte[] GetBytes()
    {
        return BitConverter.GetBytes(X); // and anyother
    }
}

并将您的方法更改为:

public void WriteList<T>(List<T> value) where T:IBinarySerializable
{
    for (int i = 0; i < value.Count; i++)
    {
        _writer.Write(value[i].GetBytes());
    }
}

答案 1 :(得分:4)

如果查看BinaryWriter的文档,你会发现它不接受对象的参数(写入原始类型),并且编译器在重载时正在尽力而为,并且失败了,因为你无法将你的T投射到bool,或BinarwWriter想要的任何其他东西。

您将不得不将对象转换为BinaryWriter将使用的对象。

答案 2 :(得分:4)

你不能将泛型T传递给BinaryWriter.Write - 它有许多特定类型(bool,byte,byte [],int,string等)的重载,但没有泛型。所以你需要自己做,以某种方式类似于下面的代码。

    public void WriteList<T>(List<T> value)
    {
        for (int i = 0; i < value.Count; i++)
        {
            switch (Type.GetTypeCode(typeof(T))){
            //_writer.Write(value[i]);
                case TypeCode.Boolean:
                    _writer.Write((bool)(object)value[i]);
                    break;
                case TypeCode.Byte:
                    _writer.Write((byte)(object)value[i]);
                    break;
                case TypeCode.Char:
                    _writer.Write((char)(object)value[i]);
                    break;
                case TypeCode.Decimal:
                    _writer.Write((decimal)(object)value[i]);
                    break;
                case TypeCode.Double:
                    _writer.Write((double)(object)value[i]);
                    break;
                case TypeCode.Single:
                    _writer.Write((float)(object)value[i]);
                    break;
                case TypeCode.Int16:
                    _writer.Write((short)(object)value[i]);
                    break;
                case TypeCode.Int32:
                    _writer.Write((int)(object)value[i]);
                    break;
                case TypeCode.Int64:
                    _writer.Write((short)(object)value[i]);
                    break;
                case TypeCode.String:
                    _writer.Write((string)(object)value[i]);
                    break;
                case TypeCode.SByte:
                    _writer.Write((sbyte)(object)value[i]);
                    break;
                case TypeCode.UInt16:
                    _writer.Write((ushort)(object)value[i]);
                    break;
                case TypeCode.UInt32:
                    _writer.Write((uint)(object)value[i]);
                    break;
                case TypeCode.UInt64:
                    _writer.Write((ulong)(object)value[i]);
                    break;
                default:
                    if (typeof(T) == typeof(byte[]))
                    {
                        _writer.Write((byte[])(object)value[i]);
                    }
                    else if (typeof(T) == typeof(char[]))
                    {
                        _writer.Write((char[])(object)value[i]);
                    }
                    else
                    {
                        throw new ArgumentException("List type not supported");
                    }

                    break;
            }
        }