移动平均线交叉策略是一种 momentum 策略,利用双移动平均线的交叉信号来判断趋势方向,产生买入和卖出信号。该策略使用 2 条简单移动平均线和 1 条指数移动平均线,根据它们的交叉情况判定多空,属于中短期交易策略。
该策略使用 3 条移动平均线:
策略以 EMA1, SMA1, SMA2 的大小关系来判断趋势:
进入信号:
退出信号:
该策略提供了多种参数配置,可以选择不同的移动平均线来判断进入和退出。
该策略具有以下优势:
该策略也存在以下风险:
针对 whipsaws 风险,可以适当调整移动平均线周期;对参数敏感性风险,可以优化参数;对滞后性风险,可以结合其他先行指标优化。
该策略可以从以下几个方面进行优化:
移动平均线交叉策略整体较为简单直接,通过快慢均线的交叉判定趋势方向和参与时机。该策略优势是可以捕捉momentum,灵活配置参数,但也存在一定的whipsaw风险、滞后风险等问题。通过引入其他指标进行过滤优化,该策略可以成为一个非常实用的量化交易策略。
/*backtest start: 2023-09-26 00:00:00 end: 2023-10-26 00:00:00 period: 1h basePeriod: 15m exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}] */ // This source code is subject to the terms of the Mozilla Public License 2.0 at https://mozilla.org/MPL/2.0/ // © Decam9 //@version=5 strategy(title = "Moving Average Crossover", shorttitle = "MA Crossover Strategy", overlay=true, initial_capital = 100000,default_qty_type = strategy.percent_of_equity, default_qty_value = 10) //Moving Average Inputs EMA1 = input.int(title="Fast EMA", group = "Moving Averages:", inline = "EMAs", defval=5, minval = 1) isDynamicEMA = input.bool(title = "Dynamic Exponential Moving Average?", defval = true, inline = "EMAs", group = "Moving Averages:", tooltip = "Changes the source of the MA based on trend") SMA1 = input.int(title = "Slow SMA", group = "Moving Averages:", inline = "SMAs", defval = 10, minval = 1) isDynamicSMA = input.bool(title = "Dynamic Simple Moving Average?", defval = false, inline = "SMAs", group = "Moving Averages:", tooltip = "Changes the source of the MA based on trend") SMA2 = input.int(title="Trend Determining SMA", group = "Moving Averages:", inline = "MAs", defval=13, minval = 1) //Moving Averages Trend = ta.sma(close, SMA2) Fast = ta.ema(isDynamicEMA ? (close > Trend ? low : high) : close, EMA1) Slow = ta.sma(isDynamicSMA ? (close > Trend ? low : high) : close, SMA1) //Allowed Entries islong = input.bool(title = "Long", group = "Allowed Entries:", inline = "Entries",defval = true) isshort = input.bool(title = "Short", group = "Allowed Entries:", inline = "Entries", defval= true) //Entry Long Conditions buycond = input.string(title="Buy when", group = "Entry Conditions:", inline = "Conditions",defval="Fast-Slow Crossing", options=["Fast-Slow Crossing", "Fast-Trend Crossing","Slow-Trend Crossing"]) intrendbuy = input.bool(title = "In trend", defval = true, group = "Entry Conditions:", inline = "Conditions", tooltip = "In trend if price is above SMA 2") //Entry Short Conditions sellcond = input.string(title="Sell when", group = "Entry Conditions:", inline = "Conditions2",defval="Fast-Slow Crossing", options=["Fast-Slow Crossing", "Fast-Trend Crossing","Slow-Trend Crossing"]) intrendsell = input.bool(title = "In trend",defval = true, group = "Entry Conditions:", inline = "Conditions2", tooltip = "In trend if price is below SMA 2?") //Exit Long Conditions closebuy = input.string(title="Close long when", group = "Exit Conditions:", defval="Fast-Slow Crossing", options=["Fast-Slow Crossing", "Fast-Trend Crossing","Slow-Trend Crossing"]) //Exit Short Conditions closeshort = input.string(title="Close short when", group = "Exit Conditions:", defval="Fast-Slow Crossing", options=["Fast-Slow Crossing", "Fast-Trend Crossing","Slow-Trend Crossing"]) //Filters filterlong =input.bool(title = "Long Entries", inline = 'linefilt', group = 'Apply Filters to', defval = true) filtershort =input.bool(title = "Short Entries", inline = 'linefilt', group = 'Apply Filters to', defval = true) filterend =input.bool(title = "Exits", inline = 'linefilt', group = 'Apply Filters to', defval = true) usevol =input.bool(title = "", inline = 'linefiltvol', group = 'Relative Volume Filter:', defval = false) rvol = input.int(title = "Volume >", inline = 'linefiltvol', group = 'Relative Volume Filter:', defval = 1) len_vol = input.int(title = "Avg. Volume Over Period", inline = 'linefiltvol', group = 'Relative Volume Filter:', defval = 30, minval = 1, tooltip="The current volume must be greater than N times the M-period average volume.") useatr =input.bool(title = "", inline = 'linefiltatr', group = 'Volatility Filter:', defval = false) len_atr1 = input.int(title = "ATR", inline = 'linefiltatr', group = 'Volatility Filter:', defval = 5, minval = 1) len_atr2 = input.int(title = "> ATR", inline = 'linefiltatr', group = 'Volatility Filter:', defval = 30, minval = 1, tooltip="The N-period ATR must be greater than the M-period ATR.") usersi =input.bool(title = "", inline = 'linersi', group = 'Overbought/Oversold Filter:', defval = false) rsitrhs1 = input.int(title = "", inline = 'linersi', group = 'Overbought/Oversold Filter:', defval = 0, minval=0, maxval=100) rsitrhs2 = input.int(title = "< RSI (14) <", inline = 'linersi', group = 'Overbought/Oversold Filter:', defval = 100, minval=0, maxval=100, tooltip="RSI(14) must be in the range between N and M.") issl = input.bool(title = "SL", inline = 'linesl1', group = 'Stop Loss / Take Profit:', defval = false) slpercent = input.float(title = ", %", inline = 'linesl1', group = 'Stop Loss / Take Profit:', defval = 10, minval=0.0) istrailing = input.bool(title = "Trailing", inline = 'linesl1', group = 'Stop Loss / Take Profit:', defval = false) istp = input.bool(title = "TP", inline = 'linetp1', group = 'Stop Loss / Take Profit:', defval = false) tppercent = input.float(title = ", %", inline = 'linetp1', group = 'Stop Loss / Take Profit:', defval = 20) //Conditions for Crossing fscrossup = ta.crossover(Fast,Slow) fscrossdw = ta.crossunder(Fast,Slow) ftcrossup = ta.crossover(Fast,Trend) ftcrossdw = ta.crossunder(Fast,Trend) stcrossup = ta.crossover(Slow,Trend) stcrossdw = ta.crossunder(Slow,Trend) //Defining in trend uptrend = Fast >= Slow and Slow >= Trend downtrend = Fast <= Slow and Slow <= Trend justCrossed = ta.cross(Fast,Slow) or ta.cross(Slow,Trend) //Entry Signals crosslong = if intrendbuy (buycond =="Fast-Slow Crossing" and uptrend ? fscrossup:(buycond =="Fast-Trend Crossing" and uptrend ? ftcrossup:(buycond == "Slow-Trend Crossing" and uptrend ? stcrossup : na))) else (buycond =="Fast-Slow Crossing"?fscrossup:(buycond=="Fast-Trend Crossing"?ftcrossup:stcrossup)) crossshort = if intrendsell (sellcond =="Fast-Slow Crossing" and downtrend ? fscrossdw:(sellcond =="Fast-Trend Crossing" and downtrend ? ftcrossdw:(sellcond == "Slow-Trend Crossing" and downtrend ? stcrossdw : na))) else (sellcond =="Fast-Slow Crossing"?fscrossdw:(buycond=="Fast-Trend Crossing"?ftcrossdw:stcrossdw)) crossexitlong = (closebuy =="Fast-Slow Crossing"?fscrossdw:(closebuy=="Fast-Trend Crossing"?ftcrossdw:stcrossdw)) crossexitshort = (closeshort =="Fast-Slow Crossing"?fscrossup:(closeshort=="Fast-Trend Crossing"?ftcrossup:stcrossup)) // Filters rsifilter = usersi?(ta.rsi(close,14) > rsitrhs1 and ta.rsi(close,14) < rsitrhs2):true volatilityfilter = useatr?(ta.atr(len_atr1) > ta.atr(len_atr2)):true volumefilter = usevol?(volume > rvol*ta.sma(volume,len_vol)):true totalfilter = volatilityfilter and volumefilter and rsifilter //Filtered signals golong = crosslong and islong and (filterlong?totalfilter:true) goshort = crossshort and isshort and (filtershort?totalfilter:true) endlong = crossexitlong and (filterend?totalfilter:true) endshort = crossexitshort and (filterend?totalfilter:true) // Entry price and TP startprice = ta.valuewhen(condition=golong or goshort, source=close, occurrence=0) pm = golong?1:goshort?-1:1/math.sign(strategy.position_size) takeprofit = startprice*(1+pm*tppercent*0.01) // fixed stop loss stoploss = startprice * (1-pm*slpercent*0.01) // trailing stop loss if istrailing and strategy.position_size>0 stoploss := math.max(close*(1 - slpercent*0.01),stoploss[1]) else if istrailing and strategy.position_size<0 stoploss := math.min(close*(1 + slpercent*0.01),stoploss[1]) if golong and islong strategy.entry("long", strategy.long ) if goshort and isshort strategy.entry("short", strategy.short) if endlong strategy.close("long") if endshort strategy.close("short") // Exit via SL or TP strategy.exit(id="sl/tp long", from_entry="long", stop=issl?stoploss:na, limit=istp?takeprofit:na) strategy.exit(id="sl/tp short",from_entry="short",stop=issl?stoploss:na, limit=istp?takeprofit:na)