исключение StackOverflow

#exception #stack-overflow #dde

#исключение #переполнение стека #dde

Вопрос:

Я использую клиент DDE для подключения и прослушивания цен на фондовом рынке. У этого клиента есть метод обратного вызова, который я реализовал, что делать, когда он получает изменения цен. Проблема в том, что я получаю исключение StackOverflowException (периодически, а не через один и тот же интервал времени). Я нашел кое-что о потоке.BeginCriticalRegion(), но я не уверен, поможет ли это. У меня есть еще несколько часов до открытия рынка, когда я смогу его протестировать. Я был бы более чем благодарен, если бы кто-нибудь мог подсказать мне, как переопределить это исключение.

Заранее спасибо, Александр

IList<SymbolObject> _symbols; //initialized when the app runs for the first time

 void _ddeClient_Advise(object sender, DdeAdviseEventArgs args)
    {
        if (!IsReady)
            return;

        if (string.IsNullOrEmpty(args.Text))
        {
            _logMessages.LogMessagesAdd("advise dde symbol", string.Format("args.Text is empty or NULL for {0}", args.Item), true);
            return;
        }

        try
        {
            string[] argsArray = args.Text.Replace("", "").Replace(''.ToString(), "").Split(' '); // sometimes happens here

            var list = _symbols.Where(s => s.DDESymbol == args.Item).ToList();
            if (list.Count == 0)
                return;

            decimal? val = null;
            try
            {
                var stringParts = StringUtils.CleanProphitXUrl(argsArray[0]).Split('.');
                argsArray = null;

                if (stringParts.Length >= 2)
                    val = decimal.Parse(stringParts[0]   "."   (stringParts[1].Length > 2 ? stringParts[1].Substring(0, 2) : stringParts[1]));
                else
                    val = decimal.Parse(stringParts[0]);

                stringParts = null;
            }
            catch (Exception ex)
            {
                _logMessages.LogMessagesAdd("call Price Alerts application service", ex.Message, true);
                return;
            }

            foreach (var l in list)
            {
                if (_lastPrices[l.DDESymbol] == null)
                    continue;

                if (_lastPrices[l.DDESymbol].ToString() != val.ToString())
                {
                    try
                    {
                        _quotePublishingService.PublishQuote(l.DDESymbolId, l.Symbol, args.Item, val, WebSyncPublisherUrl,
                            PublishingChannel); // a call to wcf service
                    }
                    catch (Exception ex)
                    {
                        _logMessages.LogMessagesAdd("call the service", ex.Message, true); // save to sql db
                        return;
                    }

                    _lastPrices[l.DDESymbol] = val.ToString();
                }
            }
            list = null;
            val = null;
        }
        catch
        {
        }            
    }

 public static string CleanProphitXUrl(string value) // StringUtils.CleanProphitXUrl snippet
    {
        StringBuilder sb = new StringBuilder();
        sb.Append(value.Substring(0, value.LastIndexOf(".")   1));

        try
        {
            value = value.Replace('r'.ToString(), "").Replace('t'.ToString(), "").Replace('n'.ToString(), "");
            for (int i = sb.Length; i < value.Length; i  )
            {
                if (char.IsNumber(value[i]))
                    sb.Append(value[i]);
            }
        }
        catch
        {

        }

        return sb.ToString();
    }
  

Ответ №1:

StackOverflowException Вызвано выполнением множества вызовов методов, обычно возникающих в результате непреднамеренной рекурсии. Основываясь на беглой проверке опубликованного вами кода, я не верю, что это виновник. Проблема, вероятно, кроется где-то в другом месте.