0

在我们的应用程序中,我们有一个跟踪窗口,我们可以在客户端位置启用它以允许进行一些调试,它被认为是一个静态库。

问题是,当有大量日志消息进入窗口时,它会因 AccessViolation 错误而崩溃。崩溃的代码链接是 RichTextBox.AppendText(..,..,..)。

这是我们创建窗口的地方。

public static void Start(Form parent)
{
  if (_runningThread == null || !_runningThread.IsAlive)
  {

      _runningThread = new Thread(() =>
          {
              _traceView = new TraceView(parent) { Text = "Tracing ~ " + parent.Text };
              Application.Run(_traceView);

          });

      _runningThread.SetApartmentState(ApartmentState.MTA);
      _runningThread.Start();
  }

}

这是我们在文本框中写一行

public void Write(string line, Color color)
{
  try
  {
      _msgQueue.Enqueue(new Tuple<string, Color>(line, color));

      MethodInvoker gui = delegate
          {
              try
              {
                  // Was getting an overflow so trim out some lines
                  if (uiTrace.Lines.Length > 5000)
                  {
                      uiTrace.Lines = new string[0];
                      uiTrace.SelectionStart = uiTrace.TextLength;
                      Application.DoEvents();
                  }

                  while (_msgQueue.Count != 0)
                  {

                      bool retry;
                      var count = 0;
                      do
                      {
                          try
                          {
                              count++;
                              if (_indent < 0)
                                  _indent = 0;

                              var msg = _msgQueue.Dequeue();
                              var selectionStart = uiTrace.TextLength;
                              uiTrace.AppendText(string.Format("[{0}] {1}{2}", _stopwatch.ElapsedMilliseconds, string.Empty.PadLeft(_indent * 4), msg.Item1));


                              uiTrace.Select(selectionStart, uiTrace.TextLength);
                              uiTrace.SelectionColor = msg.Item2;
                              uiTrace.SelectionStart = uiTrace.TextLength;
                              uiTrace.ScrollToCaret();
                              retry = false;
                          }
                          catch (Exception)
                          {
                              retry = true;
                          }
                      } while (retry && count < 5);
                  }
              }
              catch (Exception)
              {
                  // We don't care about exceptions in here, for now anyway
              }
          };

      if (uiTrace.InvokeRequired && !uiTrace.Disposing && !uiTrace.IsDisposed)
      {
          uiTrace.BeginInvoke(gui);
          return;
      }
      gui();
  }
  catch (Exception)
  {

   //   QIT_Backoffice.Processes.Errors.ErrorHandler.WriteErrorLog(Sources.SourceEnum.External, ex, "Error writing to trace");
  }
}

我真的不知道如何解决这个问题,我认为调用 BeginInvoke() 是需要的。

寻找任何可能的帮助,或者如果有人知道可以更好地处理这个问题的第三方工具,我很高兴看到。

4

2 回答 2

0

以下是我对您的记录器的修改。注意如何使用_processinglock来避免重入和保护_queue。此外,我使用SynchronizationContext而不是Control.BeginInvoke避免对窗口处置状态的任何依赖。TraceView可以TraceView.Create从任何线程创建(使用)和使用,但它的窗口属于parent窗口的线程,这也是它将文本传递到richedit. 可以为此设置一个专用的 STA 线程,但我认为没有必要。

[已编辑]我已经消除了检查中可能存在的竞争条件,_processing并添加CreateOnOwnThread了以防需要记录器 UI 的专用线程。我还决定保留从紧密循环中调用的Application.DoEvents()情况,以保持 UI 响应。Write

用法(压力测试):

private void Form1_Load(object sender, EventArgs ev)
{
    var traceView = TraceView.Create(this);
    for (var i = 0; i < 1000; i++)
    {
        var _i = i;
        Task.Run(() => 
        {
            traceView.Write(String.Format("Line: {0}\n", _i), System.Drawing.Color.Green);
        });
    }
}

执行:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace Logger
{
    public partial class TraceView : Form
    {
        private Form _parent = null;

        private SynchronizationContext _context = SynchronizationContext.Current;
        private int _threadId = Thread.CurrentThread.ManagedThreadId;

        private object _lock = new Object(); // sync lock to protect _queue and _processing
        private Queue<Tuple<string, Color>> _queue = new Queue<Tuple<string, Color>>();
        private volatile bool _processing = false; // reentracy check flag

        public TraceView(Form parent)
        {
            _parent = parent;
            InitializeComponent();
        }

        public static TraceView Create(Form parent)
        {
            TraceView view = null;
            // create it on the parent window's thread
            parent.Invoke(new Action(() => {
                view = new TraceView(parent);
                view.Show(parent);
            }));
            return view;
        }

        private void DequeueMessages()
        {
            // make sure we are on the UI thread
            Debug.Assert(Thread.CurrentThread.ManagedThreadId == _threadId); 

            lock (_lock)
            {
                // prevent re-entracy
                if (_processing)
                    return;
                // mark the beginning of processing
                _processing = true;
            }

            // process pending messages
            for (; ; )
            {
                Tuple<string, Color> msg = null;

                lock (_lock)
                {
                    if (!_queue.Any())
                    {
                        // mark the end of processing
                        _processing = false;
                        return;
                    }
                    msg = _queue.Dequeue();
                }

                if (this.Disposing || this.IsDisposed)
                {
                    // do not just loose messages if the window is disposed
                    Trace.Write(msg.Item1); 
                }
                else
                {
                    var selectionStart = _richTextBox.TextLength;
                    _richTextBox.AppendText(msg.Item1);
                    _richTextBox.Select(selectionStart, _richTextBox.TextLength);
                    _richTextBox.SelectionColor = msg.Item2;
                    _richTextBox.SelectionStart = _richTextBox.TextLength;
                    _richTextBox.ScrollToCaret();
                    _richTextBox.Refresh(); // redraw;
                    // DoEvents is required if logging from a tight loop, 
                    // to keep the UI responsive
                    Application.DoEvents(); 
                }
            }
        }

        public void Write(string line, Color color)
        {
            lock (_lock)
            {
                _queue.Enqueue(new Tuple<string, Color>(line, color));
                // prevent re-entracy
                if (_processing)
                    return; // DequeueMessages is already in progress
            }

            if (Thread.CurrentThread.ManagedThreadId == _threadId)
                DequeueMessages();
            else
                _context.Post((_) => 
                { 
                    DequeueMessages();  
                }, null);
        }

        public static TraceView CreateOnOwnThread()
        {
            TraceView view = null;
            using (var sync = new ManualResetEventSlim())
            {
                // create it on its own thread
                var thread = new Thread(() =>
                {
                    SynchronizationContext.SetSynchronizationContext(new WindowsFormsSynchronizationContext());
                    view = new TraceView(null);
                    view.Show();
                    sync.Set(); // ready Write calls
                    Application.Run(view); // view does Application.ExitThread() when closed
                    return;
                });

                thread.SetApartmentState(ApartmentState.STA);
                thread.Start();
                sync.Wait();
            }
            return view;
        }

    }
}
于 2013-08-28T05:37:41.907 回答
0

与 C# 相比,我对 VB 的 .Net 经验要多得多,但以下代码没有:

  if (uiTrace.InvokeRequired && !uiTrace.Disposing && !uiTrace.IsDisposed)
  {
      uiTrace.BeginInvoke(gui);
      return;
  }
  gui();

导致在语句中gui被调用 ifInvokeRequired等,并在.Ifgui()

不会:

  If (uiTrace.InvokeRequired && !uiTrace.Disposing && !uiTrace.IsDisposed)
  {
      uiTrace.BeginInvoke(gui);
      return;
  }
  Else
      gui();

更合适?

于 2013-08-28T02:24:51.460 回答