Количественная стратегия торговли на основе фильтра Восса и индикатора тренда

Автор:Чао Чжан, Дата: 2023-09-19 16:59:10
Тэги:

Обзор

Эта стратегия объединяет предсказательный фильтр Восса и индикатор мгновенной линии тренда Элерса для выявления циклических поворотных точек на рынке для количественной торговли.

Логика стратегии

Прогнозный фильтр Voss

Фильтр Восса взят из статьи Джона Ф. Элерса "Взгляд в будущее".

_filt = 0.5 * _s3 * _x1 + _f1 * _s2 * _filt[1] - _s1 * _filt[2] 
_voss = _x2 * _filt - _sumC

где _x1 - это разница в цене 1-го порядка; _x2 - это коэффициент сглаживания; _s1, _s2, _s3 - параметры фильтра; _f1 - параметр цикла; _filt - выход фильтра; _voss - конечный выход.

Фильтр можно рассматривать как сглаженный фильтр, который подчеркивает текущую и прошедшую информацию о цикле, чтобы генерировать ранние сигналы.

Мгновенный индикатор трендовой линии

Индикатор линии тренда рассчитывается как:

_it = (_a-((_a*_a)/4.0))*_src+0.5*_a*_a*_src[1]-(_a-0.75*_a*_a)*_src[2]+2*(1-_a)*nz(_it[1])+-(1-_a)*(1-_a)*nz(_it[2]) 

Он показывает трендовую линию в режиме реального времени, которая точно соответствует движению цены, точно определяя направление и силу тренда.

Логика стратегии

Сигнал покупки генерируется, когда Voss пересекает результат фильтра.

Сигнал продажи генерируется, когда Voss пересекает результат фильтра.

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

Преимущества

  • Фильтр Восса обеспечивает ранние предсказательные сигналы для улавливания циклических поворотов
  • Индикатор трендовой линии точно определяет направление тренда, избегая неправильных сигналов
  • Параметры можно оптимизировать для различных циклов и рыночных условий
  • Для контроля риска могут быть добавлены остановки

Риски и их смягчение

  • Стратегия опирается на ранние сигналы, потенциально пропуская некоторые тенденции
  • Сигналы противоположной тенденции могут возникать в сильных тенденциях, вызывая убытки

Риски могут быть уменьшены:

  • Параметр периода оптимизации для циклов прибора
  • Уменьшение пропускной способности фильтра для уменьшения ложных сигналов
  • Добавление фильтров тенденций для предотвращения сигналов против сильных тенденций
  • Использование остановок для контроля потерь на одну сделку

Возможности для расширения

Стратегия может быть улучшена путем:

  • Пробовать различные источники цен, такие как закрытие, скользящие средние и т.д.
  • Период регулирования фильтра для конкретных циклов прибора
  • Оптимизация параметров индикатора тренда для лучшего обнаружения тренда
  • Испытание различных индикаторов тенденции для лучших комбинаций
  • Добавление остановок, остановок для лучшего контроля риска
  • Оптимизация параметров для лучших наборов параметров

Заключение

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


/*backtest
start: 2023-08-19 00:00:00
end: 2023-09-18 00:00:00
period: 1h
basePeriod: 15m
exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}]
*/

// A Peek Into the Future
// John F. Ehlers
// TASC Aug 2019

// Created by e2e4mfck for tradingview.com
// Modified by © Bitduke

//@version=4
//strategy("Voss Strategy (Filter + IT)", overlay=false, calc_on_every_tick=false,pyramiding=0, default_qty_type=strategy.cash,default_qty_value=1000, currency=currency.USD, initial_capital=1000,commission_type=strategy.commission.percent, commission_value=0.075)

// voss filter

source = input(close, type = input.source)
period = input(20, type = input.integer)
predict = input(4, type = input.integer)
bandwidth = input(0.25, type = input.float)

// it trendline

src = input(hl2, title="Source IT")
a = input(0.07, title="Alpha", step=0.01) 
fr = input(false, title="Fill Trend Region")
ebc = input(false, title="Enable barcolors")
hr = input(false, title="Hide Ribbon")


voss_filter (_period, _predict, _bandwidth, _source) =>
	float _filt = 0, float _sumC = 0, float _voss = 0
	_PI		= 2 * asin(1)
	_order	= 3 * _predict
	_f1		= cos(2 * _PI / _period)
	_g1		= cos(_bandwidth * 2 * _PI / _period)
	_s1		= 1 / _g1 - sqrt(1 / (_g1 * _g1) - 1)
	_s2		= 1 + _s1
	_s3		= 1 - _s1
	_x1		= _source - _source[2]
	_x2		= (3 + _order) / 2

	for _i = 0 to (_order - 1)
		_sumC := _sumC + ((_i + 1) / _order) * _voss[_order - _i]

	if bar_index <= _order
		_filt := 0		
		_voss := 0		
	else			
		_filt := 0.5 * _s3 * _x1 + _f1 * _s2 * _filt[1] - _s1 * _filt[2]
		_voss := _x2 * _filt - _sumC

	[_voss, _filt]


[Voss, Filt] = voss_filter(period, predict, bandwidth, source)


instantaneous_trendline (_src, _a, _freq, _ebc, _hr) =>
    _it = 0.0
    _it := (_a-((_a*_a)/4.0))*_src+0.5*_a*_a*_src[1]-(_a-0.75*_a*_a)*_src[2]+2*(1-_a )*nz(_it[1], ((_src+2*_src[1]+_src[2])/4.0))-(1-_a)*(1-_a)*nz(_it[2], ((_src+2*_src[1]+_src[2])/4.0))
    _lag = 2.0*_it-nz(_it[2])
    
    [_it, _lag]

[it, lag] = instantaneous_trendline(src, a, fr, ebc, hr)

// - - - - -  - - - - - //

plot(Filt, title = "Filter", style = plot.style_line, color = color.red, linewidth = 2)
plot(Voss, title = "Voss", style = plot.style_line, color = color.blue,	linewidth = 2)
hline(0.0, title = "Zero", linestyle = hline.style_dashed, color = color.black,	linewidth = 1)
plot(hr? na:it, title="IT Trend", color= fr? color.gray : color.red, linewidth=1)
plot(hr? na:lag, title="IT Trigger", color=fr? color.gray : color.blue, linewidth=1)


// Strategy Logic
longCondition =  lag < it  and crossover(Voss,Filt) 
shortCondition = it > lag and crossover(Filt,Voss) 

strategy.entry("Voss_Short", strategy.short, when=shortCondition)
strategy.entry("Voss_Long", strategy.long, when=longCondition)


// === Backtesting Dates === thanks to Trost

testPeriodSwitch = input(true, "Custom Backtesting Dates")
testStartYear = input(2019, "Backtest Start Year")
testStartMonth = input(1, "Backtest Start Month")
testStartDay = input(1, "Backtest Start Day")
testStartHour = input(0, "Backtest Start Hour")
testPeriodStart = timestamp(testStartYear, testStartMonth, testStartDay, testStartHour, 0)
testStopYear = input(2020, "Backtest Stop Year")
testStopMonth = input(2, "Backtest Stop Month")
testStopDay = input(29, "Backtest Stop Day")
testStopHour = input(0, "Backtest Stop Hour")
testPeriodStop = timestamp(testStopYear, testStopMonth, testStopDay, testStopHour, 0)
testPeriod() =>
    time >= testPeriodStart and time <= testPeriodStop ? true : false
testPeriod_1 = testPeriod()
isPeriod = true
// === /END

if not isPeriod
    strategy.cancel_all()
    strategy.close_all()


Больше