トリプルフィルタリングト​​レンドキャッチャー

EMA HULL ADX ATR STREAK
作成日: 2025-10-29 15:37:11 最終変更日: 2025-10-29 15:37:11
コピー: 5 クリック数: 155
2
フォロー
319
フォロワー

トリプルフィルタリングト​​レンドキャッチャー トリプルフィルタリングト​​レンドキャッチャー

この2:1の負債比率は,設計から明らかです.

回測データによると,このEMA+HULL+ADX組合せ戦略の核心論理は,入場品質を向上させるために三重フィルター機構を使用することである。20周期EMAは大きな方向を判断し,21周期Hullはトレンドの強さを確認し,14周期ADXは震動の状況をフィルターする。最も重要なのは40点TP配合20点SLであり,リスクリターン比は2:1に達し,これは量化戦略において比較的激進的だが合理的な設定である。

しかし,この見かけが単純で利回りの比率に惑わされてはいけません.実際の取引では,特定の品種で40ポイントのストップは長く待たされることがありますが,20ポイントのストップは高い変動環境で頻繁に誘発されることがあります.戦略の実際のパフォーマンスは,あなたが取引する特定の品種と時間枠に大きく依存します.

ADXの値20は分水であり,この数字を下回る信号は直接無視されます.

ADXはトレンドの強さの値として20を設定しており,このパラメータの選択には理由がある.ADXが20を下回ると,市場は通常横横整理状態にあり,この時点でEMAとHullの信号はしばしば偽突破である.歴史的なデータによると,ADXが20を超えた信号の勝率は,フィルターのない信号よりも15-25%高い.

しかし,隠されたリスクがあります:ADXは遅れの指標であり,トレンドを確認するときに最適なエントリーポイントが逃れている可能性があります.したがって,選択可能なADXスイッチを策略的に設計し,ADXフィルターをオフにすると,より多くの機会を捕捉し,より多くの偽信号を承受するコストで,いくつかの急速に変化する市場です.

この反人間的なデザインは賢明です.

策略の最も興味深い部分は,連続したK線フィルタリングメカニズムである。連続した3根またはそれ以上の陽線が現れたとき,多めにすることは禁じられる;連続した3根またはそれ以上の陰線があるとき,空きをすることは禁じられる。これは,完全に”追殺し落下”の本能に反しているが,データは,この逆思考が正しいことを証明している。

連続したK線は,短期的な動力の過剰放出を意味し,この時点で入場は技術的なリターンリスクに直面する.反省は,このフィルター条件を追加した後に,戦略の最大リターンが約30%減少したことを示し,いくつかの極端なトレンド状況を逃す可能性があるが,全体的なリスク調整後の収益は明らかに改善した.

EMAのトラップから遠ざけるための2倍ATRの遠隔操作

EMA距離フィルタは,この戦略のもう一つの亮点である.価格が20サイクルEMAから2倍ATRを超えると,ポジション開設は禁止される.この設計は,価格が平均線から大きく離れているときに衝動取引を防ぐ.

2倍ATRの倍数は,最適化されて, 1倍も保守的であれば多くの機会を逃し, 3倍も寛大な場合は,フィルタリング効果を及ぼさない.実用的な応用では,このパラメータは,異なる品種で調整する必要があるかもしれません:外為対は1.5-2倍に適しているかもしれません,株式指数期貨は2.5-3倍が必要かもしれません,暗号通貨は3-4倍が必要かもしれません.

ハル平均線の方向判断は,従来のMAよりも敏感ですが,騙されやすい.

ハル移動平均は,この戦略の核心となる技術指標であり,従来のEMAよりも早く反応し,トレンド転換を早期に捉えることができる.21周期の設定は,感度と安定性の間のバランスの取れた点を発見した.

しかし,Hullの迅速な反応は,両刃の剣でもある. 揺れ動いている市場では,Hullは,より多くの方向変化を生じ,より多くの偽信号を生じさせるだろう. これが,ADXフィルタリングと他の条件を組み合わせた戦略が必要である理由であり,単独でHull信号を使用する勝率は45-50%しかいないだろう.

この戦略はトレンド市場ではうまく機能していますが, 変動の動きは反復的に反発します.

適用されるシナリオからすると,このポートフォリオは,特に日内取引とショートライン波段の明確なトレンド状況で優れたパフォーマンスを発揮します.ADXフィルタリングは,方向性のある市場でのみ取引を保証し,複数のフィルタリング条件は,信号の質を向上させます.

しかし,戦略の弱点も明らかである:横軸の振動市場では,ADXフィルターがあっても,いくつかの偽の突破信号が生じます. 20点のストップは,高周波振動で頻繁に誘発されることがありますが,40点のストップは,トレンドがない市場では,達成することは困難です.

リスク・ヒント:過去回顧は将来の利益を意味せず,厳格なリスク管理が必要

この戦略には,特に市場環境の変化に伴い,明らかに損失のリスクがあります.連続損失は5-8回に達し,最大回収は15-20%を超える可能性があります.異なる市場環境でのパフォーマンスは大きく異なっており,実際の状況に応じてパラメータを調整するか使用を一時停止する必要があります.

単一のリスクは,口座の1-2%以内で制御され,戦略レベルでの最大撤退制限を設定することが推奨されます. 連続して3回以上の損失が発生した場合,取引を一時停止し,市場環境を再評価する必要があります.

ストラテジーソースコード
/*backtest
start: 2025-10-18 00:00:00
end: 2025-10-27 08:00:00
period: 5m
basePeriod: 5m
exchanges: [{"eid":"Futures_Binance","currency":"SOL_USDT"}]
*/

//@version=6
strategy("Iriza4 - DAX EMA+HULL+ADX TP40 SL20 (Streak & EMA/ATR Distance Filter)", overlay=true, default_qty_type=strategy.percent_of_equity, default_qty_value=1)

// === INPUTS ===
emaLen       = input.int(20, "EMA Length")
hullLen      = input.int(21, "HULL Length")
adxLen       = input.int(14, "ADX Length")
adxThreshold = input.float(20, "ADX Threshold")
useADX       = input.bool(true, "Use ADX filter (entry only)")

tpPoints     = input.int(40, "TP (points)")
slPoints     = input.int(20, "SL (points)")

// Filters
atrLen       = input.int(14, "ATR Length")
atrMult      = input.float(2.0, "Max distance from EMA (ATR multiples)")
maxBullStreak= input.int(3, "Block LONG if ≥ this many prior bull bars")
maxBearStreak= input.int(3, "Block SHORT if ≥ this many prior bear bars")

// === FUNCTIONS ===
// Hull Moving Average (HMA)
hma(src, length) =>
    half   = math.round(length / 2)
    sqrt_l = math.round(math.sqrt(length))
    w1 = ta.wma(src, half)
    w2 = ta.wma(src, length)
    ta.wma(2 * w1 - w2, sqrt_l)

// ADX (Wilder) manual calc
calc_adx(len) =>
    upMove   = ta.change(high)
    downMove = -ta.change(low)
    plusDM   = na(upMove) ? na : (upMove > downMove and upMove > 0 ? upMove : 0)
    minusDM  = na(downMove) ? na : (downMove > upMove and downMove > 0 ? downMove : 0)
    tr       = ta.tr(true)
    trRma    = ta.rma(tr, len)
    plusDI   = 100 * ta.rma(plusDM, len) / trRma
    minusDI  = 100 * ta.rma(minusDM, len) / trRma
    dx       = 100 * math.abs(plusDI - minusDI) / (plusDI + minusDI)
    ta.rma(dx, len)

// === INDICATORS ===
ema  = ta.ema(close, emaLen)
hull = hma(close, hullLen)
adx  = calc_adx(adxLen)
atr  = ta.atr(atrLen)

// HULL slope state
var float hull_dir = 0.0
hull_dir := hull > hull[1] ? 1 : hull < hull[1] ? -1 : hull_dir

// === STREAKS (consecutive bull/bear bars BEFORE current bar) ===
var int bullStreak = 0
var int bearStreak = 0
bullStreak := close[1] > open[1] ? bullStreak[1] + 1 : 0
bearStreak := close[1] < open[1] ? bearStreak[1] + 1 : 0

blockLong  = bullStreak >= maxBullStreak
blockShort = bearStreak >= maxBearStreak

// === EMA DISTANCE FILTER ===
distFromEMA = math.abs(close - ema)
farFromEMA  = distFromEMA > atrMult * atr

// === ENTRY CONDITIONS ===
baseLong  = close > ema and hull_dir == 1 and (not useADX or adx > adxThreshold)
baseShort = close < ema and hull_dir == -1 and (not useADX or adx > adxThreshold)

longSignal  = barstate.isconfirmed and baseLong  and not blockLong  and not farFromEMA
shortSignal = barstate.isconfirmed and baseShort and not blockShort and not farFromEMA

// === ENTRIES ===
if (longSignal and strategy.position_size == 0)
    strategy.entry("Long", strategy.long)
if (shortSignal and strategy.position_size == 0)
    strategy.entry("Short", strategy.short)

// === EXITS === (no partials, no breakeven)
if (strategy.position_size > 0)
    entryPrice = strategy.position_avg_price
    strategy.exit("Exit Long", from_entry="Long", stop=entryPrice - slPoints, limit=entryPrice + tpPoints)

if (strategy.position_size < 0)
    entryPrice = strategy.position_avg_price
    strategy.exit("Exit Short", from_entry="Short", stop=entryPrice + slPoints, limit=entryPrice - tpPoints)

// === VISUALS ===
plot(ema,  color=color.orange, title="EMA 20")
plot(hull, color=hull_dir == 1 ? color.green : color.red, title="HULL 21")
plot(adx,  title="ADX 14", color=color.new(color.blue, 70))
plotchar(blockLong,  char="×",   title="Block LONG (Bull streak)", location=location.top,   color=color.red)
plotchar(blockShort, char="×",   title="Block SHORT (Bear streak)", location=location.bottom,color=color.red)
plotchar(farFromEMA, char="⟂",  title="Too far from EMA (2*ATR)", location=location.top,   color=color.orange)

plotshape(longSignal and strategy.position_size == 0,  title="Iriza4 Long",  style=shape.triangleup,   location=location.belowbar, size=size.tiny, color=color.green)
plotshape(shortSignal and strategy.position_size == 0, title="Iriza4 Short", style=shape.triangledown, location=location.abovebar, size=size.tiny, color=color.red)

bgcolor(strategy.position_size > 0 ? color.new(color.green, 92) : strategy.position_size < 0 ? color.new(color.red, 92) : na)

// === ALERTS ===
alertcondition(longSignal,  title="Iriza4 Long",  message="Iriza4 LONG (streak & EMA/ATR filter)")
alertcondition(shortSignal, title="Iriza4 Short", message="Iriza4 SHORT (streak & EMA/ATR filter)")