2

我有一个不同类型对象的数组,我使用 BinaryWriter 将每个项目转换为其二进制等效项,以便我可以通过网络发送结构。

我目前正在做类似的事情

for ( i=0;i<tmpArrayList.Count;i++)
{
   object x=tmpArrayList[i];
   if (x.GetType() ==  typeof(byte))
   {
      wrt.Write((byte)x);
   }
   ........

问题是,如果错过了它们的类型,我的代码将来可能会中断。

我想做类似的事情。

object x=tmpArrayList[i];
wrt.Write(x);

但除非我每次都做演员,否则它不起作用。

编辑:

在咨询了答案之后,这就是我想出的功能。为了测试这个函数,将数组发送到 syslog。

  private void TxMsg(ArrayList TxArray,IPAddress ipaddress)
  {
     Byte[] txbuf=new Byte[0];
     int sz=0;

     // caculate size of txbuf
     foreach (Object o in TxArray)
     {
        if ( o is String ) 
        {
           sz+=((String)(o)).Length;
        }
        else if ( o is Byte[] )
        {
           sz+=((Byte[])(o)).Length;
        }
        else if ( o is Char[] )
        {
           sz+=((Char[])(o)).Length;
        }
        else // take care of non arrays
        {
           sz+=Marshal.SizeOf(o);
        }
     }
     txbuf = new Byte[sz];

     System.IO.MemoryStream stm_w = new System.IO.MemoryStream( txbuf, 0,txbuf.Length);
     System.IO.BinaryWriter wrt = new System.IO.BinaryWriter( stm_w );

     foreach (Object o in TxArray)
     {
        bool otypefound=false;
        if (o is String) // strings need to be sent one byte per char
        {
           otypefound=true;
           String st=(String)o;
           for(int i=0;i<st.Length;i++)
           {
              wrt.Write((byte)st[i]);
           }
        }
        else
        {
           foreach (MethodInfo mi in typeof(BinaryWriter).GetMethods())
           {
              if (mi.Name == "Write")
              {
                 ParameterInfo[] pi = mi.GetParameters();
                 if ((pi.Length == 1)&&(pi[0].ParameterType==o.GetType()))
                 {
                    otypefound=true;
                    mi.Invoke(wrt, new Object[] { o });
                 }
              }
           }
        }
        if(otypefound==false)
        {
           throw new InvalidOperationException("Cannot write data of type " + o.GetType().FullName);
        }
     }
     IPEndPoint endpoint = new IPEndPoint(ipaddress, 514); //syslog port
     UdpClient udpClient_txmsg = new UdpClient();
     udpClient_txmsg.Send(txbuf, txbuf.Length,endpoint); // send udp packet to syslog             
  }
4

6 回答 6

7

不。强制转换必须在编译时知道,但实际类型只在执行时知道。

但是请注意,有更好的方法来测试调用 GetType 的类型。代替:

if (x.GetType() == typeof(byte))

采用:

if (x is byte)

编辑:回答额外的问题:

“所有类型是什么?” 好吧,我想看看 BinaryWriter 的文档...

“我需要担心字节和字节吗?” 不,byte 是 C# 中 System.Byte 的别名。他们是同一类型。

于 2008-12-11T17:05:16.857 回答
3

乔恩是对的,但我有另一个想法,你可能会觉得有用。您是否考虑过在每个对象的传输中添加另一个字节,然后将该字节用作类型代码,告诉您在另一端将其转换为什么?

于 2008-12-11T17:09:30.733 回答
3

您是否考虑过使用BinaryFormatter而不是 BinaryWriter?

好处

  • 您可以传递object s(即任何东西),因此它解决了您的铸造问题。
  • 自动类型管理(实际上将类型标头写入流)。
  • 也支持复杂的引用类型。

缺点

在内部使用序列化,因此:

  • 应该比较慢吧。
  • 字节流变大(由于类型标头)。
  • 您无法控制字节格式,因此在互操作方案中不是一个选项。
  • 潜在的版本问题(序列化类型的不同程序集版本之间的兼容性)。
  • 需要序列化代码访问权限(部分信任场景相关)。
于 2008-12-11T17:41:26.817 回答
3

这是使用反射的 BinaryWriter 的解决方案。

这基本上扫描 BinaryWriter 以查找名为 Write 的方法,该方法只接受一个参数,然后构建一个字典,其中哪个方法处理哪种类型,然后为每个要写入的对象,找到正确的方法并在 writer 上调用它。

肮脏,您可能应该寻找更好的方法来完成整个事情(不仅仅是写作部分),但它应该适合您当前的需求:

using System.IO;
using System;
using System.Reflection;
using System.Collections.Generic;
namespace ConsoleApplication14
{
    public class Program
    {
        public static void Main()
        {
            Dictionary<Type, MethodInfo> mapping = new Dictionary<Type, MethodInfo>();
            foreach (MethodInfo mi in typeof(BinaryWriter).GetMethods())
            {
                if (mi.Name == "Write")
                {
                    ParameterInfo[] pi = mi.GetParameters();
                    if (pi.Length == 1)
                        mapping[pi[0].ParameterType] = mi;
                }
            }

            List<Object> someData = new List<Object>();
            someData.Add((Byte)10);
            someData.Add((Int32)10);
            someData.Add((Double)10);
            someData.Add((Char)10);
            someData.Add("Test");

            using (FileStream file = new FileStream(@"C:\test.dat", FileMode.Create, FileAccess.ReadWrite))
            using (BinaryWriter writer = new BinaryWriter(file))
            {
                foreach (Object o in someData)
                {
                    MethodInfo mi;
                    if (mapping.TryGetValue(o.GetType(), out mi))
                    {
                        mi.Invoke(writer, new Object[] { o });
                    }
                    else
                        throw new InvalidOperationException("Cannot write data of type " + o.GetType().FullName);
                }
            }
        }
    }
}
于 2008-12-11T18:07:15.140 回答
2

您要的是Dynamic Dispatch,而 C# 3.0 没有它。

您至少应该使用运行时检查来验证您没有丢失类型。

Dictionary如果您有从类型映射到处理函数的映射,您也许可以做一些聪明的事情。您可以在一处填写所有处理功能的映射。与在处理发生的任何地方编写开关相比,您更有可能做到这一点。

于 2008-12-11T17:15:33.107 回答
1

这是需要称为Double Dispatch的情况。

我将假设 wrt 对象是您自己编写的对象(假设它是 Writer 类型)。这是您可以执行的操作:

class Writer
{
    void write(byte b)
    {
        // write bytes here
    }

    void write(Writable something)
    {
        something.writeOn(this);
    }
}

interface Writeable
{
    void writeOn(Writer writer);
}

class SomeObject implements Writeable
{
    private Object someData;
    private Object moreData;

    void writeOn(Writer writer)
    {
        writer.write(convertToByte(someData));
        writer.write(convertToByte(moreData));
    }
}

class AnotherObject implements Writeable
{
    private int x;
    private int y;
    private int z;

    void writeOn(Writer writer)
    {
        writer.write((byte)x);
        writer.write((byte)y);
        writer.write((byte)z);
    }
}

Writer 正在做的是分派回输入,告诉它使用它(Writer)来写入,但是这是为那个对象完成的,这是无法提前知道的。

于 2008-12-11T17:29:38.180 回答