Кеширование индикаторов

Для кеширования результатов индикаторов существует методы IContext.GetData(...), которые могут работать с тремя типами списков: double, int, bool.

В прошлой статье мы рассчитывали индикатор SMA таким образом:

var smaFast = Series.SMA(sec.ClosePrices, PeriodFast)

При проведении оптимизации расчет SMA с одинаковым периодом будет повторяться много раз. Чтобы убрать лишние расчеты и уменьшить время оптимизации существует кеширование. Предыдущий пример можно переписать так:

var smaFast = ctx.GetData("SMA", new[] { PeriodFast.ToString() }, () => Series.SMA(sec.ClosePrices, PeriodFast));

Метод GetData работает с кешем данных, кеш представляет из себя таблицу КЛЮЧ - ЗНАЧЕНИЕ, где:

  • КЛЮЧ - это строка, она всегда уникальная в таблице, не может существовать несколько одинаковых ключей.

  • ЗНАЧЕНИЕ - это список значений (double, int, bool).

Метод GetData принимает три параметра:

  1. Первый параметр - префикс ключа. В данном примере это "SMA".

  2. Второй параметр - это массив строк из которых строится окончание ключа. В данном примере передается значение периода в виде строки (PeriodFast.ToString()), например оно равно 100. Тогда полное название ключа будет таким "SMA:100". Если бы передавалось несколько значений, например 100, 20, 40, то ключ был бы таким "SMA:100:20:40".

  3. Третий параметр - это делегат, который вызывается если нету в кеше данных по заданному ключу. То есть это и есть расчет значений индикатора. Делегат должен возвращать список значений double, int или bool.

Как это работает.

Изначально кеш данных пустой. Когда мы запрашиваем расчет SMA с периодом 100, то формируется ключ "SMA:100". Этот ключ ищется в кеш таблице, но его там нету, так как таблица пустая. Тогда вызывается делегат, который рассчитывает значение SMA. Результат записывается в кеш таблицу с ключом "SMA:100" и метод GetData возвращает этот результат. Если мы снова запрашиваем расчет SMA с периодом 100, то в кеш таблице будет искаться ключ "SMA:100". Так как он есть, то метод GetData сразу вернет значение по этому ключу, при этом не будет вызываться делегат.

Кеш значений сбрасывается на каждом пересчете скрипта. В режиме оптимизации кеш не сбрасывается.

Пример скрипта с кеширированием двух скользящих средних:

using TSLab.Script; using TSLab.Script.Handlers; using TSLab.Script.Helpers; using TSLab.Script.Optimization; namespace MyLib { public class SimpleCache : IExternalScript { // Настраиваемые параметры public OptimProperty PeriodFast = new OptimProperty(100, 10, 200, 10); public OptimProperty PeriodSlow = new OptimProperty(500, 200, 2000, 100); public void Execute(IContext ctx, ISecurity sec) { // Расчет быстрой SMA var smaFast = ctx.GetData("SMA", new[] { PeriodFast.ToString() }, () => Series.SMA(sec.ClosePrices, PeriodFast)); // Расчет медленной SMA var smaSlow = ctx.GetData("SMA", new[] { PeriodSlow.ToString() }, () => Series.SMA(sec.ClosePrices, PeriodSlow)); // Торговый цикл for (int i = 0; i < ctx.BarsCount; i++) { var posLong = sec.Positions.GetLastActiveForSignal("LE", i); if (posLong == null) { if (smaFast[i] > smaSlow[i] && smaFast[i - 1] <= smaSlow[i - 1]) sec.Positions.BuyAtMarket(i + 1, 1, "LE"); } else { if (smaFast[i] < smaSlow[i] && smaFast[i - 1] >= smaSlow[i - 1]) posLong.CloseAtMarket(i + 1, "LX"); } } // Если скрипт в режиме оптимизации, то не нужно строить графики if (ctx.IsOptimization) { return; } // Построение графиков ctx.First.AddList(string.Format("SMA fast ({0})", PeriodFast), smaFast, ListStyles.LINE, ScriptColors.Green, LineStyles.SOLID, PaneSides.RIGHT); ctx.First.AddList(string.Format("SMA slow ({0})", PeriodSlow), smaSlow, ListStyles.LINE, ScriptColors.Red, LineStyles.SOLID, PaneSides.RIGHT); } } }


Рассмотрим пример со сложным кешированием.

К примеру, нам надо построить границу верхнего канала по максимальным ценам, а потом сгладить эту границу по SMA. Получается что расчет канала зависит от периода канала, а расчет SMA зависит от периода SMA и от периода канала. Поэтому при расчете SMA надо передавать два параметра, период SMA и период канала. Полный пример:

var highest = ctx.GetData("Highest", new[] { PeriodHighest.ToString() }, () => Series.Highest(sec.HighPrices, PeriodHighest)); var highestSMA = ctx.GetData("HighestSMA", new[] { PeriodHighest.ToString(), PeriodHighestSMA.ToString() }, () => Series.SMA(highest, PeriodHighestSMA));

Помните, что если неверно задать параметры кеширования, то результаты оптимизации будут неверные.