c# - 无法从Android应用程序接收Windows 10 UWP中的UDP广播

标签 c# xamarin.forms uwp broadcast udpclient

  • 我正在为共享代码使用 Xamarin.forms 和 .net Standard 2 编写 UWP 和 Android 应用。
  • 我正在尝试从两个应用程序发送和接收 UPD 广播,但问题是我只能从 Android 应用程序按预期发送和接收,而 UWP 应用程序无法接收 UDP 广播数据包?
  • 我在 Windows 上安装了 wireshark,它可以接收来自这两个应用程序的所有包。
  • 我为我的 app.uwp.exe 添加了防火墙规则(允许任何网络、任何端口、任何协议(protocol)、任何 ip)我重新启动了电脑,我的 UWP 应用程序仍然无法从 android 应用程序接收。

代码:

    using System.Threading.Tasks;
    using Xamarin.Forms;
    using Xamarin.Forms.Xaml;
    using System.Net.Sockets;
    using System.Net;    

    [XamlCompilation(XamlCompilationOptions.Compile)]
    public partial class ItemsPage : ContentPage
    {
     public async static Task<String> UDP_receive()
     {
        var myUDP = new UdpClient(8888);
        myUDP.EnableBroadcast = true;            
        var myResult = await myUDP.ReceiveAsync();
        myUDP.Close();
        return Encoding.ASCII.GetString(myResult.Buffer) + " : " + myResult.RemoteEndPoint.Address.ToString();
     }

     public async static Task UDP_Send()
     {
        var myUDP = new UdpClient();
        var RequestData = Encoding.ASCII.GetBytes(DateTime.Now.ToLongTimeString());
        myUDP.EnableBroadcast = true;
        await myUDP.SendAsync(RequestData, RequestData.Length, new IPEndPoint(IPAddress.Broadcast, 8888));
        myUDP.Close();
     }
    void Button1_click(object sender, EventArgs e)
     {
        StartListen();
        StartSending();   
     }
     async Task StartListen()
     {
        Exception eXX = null;
        while (eXX == null)
        {
            try
            {
                myText.Text += Environment.NewLine + await UDP_receive();
            }
            catch (Exception ex2)
            {
                eXX = ex2;
                log(ex2);
            }
        }
     }

     async Task StartSending()
     {
        Exception eXX = null;
        while (eXX == null)
        {
            try
            {
                await MyData.UDP_Send();
                await Task.Delay(1000);
            }
            catch (Exception ex2)
            {
                eXX = ex2;
                log(ex2);
            }
        }
     } 
   }    

安卓输出:
上午 11:57:11:192.168.1.155
上午 11:57:11:192.168.1.100
上午 11:57:12:192.168.1.155
上午 11:57:12:192.168.1.100
上午 11:57:13:192.168.1.155
上午 11:57:13:192.168.1.100
....
注:192.168.1.100为Windows PC IP

UWP 输出:
上午 11:57:11:192.168.1.100
上午 11:57:12:192.168.1.100
上午 11:57:13:192.168.1.100
上午 11:57:14:192.168.1.100
上午 11:57:15:192.168.1.100
....

最佳答案

要接收 UDP 广播,请尝试将创建与绑定(bind)分开。例如,

            var myUDP = new UdpClient()
            {
                ExclusiveAddressUse = false,
                EnableBroadcast = true
            };

            IPEndPoint localIPEndPoint = new IPEndPoint(myIP.BroadcastIPAddress, 8888);
            myUDP.Client.Bind(localIPEndPoint);

根据我的经验,短格式构造函数(如 UdpClient(8888))似乎过早地绑定(bind)了底层套接字。这可以防止更改 EnableBroadcast 属性。

此外,从平台到平台,我发现接收广播的唯一可靠方法是使用子网的实际广播地址。如下所示,捕获这些信息很容易(?)。

    public class BroadcastAddress
    {
        public IPAddress IPAddress { get; set; }
        public IPAddress BroadcastIPAddress { get; set; }
        public NetworkInterfaceType NetworkInterfaceType { get; set; }
    }

    public static IEnumerable<BroadcastAddress> GetBroadcastAddresses()
    {
        List<BroadcastAddress> broadcastAddresses = new List<BroadcastAddress>();

        NetworkInterface[] networkInterfaces = NetworkInterface.GetAllNetworkInterfaces();
        foreach (NetworkInterface networkInterface in networkInterfaces)
        {
            if (networkInterface != null 
                && (networkInterface.NetworkInterfaceType == NetworkInterfaceType.Wireless80211
                ||  networkInterface.NetworkInterfaceType == NetworkInterfaceType.Ethernet))
            {
                IPInterfaceProperties iPInterfaceProperties = networkInterface.GetIPProperties();
                UnicastIPAddressInformationCollection unicastIPAddressInformationCollection = iPInterfaceProperties.UnicastAddresses;
                foreach (UnicastIPAddressInformation unicastIPAddressInformation in unicastIPAddressInformationCollection)
                {
                    if (unicastIPAddressInformation.Address.AddressFamily == System.Net.Sockets.AddressFamily.InterNetwork)
                    {
                        IPAddress broadcastIPAddress = 
                            new IPAddress(unicastIPAddressInformation.Address.Address | (UInt32.MaxValue ^ (UInt32)(unicastIPAddressInformation.IPv4Mask.Address)));

                        broadcastAddresses.Add(new BroadcastAddress()
                        {
                            IPAddress = unicastIPAddressInformation.Address,
                            BroadcastIPAddress = broadcastIPAddress,
                            NetworkInterfaceType = networkInterface.NetworkInterfaceType
                        });
                    }
                }
            }
        }

        return broadcastAddresses;
    }

当您处理移动设备时,您只需使用 Wireless80211 广播地址。在笔记本电脑或工作站上,它可能会变得稍微复杂一些。这些设备可以有多个网络接口(interface)和每个接口(interface)上的多个子网。因此,返回列表的原因。而且,这意味着您可能需要多个 UdpClient 实例,每个子网一个。

这对 OP 来说可能为时已晚。希望这些信息能帮助其他一些疲惫的搜索者尝试让广播正常工作(并且他们可以停止用额头敲 table )。

关于c# - 无法从Android应用程序接收Windows 10 UWP中的UDP广播,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50621974/

相关文章:

c# - 如何查找列表的元素是否在另一个列表中?

c# - 从 varbinary case sql server c# 中读取图像

xamarin.forms - VS2017 是否支持 Xamarin 表单中的“编辑并继续”?

c# - 我想将我的 Iphone 连接到 MAC mini,并使用 Xamarin UI 测试从我的 Windows 计算机远程运行测试

c# - 页面不包含 X 的定义

c# - 通过 Xamarin 在 UWP 上使用文本转语音

c# - Linq - SQL 将参数传递给 GroupBy()

C# - 通过 if 条件检查值的数据类型

c# - Xamarin布局如何在屏幕的左下角和右下角放置标签

.net - 从设备门户将 appxbundle 部署到 Hololens 时出错