循环读取/处理网络太紧了?

时间:2012-08-03 20:07:00

标签: c# loops

我的应用程序启动并绘制MDI Parent表单。此时,它等待事件发生。为了启动主循环,用户必须单击工具栏按钮。这称为主处理程序,这是代码。

    private void MainAPRSTW_Main()
        {
        //frmSplash objfrmSplash = new frmSplash();
        //objfrmSplash.ShowDialog();

        //this is the main list
        TopNodeList MainList = new TopNodeList(this);

        //setup the filter
        FilterList CallFilter = new FilterList();
        CallFilter.ReadPassList();
        CallFilter.ReadBlockList();

        //load what's in the cache
        MainList.ReadCache(this);

        //initiate the Socket
        Socket server = new Socket(AddressFamily.InterNetwork,
                                   SocketType.Stream,
                                   ProtocolType.Tcp);

        //initiate the endpoints
        IPEndPoint AGWPEServer=new IPEndPoint(IPAddress.Parse(AGWIPAddress), AGWPort);

        try
            {
            server.Connect(AGWPEServer);
            }
        catch (SocketException e)
            {
            System.Windows.Forms.MessageBox.Show("Unable to connect to server.");
            junk = e.ToString();//this keeps the compiler from complaining
            return;
            }

        //send command to enable monitor frames
        try
            {
            server.Send(MonitorCommand, 36, SocketFlags.None);
            }
        catch (SocketException e)
            {
            System.Windows.Forms.MessageBox.Show("Unable to write monitor command.");
            junk = e.ToString();//this keeps the compiler from complaining
            return;
            }

        //-problem code starts here--------------------------------------------
        while (TerminateFlag == false)
            {
            if (server.Available > 35)
                {
                //only go here if we have enough data available to process
                //get the next header
                TelemetryHeader.readHeader(server);

                //read the rest of the packet and ignore
                int recv = server.Receive(AGWServerData,
                                          TelemetryHeader.MessageSize,
                                          SocketFlags.None);

                //Is it a telemetry packet?
                if (Detector.isTelemetry(AGWServerData,TelemetryHeader.SourceCallsign))
                    {
                    if (CallFilter.Find(TelemetryHeader.SourceCallsign))
                        MainList.ProcessPacket(AGWServerData, Detector, this);
                    }
                }//close If (server.Available).......
            }//close while (terminateflag)......
        //-problem code stops here-------------------------------------------
        }//close MainAPRSTW_Main

您可以看到我在哪里描述了问题代码。如果我保留此代码,子表单只会部分填充,程序将无响应。这是一个屏幕截图。

http://www.blandranch.net/Files/broke.jpg

如果我注释掉问题代码,那么所有子代表都会绘制。这是屏幕截图。

http://www.blandranch.net/Files/working.jpg

所以我很明显需要做一些与众不同的事情。我猜这个循环太紧了。是这样的情况还是还有别的东西是错的?

问题是,什么是正确的方法?我是否使用Backgroundworker?还有另一种方式吗?

1 个答案:

答案 0 :(得分:0)

使用BackgroundWorker或线程池。永远不应该阻止UI线程。进入这样的循环会阻止UI线程处理UI更新消息(如重绘控件或更新新条目)。

在您的情况下,我建议使用BeginConnect和BeginSend而不是Connect,并使用异步IO来处理后台。