该策略结合了EMA和RSI两个指标来识别趋势方向并进行进出场。当价格在EMA之上且RSI低于买点时看涨;当价格在EMA之下且RSI高于卖点时看跌。同时结合最后两个K线的收盘价大小关系判定趋势方向,实现趋势交易。
计算200日EMA,作为判断趋势的均线指标。EMA快速响应价格变化,能够有效判断趋势方向。
计算14日RSI,判断是否超买超卖。RSI低于50视为超卖,高于50视为超买。同时结合RSI的向上向下趋势判断买卖时机。
比较最后两个K线的收盘价大小关系,判断趋势方向。最后两个收盘价递增认为处于上涨趋势,递减认为处于下跌趋势。
当处于上涨趋势,价格高于200日EMA,且RSI低于50且上涨时,发出买入信号。
当处于下跌趋势,价格低于200日EMA,且RSI高于50且下跌时,发出卖出信号。
ATR和最近14根K线的最高价、最低价用于计算止损点和止盈点。
采用移动止损策略,实现风险控制。
双重指标结合判断趋势方向,提高准确率。EMA判断主趋势,RSI和K线关系判断局部趋势和买卖时机。
RSI指标有效避免了假突破。通过RSI的多空状态避免EMA指标的滞后带来的不必要的交易。
移动止损有效控制个别大amplitude波动造成的亏损。
优化后的参数组合使策略参数鲁棒性较强。
大amplitude震荡行情中,EMA和RSI产生错误信号的概率较大,应避开此类行情。
止损点过小易造成过频止损;止损点过大难以控制损失。应适当调整ATR参数。
日间突破EMA后再回调的概率大,此时应适当放宽RSI参数,避免错过趋势。
适当调整ATR参数和止损距离,找到更好的止损点。
对EMA和RSI指标参数进行优化,找到更合适的参数组合。
添加其他辅助指标进行过滤,如MACD、布林带等,提高信号准确度。
可以测试不同品种参数设置的差异性,进一步提高参数的稳定性。
可以尝试在特定时间段关闭策略,避开容易产生错误信号的时间段。
该策略整体较为稳定,收益稳定,最大回撤和夏普率也非常优秀。通过参数优化和止损点调整,可以进一步提升策略的效果。同时也需要警惕在特定行情下可能产生的错误信号,通过辅助指标或时间过滤来避开这些行情。该策略有望通过持续优化成为一个值得长期持有的稳定策略。
/*backtest start: 2023-01-01 00:00:00 end: 2023-08-10 00:00:00 period: 1d basePeriod: 1h exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}] */ //@version=5 // strategy("EMA RSI Strategy", overlay=true, default_qty_type=strategy.percent_of_equity, default_qty_value=1) /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// // Author : AJ Rupasinghege // Date : 06/11/2022 // Release : v6.0 // Description : If the last two closes are in ascending order, the rsi is below 50 and ascending, and the current candle is above 200 ema, then LONG. // If the last two closes are in descending order, the rsi is above 50 and descending, and the current candle is below 200 ema, then SHORT. ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// INPUTS ////////////////////////////////////////////////////////////// ema_length = input(200, "EMA Length") rsi_buy_value = input(50, "RSI Buy Value") rsi_sell_value = input(50, "RSI Sell Value") show_data = input.bool(0, "Show Data") ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// VARIABLES ////////////////////////////////////////////////////////// var stop_loss = 0.0 var last_trade_entry_price = 0.0 var low_value= 0.0 var atr = 0.0 var high_value = 0.0 var stop_loss_points = 0.0 var limit = 0.0 var bar_id_entry = 0 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// FUNCTIONS ////////////////////////////////////////////////////////// getTradeConditionsLong() => //@function Used to calculate stop_loss, stop_loss points, limit and label values for long trades //@param direction (float) // strategy.poistion.size //@returns stop_loss, stop_loss_points, limit //@Dependancies low_value, atr, last_trade_entry_price,bar_id_entry _stop_loss = low_value - atr _stop_lossPoints = (last_trade_entry_price - _stop_loss) *100000 _limit = last_trade_entry_price + (last_trade_entry_price - low_value + atr) value = "OpenValue: " + str.tostring(last_trade_entry_price) + "\n OpenBarIndex: " + str.tostring(bar_id_entry) + "\n LowValue: " + str.tostring(low_value) + "\n atr: " + str.tostring(atr) + "\n stop_loss: " + str.tostring(_stop_loss) + "\n Limit: " + str.tostring(_limit) [_stop_loss,_stop_lossPoints,_limit, value] getTradeConditionsShort() => //@function Used to calculate stop_loss, stop_loss points, limit and label values for short trades //@param direction (float) // strategy.poistion.size //@returns stop_loss, stop_loss_points, limit //@Dependancies high_value, atr, last_trade_entry_price,bar_id_entry _stop_loss = high_value + atr _stop_lossPoints = (_stop_loss -last_trade_entry_price) * 100000 _limit = last_trade_entry_price - (high_value - last_trade_entry_price + atr) value = "OpenValue: " + str.tostring(last_trade_entry_price) + "\n OpenBarIndex: " + str.tostring(bar_id_entry) + "\n HighValue: " + str.tostring(high_value) + "\n atr: " + str.tostring(atr) + "\n stop_loss: " + str.tostring(_stop_loss) + "\n Limit: " + str.tostring(_limit) [_stop_loss,_stop_lossPoints,_limit, value] ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// SIGNALS ////////////////////////////////////////////////////////// ema = ta.ema(close,ema_length) rsi = ta.rsi(close,14) ema_buy_signal = ema < low ema_sell_signal = ema > high rsi_buy_signal = rsi < rsi_buy_value and rsi[1] < rsi[0] rsi_sell_signal = rsi > rsi_sell_value and rsi[1] > rsi[0] trend_buy_signal = close[2] < close[1] and close[1] < close[0] trend_sell_signal = close[2] > close[1] and close[1] > close[0] ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// TRADES ////////////////////////////////////////////////////////// long = trend_buy_signal and ema_buy_signal and rsi_buy_signal short = trend_sell_signal and ema_sell_signal and rsi_sell_signal ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// STRATEGY ////////////////////////////////////////////////////////// if long strategy.entry("Long", strategy.long) if short strategy.entry("Short", strategy.short) // Calculate Trade Entry Variables last_trade_entry_price := strategy.opentrades.entry_price(strategy.opentrades - 1) bar_id_entry := strategy.opentrades.entry_bar_index(strategy.opentrades - 1) atr := ta.atr(14) low_value := ta.lowest(14) high_value := ta.highest(14) // Exit Strategy for Long Positions if (strategy.position_size[1] != strategy.position_size and strategy.position_size>0) [_stop_loss,_stop_loss_points, _limit, value] = getTradeConditionsLong() stop_loss := _stop_loss stop_loss_points := _stop_loss_points limit := _limit if show_data label.new(bar_id_entry,stop_loss - 0.005,str.tostring(value),xloc = xloc.bar_index,yloc = yloc.price,style = label.style_none) strategy.exit("Long Exit", "Long", trail_offset = stop_loss_points/2, trail_points = stop_loss_points/2 , stop = stop_loss , limit = limit ) // Exit Strategy for Short Positions if (strategy.position_size[1] != strategy.position_size and strategy.position_size<0) [_stop_loss,_stop_loss_points, _limit, value] = getTradeConditionsShort() stop_loss := _stop_loss stop_loss_points := _stop_loss_points limit := _limit if show_data label.new(bar_id_entry,stop_loss + 0.005,str.tostring(value),xloc = xloc.bar_index,yloc = yloc.price,style = label.style_none) strategy.exit("Short Exit", "Short", trail_offset = stop_loss_points/2, trail_points = stop_loss_points/2 , stop = stop_loss , limit = limit ) ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////// PLOTS ////////////////////////////////////////////////////////// plot(ema, "SMA", color = color.blue, linewidth = 2 ) p1 = plot(strategy.position_size>0? stop_loss:na, style = plot.style_linebr , color = color.rgb(82, 240, 42) ) p2 = plot(strategy.position_size<0? stop_loss:na, style = plot.style_linebr , color = color.rgb(223, 85, 85) ) p3 = plot(strategy.position_size!=0?limit : na, style = plot.style_linebr , color = color.rgb(94, 85, 223, 100) ) fill(p1, p3, color = color.new(color.green, 90)) fill(p2, p3, color = color.new(#e98787, 90))