超级趋势日线反转策略(Super Trend Daily Reversal Strategy)是一个利用超级趋势指标判断市场趋势,结合价格突破和平均真实波动范围计算止损,以及采用价格变化率指标过滤超级趋势信号的量化交易策略。该策略适用于日线及更高时间周期,可用于数字货币和股票等市场。
该策略的核心指标是超级趋势指标(Super Trend Indicator)。超级趋势指标基于平均真实波动范围(ATR),可以更加清晰地判断市场趋势方向。当价格突破超级趋势上轨时为看跌信号,突破下轨时为看涨信号。
该策略配套使用价格变化率指标(ROC)对超级趋势指标进行过滤,避免无效信号。当价格波动率较大时才参与超级趋势信号,否则不参与。
在止损方面,该策略提供两种止损方式:固定止损比例和基于ATR的自动缩进止损。固定止损简单直接,ATR止损可以根据市场波动性调整止损范围。
入场条件是超级趋势指标反转且价格变化率指标通过过滤器。出场条件是超级趋势再次反转或突破止损线。该策略遵循趋势跟踪原则,每个方向只允许一个头寸。
该策略最大的优势在于利用超级趋势指标判断趋势方向的清晰度和稳定性较高,与普通移动平均线相比noise较小。另外,策略加入价格变化率指标有效过滤了部分假信号。
ATR自适应止损机制也使得该策略可以适应更广泛的市场环境。在波动加剧时止损会自动放宽,最大程度锁住盈利。
从测试结果看,该策略在牛市中表现优异。在数量级较大的长线趋势中胜率很高,连续盈利周期长。
该策略面临的主要风险在于趋势反转判断失误,从而可能错过反转信号或产生不必要的反转信号。这种情况通常发生在价格在关键支撑或阻力区域附近反复横盘整理时。
此外,止损设置过于宽松也会导致亏损扩大。ATR止损根据市场波动性进行调整,所以在市场突发事件时止损可能会拉的较宽。
针对上述风险,可适当缩短ATR计算周期或调整ATR止损的倍数系数。也可以加入附加指标确定关键支撑阻力区域,避免那些区域发出误导信号。
该策略可从以下几个方面进行优化:
调整超级趋势指标的参数,优化ATR周期和ATR倍数,使超级趋势线更加平滑。
调整价格变化率指标的参数,优化周期和变化率阈值,减少假信号。
尝试不同的止损机制,如跟踪止损,或优化固定止损的止损幅度。
增加附加判断指标,确定关键支撑阻力,避免趋势反转判断失误。
测试不同品种的参数设置和效果,寻找最优参数组合。
进行回测优化,找到最佳的参数设置。
超级趋势日线反转策略整体来说是一个较为稳定可靠的趋势跟踪策略。它结合超级趋势指标和价格变化率指标进行过滤,可以有效识别中长线趋势的方向。ATR自适应止损机制也使其可以适应大部分市场环境。通过进一步优化参数设置和增加判断指标,该策略的稳定性和盈利能力还可以得到提高。
/*backtest start: 2024-01-22 00:00:00 end: 2024-02-21 00:00:00 period: 1h basePeriod: 15m exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}] */ //@version=4 strategy("Super Trend Daily BF 🚀", overlay=true, precision=2, initial_capital=10000, default_qty_type=strategy.percent_of_equity, default_qty_value=100, commission_type=strategy.commission.percent, commission_value=0.075) /////////////// Time Frame /////////////// _1 = input(false, "════════ Test Period ═══════") testStartYear = input(2017, "Backtest Start Year") testStartMonth = input(1, "Backtest Start Month") testStartDay = input(1, "Backtest Start Day") testPeriodStart = timestamp(testStartYear,testStartMonth,testStartDay, 0, 0) testStopYear = input(2019, "Backtest Stop Year") testStopMonth = input(12, "Backtest Stop Month") testStopDay = input(31, "Backtest Stop Day") testPeriodStop = timestamp(testStopYear,testStopMonth,testStopDay, 0, 0) testPeriod() => true ///////////// Super Trend ///////////// _2 = input(false, "══════ Super Trend ══════") length = input(title="ATR Period", type=input.integer, defval=3) mult = input(title="ATR Multiplier", type=input.float, step=0.1, defval=1.3) atr = mult * atr(length) longStop = hl2 - atr longStopPrev = nz(longStop[1], longStop) longStop := close[1] > longStopPrev ? max(longStop, longStopPrev) : longStop shortStop = hl2 + atr shortStopPrev = nz(shortStop[1], shortStop) shortStop := close[1] < shortStopPrev ? min(shortStop, shortStopPrev) : shortStop dir = 1 dir := nz(dir[1], dir) dir := dir == -1 and close > shortStopPrev ? 1 : dir == 1 and close < longStopPrev ? -1 : dir ///////////// Rate Of Change ///////////// _3 = input(false, "══════ Rate of Change ══════") source = close roclength = input(30, "ROC Length", minval=1) pcntChange = input(6, "ROC % Change", minval=1) roc = 100 * (source - source[roclength]) / source[roclength] emaroc = ema(roc, roclength / 2) isMoving() => emaroc > (pcntChange / 2) or emaroc < (0 - (pcntChange / 2)) /////////////// Strategy /////////////// long = dir == 1 and dir[1] == -1 and isMoving() short = dir == -1 and dir[1] == 1 and isMoving() last_long = 0.0 last_short = 0.0 last_long := long ? time : nz(last_long[1]) last_short := short ? time : nz(last_short[1]) long_signal = crossover(last_long, last_short) short_signal = crossover(last_short, last_long) last_open_long_signal = 0.0 last_open_short_signal = 0.0 last_open_long_signal := long_signal ? open : nz(last_open_long_signal[1]) last_open_short_signal := short_signal ? open : nz(last_open_short_signal[1]) last_long_signal = 0.0 last_short_signal = 0.0 last_long_signal := long_signal ? time : nz(last_long_signal[1]) last_short_signal := short_signal ? time : nz(last_short_signal[1]) in_long_signal = last_long_signal > last_short_signal in_short_signal = last_short_signal > last_long_signal last_high = 0.0 last_low = 0.0 last_high := not in_long_signal ? na : in_long_signal and (na(last_high[1]) or high > nz(last_high[1])) ? high : nz(last_high[1]) last_low := not in_short_signal ? na : in_short_signal and (na(last_low[1]) or low < nz(last_low[1])) ? low : nz(last_low[1]) since_longEntry = barssince(last_open_long_signal != last_open_long_signal[1]) since_shortEntry = barssince(last_open_short_signal != last_open_short_signal[1]) /////////////// Dynamic ATR Stop Losses /////////////// _4 = input(false, "════════ Stop Loss ═══════") SL_type = input("Fixed", options=["Fixed", "ATR Derived"], title="Stop Loss Type") sl_inp = input(6.0, title='Fixed Stop Loss %') / 100 atrLkb = input(20, minval=1, title='ATR Stop Period') atrMult = input(1.5, step=0.25, title='ATR Stop Multiplier') atr1 = atr(atrLkb) longStop1 = 0.0 longStop1 := short_signal ? na : long_signal ? close - (atr1 * atrMult) : longStop1[1] shortStop1 = 0.0 shortStop1 := long_signal ? na : short_signal ? close + (atr1 * atrMult) : shortStop1[1] slLong = in_long_signal ? strategy.position_avg_price * (1 - sl_inp) : na slShort = strategy.position_avg_price * (1 + sl_inp) long_sl = in_long_signal ? slLong : na short_sl = in_short_signal ? slShort : na /////////////// Execution /////////////// if testPeriod() strategy.entry("L", strategy.long, when=long) strategy.entry("S", strategy.short, when=short) strategy.exit("L SL", "L", stop = SL_type == "Fixed" ? long_sl : longStop1, when=since_longEntry > 0) strategy.exit("S SL", "S", stop = SL_type == "Fixed" ? short_sl : shortStop1, when=since_shortEntry > 0) /////////////// Plotting /////////////// bgcolor(long_signal ? color.lime : short_signal ? color.red : na, transp=30) bgcolor(isMoving() ? dir == 1 ? color.lime : color.red : color.white , transp=80)