Exception при попытке расчитать ATR

Exception при попытке расчитать ATR
Atom
11/7/2011


Привет,

осваиваю Stocksharp, очень нравится :)

Нужно использовать индикатор ATR, нигде примеров не нашел, только на кодеплексе в тест кейсе.

Соответственно, пытаюсь его рассчитать:
Code

            var atr = new AverageTrueRange { Length = 15 };
            ...
         
            TimeFrameCandle cdl = _candleManager.GetTimeFrameCandle(_candleToken, 1);

            try {
                atr.Process((CandleIndicatorValue)cdl);
            } catch (Exception ex) {
                Error(ex.Message + " stacktrace: " + ex.StackTrace);
            }


В результате на строке atr.Process происходит следующее исключение:
Code

Invalid cast from 'System.Decimal' to 'StockSharp.Algo.Candles.Candle'.
 stacktrace:    at System.Convert.DefaultToType(IConvertible value, Type targetType, IFormatProvider provider)
   at System.Decimal.System.IConvertible.ToType(Type type, IFormatProvider provider)
   at System.Convert.ChangeType(Object value, Type conversionType, IFormatProvider provider)
   at Ecng.Common.Converter.To(Object value, Type destinationType)
   at Ecng.Common.Converter.To[T](Object value)
   at StockSharp.Algo.Indicators.CandleIndicatorValue.SetValue[T](T value)
   at StockSharp.Algo.Indicators.BaseIndicator`1.Process(IIndicatorValue input)
   at StockSharp.Algo.Indicators.Oscillator.AverageTrueRange.OnProcess(IIndicatorValue input)
   at StockSharp.Algo.Indicators.BaseIndicator`1.Process(IIndicatorValue input)
   at test.stocksharp.IndicatorTest.newCandle()


Версия стокшарпа: 4.0.3.

Подскажите куда копать, как побороть? В тесте вроде тоже производится просто каст свечи к CandleIndicatorValue

Tags:


Thanks:


1 2  >
esper

Avatar
Date: 11/7/2011
Reply


Тест для ATR валится с такой же ошибкой:
Quote:
Test method IndicatorsTest.Tests.AtrTest.Atr threw exception:
System.InvalidCastException: Invalid cast from 'System.Decimal' to 'StockSharp.Algo.Candles.Candle'.

Иными словами, руки до этого индикатора пока не дошли[smile] Есть желание помочь?
Thanks:

Mikhail Sukhov

Avatar
Date: 11/9/2011
Reply


kenota Go to

Соответственно, пытаюсь его рассчитать:
Code

            var atr = new AverageTrueRange { Length = 15 };
            ...
         
            TimeFrameCandle cdl = _candleManager.GetTimeFrameCandle(_candleToken, 1);

            try {
                atr.Process((CandleIndicatorValue)cdl);
            } catch (Exception ex) {
                Error(ex.Message + " stacktrace: " + ex.StackTrace);
            }



Используйте IndicatorManager. Индикаторы как раз под него заточены, а не для того, чтобы самому свечки из CandleToken в Indicator перекладывать.
Thanks:

pehas

Avatar
Date: 11/9/2011
Reply


Mikhail Sukhov Go to

Используйте IndicatorManager. Индикаторы как раз под него заточены, а не для того, чтобы самому свечки из CandleToken в Indicator перекладывать.


Не могли бы вы привести пример? Пробую так
Code

            var candleToken = this._cm.RegisterTimeFrameCandles(security, this._timeFrame);
            var ATRToken = indicatorManager.RegisterIndicator(new AverageTrueRange{Length = 2}, new CandleTokenIndicatorSource(candleToken));


Значение "((AverageTrueRange)ATRToken.Indicator).isFormed" всегда false
Как правильно инициализировать?
Thanks:

kenota

Avatar
Date: 11/10/2011
Reply


Подтверждаю тоже самое что говорит pehas: обновил стокшарп до последней версии. Создаю indicatorManager однако индикатор никогда не строится. isFormed всегда false + IndicatorToken.Indicator.OnChanged ни разу не вызывается.

Попробую взять проект с кодекплекса и посмотреть дебагом что там в индикаторе творится :)
Thanks:

kenota

Avatar
Date: 11/10/2011
Reply


Значит так, я нашел две проблемы и сделал хотфиксы, однако мне нужен ревью от кого нибудь кто больше понимает в проекте.

Проблема №1. Если использовать индикатор AverageTrueRange через IndicatorManager то он никогда не расчитывается. Проблема в файле AverageTrueRange.cs строчка 92:

Code

/// <summary>
/// Возможно ли обработать входное значение.
/// </summary>
/// <param name="input">Входное значение.</param>
/// <returns>True, если возможно, иначе, false.</returns>
public override bool CanProcess(IIndicatorValue input)
{
            return TrueRange.CanProcess(input) && MovingAverage.CanProcess(input);
}


Тут проверяется может ли индикатор обработать входящее значение. ATR построили как комбинацию TrueRange (который принимает параметр Candle а результатом вычисления является double) и SMA (вход double и выход double). Вообщем это условие будет всегда ложно, т.к. в функцию передается свечка а MA ее не может обработать. Поэтому просто меняем ее на:
Code

return TrueRange.CanProcess(input);


Проблема №2.
После этого изменения у нас будет опять выбрасываться старый эксепшн, из-за которого я открыл этот топик. Ищем откуда растут ноги, смотрим на файл BaseIndicator.cs в районе строки 129 функция Process:
Code

public virtual IIndicatorValue Process(IIndicatorValue input)	
{
	var value = OnProcess(input);

	this.LastValue = value;

	var result = value as IIndicatorValue ?? input.SetValue(value);

	if (IsFormed && !result.IsEmpty)
		RaiseChangedEvent(input, result);

	return result;
}


Экспешн происходит на строчке
Code
var result = value as IIndicatorValue ?? input.SetValue(value);

Пытаемся понять что тут происходит. Ага, получается так: мы вычисляем значение индикатор через своего потомка (нас ведь создадут наследовав от BaseIndicator), но вот незадача: что бы быть type safe нам нужно вернуть значение IIndicatorValue, а что может вернуть наша текущая реализация мы не знаем.
Поэтому мы смотрим: если то что вернул process является IIndicatorValue то, возвращаем его, иначе, мы считаем что значение должно быть того же типа что было входящее!

Вот тут то и приходит пушной зверек сверкая глазками. Дело в том что TrueRange принимает на вход Candle а возвращает Double. Здесь и происходит класс каст эксепшн.

К сожалению, я пока еще не знаю всю структуру проекта, поэтому не придумал ничего лучшего как сделать такой хотфикс, который поидее ничего не должен сломать, но починить этот эксепшн. Заменяем строку 135 на:
Code

var result = value is decimal ? new DecimalIndicatorValue(value.To<decimal>()) :  value as IIndicatorValue ?? input.SetValue(value);


После этого atr начинает работать :)

Но я думаю это стоит починить по-правильному.
Thanks:

danl

Avatar
Date: 11/11/2011
Reply


По поводу ATR, он работает без фиксов, если его инициализировать вот так:
Code
IndicatorManager.RegisterIndicator(new AverageTrueRange(), new CandleTokenIndicatorSource(candleToken, BaseCandleIndicatorSource.ByClose));


То что там используется BaseCandleIndicatorSource.ByClose реального значения не имеет, так как исходный индикатор будет получать свечу целиком, но источник будет передавать в индикатор класс CandlePartIndicatorValue, который поддерживает тип decimal.

Наверное это не очень красиво, но работает. По поводу правильного решения, может следует сделать реализацию IIndicatorValue подобную той, которая в тесте, что бы она могла содержать любые значения и обучить текущие реализации содавать такие классы? Тогда выходные значения полностью развяжутся с входными, что, как мне кажется, логично. На эту тему нужно еще подумать.
Thanks: kenota

kenota

Avatar
Date: 11/11/2011
Reply


Спасибо, буду знать что можно так инициализировать.
Thanks:

russ

Avatar
Date: 11/13/2011
Reply


Аналогичная проблема, борюсь с фракталами
Code
[left]try
                    {
                        _strategy.FractalsValues.Process((CandleIndicatorValue)candle);
                    }
                    catch (System.Exception ex)
                    {
                        MessageBox.Show(ex.Message);
                    }[/left]



Выдает ошибку

Недопустимое приведение "System.Int32" к "StockSharp.Algo.Candles.Candle".

Что не так делаю?

Посмотрел дебагом индиктор фракталов, падает в
Code
public override ComplexIndicatorValue OnProcess(IIndicatorValue input)

в
Code
return base.OnProcess(new ShiftedIndicatorValue(0, input.SetValue(0)));
Thanks:

Alexander

Avatar
Date: 11/13/2011
Reply


russ Go to
Аналогичная проблема, борюсь с фракталами
Code
[left]try
                    {
                        _strategy.FractalsValues.Process((CandleIndicatorValue)candle);
                    }
                    catch (System.Exception ex)
                    {
                        MessageBox.Show(ex.Message);
                    }[/left]



Выдает ошибку

Недопустимое приведение "System.Int32" к "StockSharp.Algo.Candles.Candle".

Что не так делаю?

Посмотрел дебагом индиктор фракталов, падает в
Code
public override ComplexIndicatorValue OnProcess(IIndicatorValue input)

в
Code
return base.OnProcess(new ShiftedIndicatorValue(0, input.SetValue(0)));



Ну видимо candle у вас Int32, а вы пытаетесь не к тому типу привести.
Thanks:

russ

Avatar
Date: 11/13/2011
Reply


Alexander Mukhanchikov Go to
russ Go to
Аналогичная проблема, борюсь с фракталами
Code
[left]try
                    {
                        _strategy.FractalsValues.Process((CandleIndicatorValue)candle);
                    }
                    catch (System.Exception ex)
                    {
                        MessageBox.Show(ex.Message);
                    }[/left]



Выдает ошибку

Недопустимое приведение "System.Int32" к "StockSharp.Algo.Candles.Candle".

Что не так делаю?

Посмотрел дебагом индиктор фракталов, падает в
Code
public override ComplexIndicatorValue OnProcess(IIndicatorValue input)

в
Code
return base.OnProcess(new ShiftedIndicatorValue(0, input.SetValue(0)));



Ну видимо candle у вас Int32, а вы пытаетесь не к тому типу привести.


Да, ошибку я понял, но я пользуюсь индикаторами, которые идут в S#, получается они могут не работать?
Thanks:
1 2  >

Attach files by dragging & dropping, , or pasting from the clipboard.

loading
clippy