2

我目前正在编写一个与 I/O 设备交互的程序,并且需要一种每隔 x 秒轮询一次设备的方法,以检查输入/输出连接。为此,我使用了一个按钮,该按钮使用计时器和计时器事件句柄创建了一个线程来进行轮询。但是,我注意到在任务管理器中,随着时间的推移,它正在慢慢消耗更多的内存。下面是一些(我认为)与我的问题相关的代码片段。

创建线程的按钮:

private void btnConnect_Click(object sender, EventArgs e)
    {
        new Thread(start).Start();
    }

包含计时器的线程:

        public void start()
    {
        timer = new System.Timers.Timer(1000);
        timer.Elapsed += new ElapsedEventHandler(timerElapsed);
        timer.Enabled = true;
    }

ElapsedEventHandler:

public void timerElapsed(object sender, ElapsedEventArgs e)
    {
        connect();
    }

最后是方法connect();:

public void connect()
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(txtIPseg1.Text + "." + txtIPseg2.Text + "." + txtIPseg3.Text + "." + txtIPseg4.Text);
        int Port = int.Parse(txtPort.Text);
        string address = sb.ToString();

        //send data
        byte[] bData = new byte[71];
        bData[0] = 240;
        bData[1] = 240;
        bData[2] = 0;
        bData[3] = 1;
        bData[68] = 240;
        bData[69] = 240;
        bData[70] = this.CalculateCheckSum(bData);

        try
        {
            byte[] result = this.SendCommandResult(address, Port, bData, 72);
            if (result != null)
            {
                this.Invoke((MethodInvoker)delegate
                {
                    txtOutput1.Text = (result[4] == 0x00 ? "HIGH" : "LOW"); // runs on UI thread
                });
            }
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.ToString());
        }

    }

我很确定泄漏要么来自计时器,要么来自方法 connect(); 中使用的匿名委托,有人有什么想法吗?

4

1 回答 1

2

每次单击按钮时,您都会创建一个新计时器。此外,您没有保留对它的引用,因此它将被垃圾收集器销毁。无需在新线程上启动计时器,因为计时器将在新线程上引发 Elapsed 事件。

class Form1 ...
{
    private System.Timers.Timer timer = null;

    public void start()
    {
    if (timer == null)
        {
        timer = new System.Timers.Timer(1000);
        timer.Elapsed += new ElapsedEventHandler(timerElapsed);
        }
    timer.Enabled = true;
    }

    ...
}

就内存泄漏而言,我不会仅仅因为您在应用程序运行时看到内存使用量看似随机波动就认为存在内存泄漏。当您在 .NET 等复杂框架中运行时,这是正常行为。每次计时器触发时,它都会调用您的 connect 方法来创建新对象。这些对象将保留在内存中,直到垃圾收集器开始清理它们。因此,看到内存慢慢上升,然后在几分钟后突然又下降,这并不奇怪。我不会怀疑有问题,除非它在更长的时间内不断失控。

您使用 StringBuilder 类的方式也很奇怪。你在做什么:

StringBuilder sb = new StringBuilder();
sb.Append(txtIPseg1.Text + "." + txtIPseg2.Text + "." + txtIPseg3.Text + "." + txtIPseg4.Text);
string address = sb.ToString();

没有比仅仅这样做更好(实际上它更糟):

string address = txtIPseg1.Text + "." + txtIPseg2.Text + "." + txtIPseg3.Text + "." + txtIPseg4.Text;

如果您正在寻找一种更有效且可能更易于阅读的方法,请尝试类似这样的方法

string address = string.Format("{0}.{1}.{2}.{3}", txtIPseg1.Text, txtIPseg2.Text, txtIPseg3.Text, txtIPseg4.Text);

然而,没有什么东西会引起内存泄漏,所以除非你有充分的理由这样想,否则我不会担心。

于 2012-05-18T14:40:51.200 回答