Estrategia de swing de subposición dinámica


Fecha de creación: 2026-02-06 10:55:35 Última modificación: 2026-03-03 10:36:03
Copiar: 9 Número de Visitas: 200
2
Seguir
413
Seguidores

Estrategia de swing de subposición dinámica Estrategia de swing de subposición dinámica

MACD, WT, BB, SMA, ATR

No es una estrategia común de la franja de Brent, sino un sistema completo de comercio por niveles de riesgo.

La estrategia tradicional de las franjas de bullón solo te dice que “el precio tocó la vía de la baja”, pero Anh Nga 6.0 invirtió esta idea por completo. Dividió las franjas de bullón en dos niveles de riesgo AAA y B, con áreas AAA (en el rango de 1x el diferencial estándar) con un 100% de posiciones, y áreas B (en el rango de 1x el diferencial estándar) con un 80% de posiciones. Este diseño es más acorde con las leyes de la volatilidad del mercado que la estrategia de posiciones fijas.

La combinación de indicadores Wave Theory: el cruce WT1/WT2 ofrece un tiempo de entrada preciso

La señal central de la estrategia proviene del indicador de teoría de ondas, WT1 sobre WT2 y WT1 < 0, y WT1 bajo WT2 y WT1> 0. Esta combinación es más sensible que el simple RSI o MACD, capaz de capturar señales de reversión al comienzo de la tendencia. La retrospectiva muestra que esta combinación es mejor que los indicadores de dinámica tradicionales en situaciones de agitación.

Foro de tiempo múltiple MACD: 15 minutos + 30 minutos de doble confirmación

El problema con un solo marco de tiempo es que es fácil generar falsas señales. La estrategia introduce un gráfico MACD de 15 minutos y 30 minutos como condición de filtración: solo se permite abrir una posición cuando el MACD de ambos marcos de tiempo no es opuesto a la dirección de la operación. Este diseño reduce la probabilidad de falsas rupturas en aproximadamente un 30%.

Gestión de la división de las existencias: ganancias del 65% + seguimiento de la tendencia del 35%

Cada transacción se divide automáticamente en dos partes: el 65% de las posiciones se liquidan cuando se alcanza el 50% de los beneficios objetivo, y el 35% restante continúa hasta el cierre completo. Este diseño garantiza un retorno de ganancias estable y no se pierde la gran tendencia. Cuando se activa una parte de las ganancias, el stop loss de las posiciones restantes se ajusta automáticamente al precio de apertura de la posición, logrando una verdadera posición sin riesgo.

Estricto control de riesgos: 1.7 veces el límite de pérdidas de la banda de Brin + límite de pérdidas máximas

El stop loss se establece en una posición de la banda de Brin 1.7 veces inferior a la norma, un parámetro optimizado con mucha retroalimentación, que evita la interferencia de la fluctuación normal y permite el cierre oportuno en una situación de verdadera inversión. Al mismo tiempo, se establece un límite máximo de stop loss de $ 35, que se salta directamente cuando la pérdida esperada supera este valor.

Mecanismo de protección inversa: evitar el consumo de fondos para el cambio frecuente a las transacciones

La estrategia tiene una función de protección de reversión incorporada, que requiere esperar un período de enfriamiento de 5 ciclos cuando la dirección de la operación anterior es contraria a la señal actual. Este diseño evita el consumo de comisiones que se produce con el cambio frecuente de dirección en situaciones de crisis, y la revisión histórica muestra que este mecanismo puede aumentar los ingresos netos en un 15-20%.

Filtrado de tendencias: doble línea media + distancia mínima para asegurar la consistencia de las tendencias

Además de las señales de teoría de ondas, la estrategia requiere que los precios estén en el mismo lado de la línea media de 70 y 140 ciclos, y a una distancia de al menos 10 puntos de la línea media lenta. Este filtro múltiple asegura que solo se negocie en entornos de tendencia claros y evita señales no válidas en la clasificación horizontal.

Protección por sobreextensión: 4 veces el límite ATR para evitar caídas de persecución

Cuando el precio está a más de 4 veces ATR de la línea media rápida, la estrategia suspende la apertura de la posición. Este mecanismo evita de manera efectiva la caza y la caída después de una prolongación excesiva del precio, especialmente en el caso de fluctuaciones anormales causadas por noticias repentinas.

Integración de PineConnector: un solo toque para conectar las transacciones en la plataforma MT5

La estrategia está totalmente integrada con la interfaz de PineConnector, que permite el envío automático de señales de transacción a la plataforma MT5. Contiene la administración completa de pedidos, la configuración de números mágicos y la configuración de parámetros de riesgo, que realmente permite una conexión sin problemas desde la retroalimentación hasta el disco duro.

Escenarios de aplicación y consejos de riesgo

La estrategia es más adecuada para un entorno de mercado con una clara tendencia, con un rendimiento relativamente débil en las oscilaciones de la bolsa lateral. Se recomienda su uso en variedades de pares de divisas principales de oro y divisas con una volatilidad moderada. El retroceso histórico no representa ganancias futuras, las operaciones en bolsa real requieren una estricta aplicación de las reglas de gestión de riesgos, se recomienda el uso de una estrategia de prueba de posiciones más pequeñas para el rendimiento real inicial.

Código Fuente de la Estrategia
/*backtest
start: 2025-04-03 19:15:00
end: 2026-01-31 08:00:00
period: 1h
basePeriod: 1h
exchanges: [{"eid":"Futures_Bybit","currency":"XAUT_USDT","balance":500000}]
*/

//@version=5
strategy(title="Anh Nga 6.0 Split (Dynamic + MACD + PC)", 
     shorttitle="Anh Nga 6.0 PC Dynamic", 
     overlay=true, 
     initial_capital=50000, 
     calc_on_order_fills=false,    
     process_orders_on_close=true, 
     calc_on_every_tick=true,      
     pyramiding=0)

// --- 1. SETTINGS ---
group_time  = "Trading Window (GMT+7)"
group_risk  = "Risk Management"
group_logic = "Strategy Logic (Signal & Trend)"
group_auto  = "PineConnector Automation"
group_guard = "Reversal Guard"



// Logic Inputs
use_ma_filter   = input.bool(true, "Use Fast/Slow MA Trend Filter?", group=group_logic)
use_macd_filter = input.bool(true, "Use MACD MTF Filter?", group=group_logic)
ma_fast_len     = input.int(70, "Fast MA Length", group=group_logic)
ma_slow_len     = input.int(140, "Slow MA Length", group=group_logic)
ma_distance_min = input.float(10.0, "Min Distance from Slow MA", group=group_logic)
lookback        = input.int(3, "Signal Window (Bars)", group=group_logic)
use_overext     = input.bool(true, "Enable Overextension Filter?", group=group_logic)
atr_limit       = input.float(4.0, "Overextension ATR Limit", step=0.1, group=group_logic)

// Guard Inputs
wait_bars       = input.int(1, "Bars Between Trades (Guard)", group=group_guard)
use_rev_guard   = input.bool(true, "Enable Reversal Guard?", group=group_guard)
rev_cooldown    = input.int(5, "Opposite Trade Cooldown (Bars)", minval=1, group=group_guard)

// Risk Inputs
rr_ratio        = input.float(1.0, "Risk:Reward Ratio", group=group_risk)
bb_mult         = input.float(1.7, "BB Stop Multiplier", group=group_risk)
use_max_sl      = input.bool(true, "Filter: Skip if SL is too wide?", group=group_risk)
max_sl_dollars  = input.float(35.0, "Max SL in Dollars", group=group_risk)

// Auto Inputs (PineConnector Ready)
license_id      = input.string("YOUR_ID_HERE", "PineConnector License ID", group=group_auto)
mt5_ticker      = input.string("XAUUSDc", "MT5 Symbol Name", group=group_auto)
base_qty        = input.float(1.0, "Total Contract Size (Lot)", step=0.01, group=group_auto)
magic_number    = input.int(12345, "MT5 Magic Number (Base)", group=group_auto)

// --- 2. INDICATORS ---
ma_fast = ta.sma(close, ma_fast_len)
ma_slow = ta.sma(close, ma_slow_len)
basis   = ta.sma(close, 20)
atr     = ta.atr(14)
stdev_val = ta.stdev(close, 20)

// Standard BB for Stop Loss (Multiplier 1.7)
dev_sl   = bb_mult * stdev_val
lower_bb_sl = basis - dev_sl
upper_bb_sl = basis + dev_sl

// ZONES for Sizing
dev_1 = 1.0 * stdev_val
upper_bb_1 = basis + dev_1
lower_bb_1 = basis - dev_1

dev_1_5 = 1.5 * stdev_val
upper_bb_1_5 = basis + dev_1_5
lower_bb_1_5 = basis - dev_1_5

wt1 = ta.ema((hlc3 - ta.ema(hlc3, 10)) / (0.015 * ta.ema(math.abs(hlc3 - ta.ema(hlc3, 10)), 10)), 21)
wt2 = ta.sma(wt1, 4)

// --- MACD Filter ---
get_macd_prev() =>
    [m, s, h] = ta.macd(close, 12, 26, 9)
    [m[1], s[1], h[1]] 

[m15, s15, hist_15] = request.security(syminfo.tickerid, "15", get_macd_prev(), lookahead=barmerge.lookahead_on)
[m30, s30, hist_30] = request.security(syminfo.tickerid, "30", get_macd_prev(), lookahead=barmerge.lookahead_on)

macd_long_ok  = not use_macd_filter or not (hist_15 < 0 and hist_30 < 0)
macd_short_ok = not use_macd_filter or not (hist_15 > 0 and hist_30 > 0)

// --- 3. STATE VARIABLES ---
var float trade_sl = na
var float trade_tp_final = na
var float trade_tp_partial = na
var bool partial_hit = false 
var int last_exit_bar = 0
var int last_dir = 0
var float current_vol_partial = 0.0
var float current_vol_runner = 0.0

if strategy.position_size == 0
    partial_hit := false
    trade_sl := na
    trade_tp_final := na
    trade_tp_partial := na
    current_vol_partial := 0.0

// --- 4. LOGIC ---
is_entry_window = true
cross_long  = ta.crossover(wt1, wt2) and wt1 < 0
cross_short = ta.crossunder(wt1, wt2) and wt1 > 0
bars_since_exit = bar_index - last_exit_bar
long_allowed    = not use_rev_guard or (last_dir != -1 or bars_since_exit > rev_cooldown)
short_allowed   = not use_rev_guard or (last_dir != 1 or bars_since_exit > rev_cooldown)
can_trade_now = bars_since_exit > wait_bars

long_signal   = ta.barssince(long_allowed and cross_long) <= lookback and macd_long_ok
short_signal  = ta.barssince(short_allowed and cross_short) <= lookback and macd_short_ok

not_overext   = not use_overext or (math.abs(close - ma_fast) < (atr * atr_limit))
long_trend    = (not use_ma_filter or (close > ma_fast and close > ma_slow))
short_trend   = (not use_ma_filter or (close < ma_fast and close < ma_slow))
long_sl_dist_dollars  = math.abs(close - lower_bb_sl)
short_sl_dist_dollars = math.abs(close - upper_bb_sl)
sl_ok_long  = not use_max_sl or (long_sl_dist_dollars <= max_sl_dollars)
sl_ok_short = not use_max_sl or (short_sl_dist_dollars <= max_sl_dollars)

// --- 5. EXECUTION ---
magic_runner = magic_number + 1

// FIX: Newline separator and License ID for Close All
msg_flat = license_id + ",closeall," + mt5_ticker + ",magic=" + str.tostring(magic_number) + "\n" + license_id + ",closeall," + mt5_ticker + ",magic=" + str.tostring(magic_runner)

// ENTRY LOGIC (Split Trades - Run on Bar Close)
if barstate.isconfirmed and is_entry_window and can_trade_now and strategy.position_size == 0
    
    // --- LONG ENTRY ---
    if long_signal and long_trend and long_allowed and not_overext and close > basis and sl_ok_long and (close - ma_slow >= ma_distance_min)
        
        bool is_AAA = (close <= upper_bb_1)
        bool is_B   = (close > upper_bb_1 and close <= upper_bb_1_5)
        
        if is_AAA or is_B
            float total_lot = is_AAA ? base_qty : (base_qty * 0.8)
            current_vol_partial := math.round(total_lot * 0.65, 2)
            current_vol_runner  := math.round(total_lot - current_vol_partial, 2)
            
            if current_vol_runner < 0.01
                current_vol_runner := 0.01
                current_vol_partial := total_lot - 0.01

            trade_sl       := math.round_to_mintick(lower_bb_sl)
            trade_tp_final := math.round_to_mintick(close + (math.abs(close - trade_sl) * rr_ratio))
            trade_tp_partial := math.round_to_mintick(close + (math.abs(close - trade_tp_final) * 0.5))
            last_dir       := 1
            
            // FIX: License ID, Split Msg, and Newline Separator
            msg_A = license_id + ",buy," + mt5_ticker + ",volume=" + str.tostring(current_vol_partial, "#.##") + ",sl=" + str.tostring(trade_sl) + ",tp=" + str.tostring(trade_tp_partial) + ",magic=" + str.tostring(magic_number)
            msg_B = license_id + ",buy," + mt5_ticker + ",volume=" + str.tostring(current_vol_runner, "#.##")  + ",sl=" + str.tostring(trade_sl) + ",tp=" + str.tostring(trade_tp_final)   + ",magic=" + str.tostring(magic_runner)
            
            string type_txt = is_AAA ? "AAA (100%)" : "B (80%)"
            strategy.entry("Long", strategy.long, qty=total_lot, comment=type_txt, alert_message=msg_A + "\n" + msg_B)

    // --- SHORT ENTRY ---
    if short_signal and short_trend and short_allowed and not_overext and close < basis and sl_ok_short and (ma_slow - close >= ma_distance_min)
        
        bool is_AAA = (close >= lower_bb_1)
        bool is_B   = (close < lower_bb_1 and close >= lower_bb_1_5)

        if is_AAA or is_B
            float total_lot = is_AAA ? base_qty : (base_qty * 0.8)
            current_vol_partial := math.round(total_lot * 0.65, 2)
            current_vol_runner  := math.round(total_lot - current_vol_partial, 2)

            if current_vol_runner < 0.01
                current_vol_runner := 0.01
                current_vol_partial := total_lot - 0.01

            trade_sl       := math.round_to_mintick(upper_bb_sl)
            trade_tp_final := math.round_to_mintick(close - (math.abs(close - trade_sl) * rr_ratio))
            trade_tp_partial := math.round_to_mintick(close - (math.abs(close - trade_tp_final) * 0.5))
            last_dir       := -1

            // FIX: License ID, Split Msg, and Newline Separator
            msg_A = license_id + ",sell," + mt5_ticker + ",volume=" + str.tostring(current_vol_partial, "#.##") + ",sl=" + str.tostring(trade_sl) + ",tp=" + str.tostring(trade_tp_partial) + ",magic=" + str.tostring(magic_number)
            msg_B = license_id + ",sell," + mt5_ticker + ",volume=" + str.tostring(current_vol_runner, "#.##")  + ",sl=" + str.tostring(trade_sl) + ",tp=" + str.tostring(trade_tp_final)   + ",magic=" + str.tostring(magic_runner)

            string type_txt = is_AAA ? "AAA (100%)" : "B (80%)"
            strategy.entry("Short", strategy.short, qty=total_lot, comment=type_txt, alert_message=msg_A + "\n" + msg_B)

// MANAGEMENT LOGIC (Run on Every Tick)
if strategy.position_size > 0 and not partial_hit
    if high >= trade_tp_partial
        // FIX: Added License ID here for the modification alert
        new_sl = strategy.position_avg_price
        msg_mod = license_id + ",modify," + mt5_ticker + ",sl=" + str.tostring(new_sl) + ",magic=" + str.tostring(magic_runner)
        alert(msg_mod, alert.freq_once_per_bar)
        
        // B. BACKTEST SYNC
        strategy.close("Long", qty=current_vol_partial, comment="Partial Hit", alert_message="IGNORE")
        
        trade_sl := new_sl 
        partial_hit := true

if strategy.position_size < 0 and not partial_hit
    if low <= trade_tp_partial
        // FIX: Added License ID here for the modification alert
        new_sl = strategy.position_avg_price
        msg_mod = license_id + ",modify," + mt5_ticker + ",sl=" + str.tostring(new_sl) + ",magic=" + str.tostring(magic_runner)
        alert(msg_mod, alert.freq_once_per_bar)
        
        // B. BACKTEST SYNC
        strategy.close("Short", qty=current_vol_partial, comment="Partial Hit", alert_message="IGNORE")
        
        trade_sl := new_sl
        partial_hit := true

// FINAL EXIT (Sync)
if strategy.position_size > 0
    if low <= trade_sl or high >= trade_tp_final
        strategy.close_all(comment="Exit Long", alert_message=msg_flat)
        last_exit_bar := bar_index

if strategy.position_size < 0
    if high >= trade_sl or low <= trade_tp_final
        strategy.close_all(comment="Exit Short", alert_message=msg_flat)
        last_exit_bar := bar_index

// --- 6. VISUALS ---
plot(ma_fast, "Fast MA", color=color.new(color.teal, 0), linewidth=2)
plot(ma_slow, "Slow MA", color=color.new(color.white, 0), linewidth=3)
plot(trade_sl, "Active SL", color=color.red, style=plot.style_linebr, linewidth=2)
plot(trade_tp_final, "Final TP", color=color.green, style=plot.style_linebr, linewidth=2)

// ZONES VISUALIZATION
fill(plot(upper_bb_1, display=display.none), plot(basis, display=display.none), color=color.new(color.green, 90), title="AAA Zone Long")
fill(plot(lower_bb_1, display=display.none), plot(basis, display=display.none), color=color.new(color.red, 90), title="AAA Zone Short")
fill(plot(upper_bb_1_5, display=display.none), plot(upper_bb_1, display=display.none), color=color.new(color.yellow, 90), title="B Zone Long")
fill(plot(lower_bb_1_5, display=display.none), plot(lower_bb_1, display=display.none), color=color.new(color.yellow, 90), title="B Zone Short")