我想使用通用的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绝对是不的选项。
答案 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;
}
}