Summary: BTC波动导航者EA是一款专为比特币设计的MQL4智能交易系统。使用布林带波动率收缩突破、RSI动量确认和动态仓位管理,适合H4周期自动运行。
BTC波动导航者EA专为比特币独特的市场行为而构建:高波动性、频繁跳空和强烈的趋势持续性。EA检测波动率收缩期(布林带收窄),并在RSI动量确认下进行突破入场。动态仓位管理机制在极端波动期间减少敞口。系统包含基于ATR的硬止损、保护利润的追踪止损以及保护资金的一周回撤限制器。
推荐加载周期: H4
策略核心逻辑:
1. 波动率收缩:布林带(20,2)带宽低于50根K线平均带宽的0.3倍。
2. 突破入场:价格收盘于上轨之上做多,或收盘于下轨之下做空,且RSI(14)做多>55、做空<45。
3. 动态手数:基础手数 = 每$2000权益0.01手。当ATR超过平均值的1.5倍时,手数减少50%。
4. 风险管理:止损2.5倍ATR,止盈5倍ATR。盈利达到2.5倍ATR后启动追踪止损。每周亏损上限10%。
```mql4
//+------------------------------------------------------------------+
//| BTCVolatilityNavigator.mq4 |
//+------------------------------------------------------------------+
#property copyright ""
#property link ""
#property version "1.00"
#property strict
//--- 输入参数及注释
input double RiskPerEquity = 1.0; // 每笔风险占权益百分比(1%)
input int BandsPeriod = 20; // 布林带周期
input double BandsDeviation = 2.0; // 布林带标准差倍数
input double BandwidthThreshold = 0.3; // 带宽收缩阈值(平均带宽的30%)
input int RSIPeriod = 14; // RSI周期(动量确认)
input int ATRPeriod = 14; // ATR周期(止损计算)
input double ATRStopMultiplier = 2.5; // 止损倍数(ATR的倍数)
input double ATRTakeMultiplier = 5.0; // 止盈倍数(ATR的倍数)
input double TrailingStart = 2.5; // 追踪止损启动倍数(ATR倍数)
input double TrailingStep = 1.0; // 追踪步长倍数(ATR倍数)
input int MagicNumber = 202417; // EA魔术号
input int MaxSpread = 200; // 最大允许点差(比特币点差较大)
input double WeeklyLossLimit = 10.0; // 每周亏损限额(账户余额百分比)
input bool UseWeekendClose = true; // 周日22:00 GMT前平仓
//--- 全局变量
double weeklyStartBalance = 0;
datetime lastBarTime = 0;
bool weekendCloseExecuted = false;
double avgBandwidth = 0;
double avgATR = 0;
//+------------------------------------------------------------------+
//| EA初始化函数 |
//+------------------------------------------------------------------+
int OnInit()
{
weeklyStartBalance = AccountBalance();
lastBarTime = 0;
weekendCloseExecuted = false;
avgATR = iATR(Symbol(), PERIOD_H4, ATRPeriod, 1);
if(avgATR <= 0) avgATR = 500 * Point;
avgBandwidth = CalculateBandwidth();
return(INIT_SUCCEEDED);
}
//+------------------------------------------------------------------+
//| EA退出函数 |
//+------------------------------------------------------------------+
void OnDeinit(const int reason)
{
Comment("");
}
//+------------------------------------------------------------------+
//| 计算布林带带宽 |
//+------------------------------------------------------------------+
double CalculateBandwidth()
{
double upper = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_UPPER, 1);
double lower = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_LOWER, 1);
double middle = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_MAIN, 1);
if(middle <= 0) return 0;
return (upper - lower) / middle;
}
//+------------------------------------------------------------------+
//| 计算最近50根K线的平均带宽 |
//+------------------------------------------------------------------+
double GetAvgBandwidth()
{
double sum = 0;
int count = 0;
for(int i = 1; i <= 50; i++)
{
double upper = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_UPPER, i);
double lower = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_LOWER, i);
double middle = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_MAIN, i);
if(middle > 0)
{
sum += (upper - lower) / middle;
count++;
}
}
if(count == 0) return 0.5;
return sum / count;
}
//+------------------------------------------------------------------+
//| EA主循环函数(每Tick执行) |
//+------------------------------------------------------------------+
void OnTick()
{
// 每周净值保护
double currentEquity = AccountEquity();
double lossPercent = (weeklyStartBalance - currentEquity) / weeklyStartBalance * 100;
if(lossPercent >= WeeklyLossLimit)
{
Comment("已达每周亏损上限,停止开新仓");
return;
}
// 周日晚间平仓(规避周末缺口)
if(UseWeekendClose && !weekendCloseExecuted)
{
datetime currentTime = TimeCurrent();
if(TimeDayOfWeek(currentTime) == 0 && TimeHour(currentTime) >= 22)
{
CloseAllOrders();
weekendCloseExecuted = true;
return;
}
if(TimeDayOfWeek(currentTime) != 0)
weekendCloseExecuted = false;
}
// 比特币点差过滤
if(MarketInfo(Symbol(), MODE_SPREAD) > MaxSpread)
{
Comment("比特币点差过大:", MarketInfo(Symbol(), MODE_SPREAD));
return;
}
// 仅在新K线开始时检测入场(H4)
if(Time[0] == lastBarTime)
return;
lastBarTime = Time[0];
// 更新平均ATR
double atr = iATR(Symbol(), PERIOD_H4, ATRPeriod, 1);
if(atr > 0) avgATR = (avgATR * 0.95) + (atr * 0.05);
// 检查现有持仓
if(CountPositions() > 0)
{
ManageTrailingStop();
return;
}
// 波动率收缩检测
double currentBW = CalculateBandwidth();
double avgBW = GetAvgBandwidth();
if(avgBW > 0 && currentBW > avgBW * BandwidthThreshold)
{
Comment("未检测到波动率收缩,带宽:", currentBW);
return;
}
// 布林带数值
double upperBand = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_UPPER, 1);
double lowerBand = iBands(Symbol(), PERIOD_H4, BandsPeriod, BandsDeviation, 0, PRICE_CLOSE, MODE_LOWER, 1);
double close1 = iClose(Symbol(), PERIOD_H4, 1);
double rsi = iRSI(Symbol(), PERIOD_H4, RSIPeriod, PRICE_CLOSE, 1);
// 基于余额的动态手数
double baseLot = NormalizeDouble(AccountBalance() * RiskPerEquity / 10000.0, 2);
if(baseLot < 0.01) baseLot = 0.01;
if(baseLot > 2.0) baseLot = 2.0;
// 高波动时减小手数
double volatilityFactor = 1.0;
if(atr > avgATR * 1.5 && avgATR > 0)
volatilityFactor = 0.5;
double finalLot = NormalizeDouble(baseLot * volatilityFactor, 2);
if(finalLot < 0.01) finalLot = 0.01;
int cmd = -1;
double sl = 0, tp = 0;
double ask = Ask;
double bid = Bid;
// 向上突破:收盘价高于上轨,RSI > 55 确认动量
if(close1 > upperBand && rsi > 55)
{
cmd = OP_BUY;
sl = bid - (atr * ATRStopMultiplier);
tp = bid + (atr * ATRTakeMultiplier);
}
// 向下突破:收盘价低于下轨,RSI < 45 确认动量
else if(close1 < lowerBand && rsi < 45)
{
cmd = OP_SELL;
sl = ask + (atr * ATRStopMultiplier);
tp = ask - (atr * ATRTakeMultiplier);
}
if(cmd != -1)
{
int ticket = OrderSend(Symbol(), cmd, finalLot, (cmd==OP_BUY?ask:bid), 5, sl, tp, "BTC Navigator", MagicNumber, 0, clrNONE);
if(ticket < 0)
Print("开仓失败,错误码:", GetLastError());
}
}
//+------------------------------------------------------------------+
//| 管理持仓的追踪止损 |
//+------------------------------------------------------------------+
void ManageTrailingStop()
{
for(int i = OrdersTotal()-1; i >= 0; i--)
{
if(OrderSelect(i, SELECT_BY_POS, MODE_TRADES))
{
if(OrderSymbol() == Symbol() && OrderMagicNumber() == MagicNumber)
{
double atr = iATR(Symbol(), PERIOD_H4, ATRPeriod, 1);
if(atr <= 0) atr = avgATR;
double newSL = 0;
double trailTrigger = atr * TrailingStart;
if(OrderType() == OP_BUY)
{
double profitPoints = (Bid - OrderOpenPrice()) / Point;
if(profitPoints >= trailTrigger / Point)
{
newSL = Bid - (atr * TrailingStep);
if(newSL > OrderStopLoss())
{
if(OrderModify(OrderTicket(), OrderOpenPrice(), newSL, OrderTakeProfit(), 0, clrNONE))
Print("买入追踪止损已更新 #", OrderTicket());
}
}
}
else if(OrderType() == OP_SELL)
{
double profitPoints = (OrderOpenPrice() - Ask) / Point;
if(profitPoints >= trailTrigger / Point)
{
newSL = Ask + (atr * TrailingStep);
if(newSL < OrderStopLoss() || OrderStopLoss() == 0)
{
if(OrderModify(OrderTicket(), OrderOpenPrice(), newSL, OrderTakeProfit(), 0, clrNONE))
Print("卖出追踪止损已更新 #", OrderTicket());
}
}
}
break;
}
}
}
}
//+------------------------------------------------------------------+
//| 统计当前魔术号的持仓数量 |
//+------------------------------------------------------------------+
int CountPositions()
{
int count = 0;
for(int i = OrdersTotal()-1; i >= 0; i--)
{
if(OrderSelect(i, SELECT_BY_POS, MODE_TRADES))
{
if(OrderSymbol() == Symbol() && OrderMagicNumber() == MagicNumber)
count++;
}
}
return count;
}
//+------------------------------------------------------------------+
//| 平仓当前品种下所有属于该EA的订单 |
//+------------------------------------------------------------------+
void CloseAllOrders()
{
for(int i = OrdersTotal()-1; i >= 0; i--)
{
if(OrderSelect(i, SELECT_BY_POS, MODE_TRADES))
{
if(OrderSymbol() == Symbol() && OrderMagicNumber() == MagicNumber)
{
if(OrderType() == OP_BUY)
OrderClose(OrderTicket(), OrderLots(), Bid, 5, clrNONE);
else if(OrderType() == OP_SELL)
OrderClose(OrderTicket(), OrderLots(), Ask, 5, clrNONE);
}
}
}
}
//+------------------------------------------------------------------+
```
参考来源: 原创MQL4代码,仅供学习参考。
免责声明: 比特币交易因高波动性、流动性缺口和经纪商差异而具有极高风险。本EA按“原样”提供,不保证盈利。实盘交易前请在模拟账户充分测试至少3个月。历史表现不代表未来结果。
```