Análise estratégica das bandas RSI e OTT

Autora:ChaoZhang, Data: 2023-10-09 16:21:20
Tags:

Resumo

Esta estratégia é chamada de RSI_OTT-TP/SL. Combina indicador RSI e bandas OTT para determinar sinais de negociação, pertencentes a estratégias de tendência seguinte. A estratégia julga a direção da tendência do mercado através do indicador RSI e usa bandas OTT para localizar pontos de entrada específicos.

Estratégia lógica

  1. Esta estratégia utiliza indicadores RSI e OTT para determinar tendências e pontos de entrada.

  2. O RSI é usado para julgar a direção geral da tendência. O RSI pode mostrar se o mercado está sobrecomprado ou sobrevendido. O cruzamento do RSI acima do nível de sobrecompra é um sinal de compra, enquanto o cruzamento abaixo do nível de sobrevenda é um sinal de venda. O comprimento padrão do RSI é 6, o nível de sobrecompra é 50 e o nível de sobrevenda também é 50 nesta estratégia.

  3. As bandas OTT são usadas para descobrir pontos de entrada. São bandas formadas com base no indicador de taxa de mudança de volatilidade (VAR). Quando o preço atravessa a banda inferior para cima, é um sinal de compra. Quando o preço quebra a banda superior para baixo, é um sinal de venda.

  4. Após determinar a tendência e confirmar o ponto de entrada, esta estratégia abrirá posições longas ou curtas quando o preço ultrapassar as faixas OTT.

  5. A estratégia fechará as posições automaticamente quando o preço do take profit ou stop loss for atingido.

  6. A estratégia também permite negociação apenas longa, curta ou em ambas as direcções.

Vantagens

  1. A combinação de bandas RSI e OTT pode encontrar pontos de entrada de alta probabilidade sob um julgamento preciso da tendência.

  2. As bandas OTT utilizam indicadores de momento e são muito sensíveis às flutuações de preços, o que pode detectar pontos de virada precocemente.

  3. As funções de take profit e stop loss ajudam a bloquear os lucros e limitar as perdas antes que elas se expandam, o que beneficia o controle de riscos.

  4. A estrutura do código é clara, com comentários suficientes, fácil de compreender e de modificar.

  5. Os parâmetros da estratégia podem ser ajustados de forma flexível através da interface para se adaptarem aos diferentes ambientes de mercado.

Riscos

  1. O RSI tem uma emissão atrasada e pode perder pontos de reversão da tendência, levando a perdas desnecessárias.

  2. As bandas OTT também podem gerar sinais falsos.

  3. As configurações incorretas de take profit e stop loss afetarão o desempenho da estratégia. Os parâmetros devem ser ajustados para diferentes produtos.

  4. A estratégia é testada apenas em um único produto. Os parâmetros devem ser otimizados separadamente para diferentes produtos na negociação ao vivo.

  5. A janela de tempo de backtest é curta e pode não validar plenamente a eficácia da estratégia.

Orientações de otimização

  1. Considerar a adição de outros indicadores para filtragem, tais como MACD, KD etc. para reduzir as entradas falsas.

  2. Os intervalos de tomada de lucro e de stop loss podem ser ajustados dinamicamente com base na volatilidade.

  3. Optimização de parâmetros de investigação para diferentes produtos para estabelecer critérios de selecção de parâmetros.

  4. Tente métodos de aprendizagem de máquina para otimizar dinamicamente os parâmetros da estratégia.

  5. Adicionar confirmação de volume para evitar falhas.

  6. Considerar a utilização da penetração de MA como stop loss em vez de simples stop loss porcentual.

Resumo

Em resumo, esta é uma estratégia típica de seguimento de tendência. Primeiro julga a direção da tendência através do RSI, depois usa bandas OTT para ajudar a determinar pontos de entrada específicos e, finalmente, define o take profit e o stop loss para bloquear os lucros e controlar os riscos. As vantagens desta estratégia são combinações de indicadores simples e eficazes e bons resultados de backtest. Mas também existem alguns riscos como o atraso do RSI e os falsos sinais da banda OTT. Isso requer que otimizemos cuidadosamente os parâmetros na negociação ao vivo e adicionemos outros indicadores técnicos para confirmação para melhorar a estabilidade da estratégia. Com otimização e verificação contínuas, essa estratégia pode se tornar um modelo de estratégia de seguimento de tendência muito prático.


/*backtest
start: 2023-09-08 00:00:00
end: 2023-10-08 00:00:00
period: 2h
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/
// © BigCoinHunter

//@version=5
strategy(title="RSI_OTT-TP/SL", overlay=true, 
     pyramiding=0, default_qty_type=strategy.percent_of_equity, 
     default_qty_value=100, initial_capital=1000, 
     currency=currency.USD, commission_value=0.05, 
     commission_type=strategy.commission.percent, 
     process_orders_on_close=true)

//----------- get the user inputs --------------

//---------- RSI -------------
price = input(close, title="Source")

RSIlength = input.int(defval=6,title="RSI Length") 
RSIoverSold = input.int(defval=50, title="RSI OverSold", minval=1)
RSIoverBought = input.int(defval=50, title="RSI OverBought", minval=1)

//------- OTT Bands ----------------
src = close
length=input.int(defval=1, title="OTT Period", minval=1)
percent=input.float(defval=5, title="OTT Percent", step=0.1, minval=0.001)

mav = input.string(title="OTT MA Type", defval="VAR", options=["SMA", "EMA", "WMA", "TMA", "VAR", "WWMA", "ZLEMA", "TSF"])

ottUpperPercent = input.float(title="OTT Upper Line Coeff", defval=0.01, minval = 0.001, step=0.001)
ottLowerPercent = input.float(title="OTT Lower Line Coeff", defval=0.01, minval = 0.001, step=0.001)

Var_Func(src,length)=>
    valpha=2/(length+1)
    vud1=src>src[1] ? src-src[1] : 0
    vdd1=src<src[1] ? src[1]-src : 0
    vUD=math.sum(vud1,9)
    vDD=math.sum(vdd1,9)
    vCMO=nz((vUD-vDD)/(vUD+vDD))
    VAR=0.0
    VAR:=nz(valpha*math.abs(vCMO)*src)+(1-valpha*math.abs(vCMO))*nz(VAR[1])
    
VAR=Var_Func(src,length)

Wwma_Func(src,length)=>
    wwalpha = 1/ length
    WWMA = 0.0
    WWMA := wwalpha*src + (1-wwalpha)*nz(WWMA[1])
    
WWMA=Wwma_Func(src,length)

Zlema_Func(src,length)=>
    zxLag = length/2==math.round(length/2) ? length/2 : (length - 1) / 2
    zxEMAData = (src + (src - src[zxLag]))
    ZLEMA = ta.ema(zxEMAData, length)
    
ZLEMA=Zlema_Func(src,length)

Tsf_Func(src,length)=>
    lrc = ta.linreg(src, length, 0)
    lrc1 = ta.linreg(src,length,1)
    lrs = (lrc-lrc1)
    TSF = ta.linreg(src, length, 0)+lrs
    
TSF=Tsf_Func(src,length)

getMA(src, length) =>
    ma = 0.0
    if mav == "SMA"
        ma := ta.sma(src, length)
        ma

    if mav == "EMA"
        ma := ta.ema(src, length)
        ma

    if mav == "WMA"
        ma := ta.wma(src, length)
        ma

    if mav == "TMA"
        ma := ta.sma(ta.sma(src, math.ceil(length / 2)), math.floor(length / 2) + 1)
        ma

    if mav == "VAR"
        ma := VAR
        ma

    if mav == "WWMA"
        ma := WWMA
        ma

    if mav == "ZLEMA"
        ma := ZLEMA
        ma

    if mav == "TSF"
        ma := TSF
        ma
    ma
    
MAvg=getMA(src, length)
fark=MAvg*percent*0.01
longStop = MAvg - fark
longStopPrev = nz(longStop[1], longStop)
longStop := MAvg > longStopPrev ? math.max(longStop, longStopPrev) : longStop
shortStop =  MAvg + fark
shortStopPrev = nz(shortStop[1], shortStop)
shortStop := MAvg < shortStopPrev ? math.min(shortStop, shortStopPrev) : shortStop
dir = 1
dir := nz(dir[1], dir)
dir := dir == -1 and MAvg > shortStopPrev ? 1 : dir == 1 and MAvg < longStopPrev ? -1 : dir
MT = dir==1 ? longStop: shortStop

OTT=MAvg>MT ? MT*(200+percent)/200 : MT*(200-percent)/200

light_green=#08ff12
light_red=#fe0808

OTTupper = nz(OTT[2])*(1+ottUpperPercent)
OTTlower = nz(OTT[2])*(1-ottLowerPercent)

p1 = plot(OTTupper, color=light_green, linewidth=1, title="OTT UPPER")
p2 = plot(nz(OTT[2]), color=color.new(color.yellow,0), linewidth=1, title="OTT MIDDLE")
p3 = plot(OTTlower, color=light_red, linewidth=1, title="OTT LOWER")

fill(plot1=p1, plot2=p3, title="OTT Background", color=color.new(color.aqua,90), fillgaps=false, editable=true)

buyEntry = ta.crossover(src, OTTlower)
sellEntry = ta.crossunder(src, OTTupper)

//---------- input TP/SL ---------------
tp = input.float(title="Take Profit:", defval=0.0, minval=0.0, maxval=100.0, step=0.1) * 0.01
sl = input.float(title="Stop Loss:  ", defval=0.0, minval=0.0, maxval=100.0, step=0.1) * 0.01

isEntryLong = input.bool(defval=true, title= 'Long Entry', inline="11")
isEntryShort = input.bool(defval=true, title='Short Entry', inline="11")

//---------- backtest range setup ------------
fromDay   = input.int(defval = 1, title = "From Day", minval = 1, maxval = 31)
fromMonth = input.int(defval = 1, title = "From Month", minval = 1, maxval = 12)
fromYear  = input.int(defval = 2021, title = "From Year", minval = 2010)
toDay     = input.int(defval = 30, title = "To Day", minval = 1, maxval = 31)
toMonth   = input.int(defval = 12, title = "To Month", minval = 1, maxval = 12)
toYear    = input.int(defval = 2022, title = "To Year", minval = 2010)

//------------ time interval setup -----------
start     = timestamp(fromYear, fromMonth, fromDay, 00, 00)  // backtest start window
finish    = timestamp(toYear, toMonth, toDay, 23, 59)        // backtest finish window
window()  => true // create function "within window of time"

//------- define the global variables ------
var bool long = true
var bool stoppedOutLong = false
var bool stoppedOutShort = false

//--------- Colors ---------------
//TrendColor = RSIoverBought and (price[1] > BBupper and price < BBupper) and BBbasis < BBbasis[1] ? color.red : RSIoverSold and (price[1] < BBlower and price > BBlower) and BBbasis > BBbasis[1] ? color.green : na
//bgcolor(switch2?(color.new(TrendColor,50)):na)


//--------- calculate the input/output points -----------
longProfitPrice  = strategy.position_avg_price * (1 + tp)     // tp -> take profit percentage
longStopPrice = strategy.position_avg_price * (1 - sl)        // sl -> stop loss percentage

shortProfitPrice  = strategy.position_avg_price * (1 - tp)
shortStopPrice = strategy.position_avg_price * (1 + sl)


//---------- RSI + Bollinger Bands Strategy -------------
vrsi = ta.rsi(price, RSIlength)

rsiCrossOver = ta.crossover(vrsi, RSIoverSold)
rsiCrossUnder = ta.crossunder(vrsi, RSIoverBought)

OTTCrossOver = ta.crossover(src, OTTlower)
OTTCrossUnder = ta.crossunder(src, OTTupper)

if (not na(vrsi))

    if rsiCrossOver and OTTCrossOver
        long := true
        
    if rsiCrossUnder and OTTCrossUnder
        long := false

//------- define the global variables ------
buySignall = false
sellSignall = false

//------------------- determine buy and sell points ---------------------
buySignall := window() and long  and (not stoppedOutLong)
sellSignall := window() and (not long)  and (not stoppedOutShort)


//---------- execute the strategy -----------------
if(isEntryLong and isEntryShort)
    if long 
        strategy.entry("LONG", strategy.long, when = buySignall, comment = "ENTER LONG")
        stoppedOutLong := true
        stoppedOutShort := false
    else 
        strategy.entry("SHORT", strategy.short, when = sellSignall, comment = "ENTER SHORT")
        stoppedOutLong  := false
        stoppedOutShort := true

else if(isEntryLong)
    strategy.entry("LONG", strategy.long,  when = buySignall)
    strategy.close("LONG", when = sellSignall)
    if long 
        stoppedOutLong := true
    else
        stoppedOutLong  := false

else if(isEntryShort)
    strategy.entry("SHORT", strategy.short, when = sellSignall)
    strategy.close("SHORT", when = buySignall)
    if not long
        stoppedOutShort := true
    else
        stoppedOutShort := false
    

//----------------- take profit and stop loss -----------------
if(tp>0.0 and sl>0.0)
    if ( strategy.position_size > 0 )
        strategy.exit(id="LONG", limit=longProfitPrice, stop=longStopPrice, comment="Long TP/SL Trigger")

    else if ( strategy.position_size < 0 )
        strategy.exit(id="SHORT", limit=shortProfitPrice, stop=shortStopPrice, comment="Short TP/SL Trigger")

else if(tp>0.0)
    if ( strategy.position_size > 0 )
        strategy.exit(id="LONG", limit=longProfitPrice, comment="Long TP Trigger")

    else if ( strategy.position_size < 0 )
        strategy.exit(id="SHORT", limit=shortProfitPrice, comment="Short TP Trigger")
        
else if(sl>0.0)
    if ( strategy.position_size > 0 )
        strategy.exit(id="LONG",  stop=longStopPrice, comment="Long SL Trigger")

    else if ( strategy.position_size < 0 )
        strategy.exit(id="SHORT",  stop=shortStopPrice, comment="Short SL Trigger")




Mais.