Summary: 比特币趋势卫士EA是一款专为BTCUSD设计的稳定趋势跟踪MQL4智能交易系统。使用三重EMA、ADX过滤和动态仓位管理,适合H4周期。




比特币趋势卫士EA是为比特币独特的市场行为设计的稳健趋势跟踪系统。该EA使用三重EMA云(快、中、慢)识别趋势方向和动量,结合ADX过滤震荡市场。基于ATR的动态止损适应比特币的高波动性,而波动率调整的仓位管理在剧烈波动期间降低风险敞口。系统包含权益保护、点差控制和周末持仓关闭,以管理加密特有的风险。

推荐加载周期: H4
策略核心逻辑:
1. 三重EMA趋势:上涨趋势要求EMA9 > EMA21 > EMA55;下跌趋势要求EMA9 < EMA21 < EMA55。
2. 趋势强度:ADX(14)必须大于25,做多时+DI > -DI,做空时-DI > +DI。
3. 入场确认:价格必须在回调后收盘于慢EMA之外(不在首次触碰时开仓)。
4. 风险管理:动态止损为2.5倍ATR,止盈为5倍ATR,盈利达到3倍ATR后启动追踪止损。最多持仓1单,每日亏损上限5%,每周亏损上限10%。

```mql4
//+------------------------------------------------------------------+
//| BitcoinTrendGuardEA.mq4 |
//+------------------------------------------------------------------+
#property copyright ""
#property link ""
#property version "1.00"
#property strict

//--- 输入参数及注释
input double BaseLotSize = 0.01; // 基础交易手数(每笔0.01 BTC)
input int FastEMAPeriod = 9; // 快EMA周期(趋势动量)
input int MidEMAPeriod = 21; // 中EMA周期(趋势方向)
input int SlowEMAPeriod = 55; // 慢EMA周期(主趋势)
input int ADXPeriod = 14; // ADX周期(趋势强度)
input int ADXThreshold = 25; // 最小ADX阈值(低于此值不开仓)
input int ATRPeriod = 14; // ATR周期(动态止损)
input double ATRStopMultiplier = 2.5; // 止损倍数(ATR的倍数)
input double ATRTakeMultiplier = 5.0; // 止盈倍数(ATR的倍数)
input double TrailingStart = 3.0; // 追踪止损启动倍数(ATR倍数)
input double TrailingStep = 1.0; // 追踪步长倍数(ATR倍数)
input int MagicNumber = 202417; // EA魔术号
input int MaxSpread = 120; // 最大允许点差(比特币点差较宽)
input double DailyLossLimit = 5.0; // 每日亏损限额(账户余额百分比)
input double WeeklyLossLimit = 10.0; // 每周亏损限额(账户余额百分比)
input bool UseSundayClose = true; // 周日22:00 GMT前平仓

//--- 全局变量
double dailyStartBalance = 0;
double weeklyStartBalance = 0;
datetime lastBarTime = 0;
bool sundayCloseExecuted = false;
double avgATR = 0;

//+------------------------------------------------------------------+
//| EA初始化函数 |
//+------------------------------------------------------------------+
int OnInit()
{
dailyStartBalance = AccountBalance();
weeklyStartBalance = AccountBalance();
lastBarTime = 0;
sundayCloseExecuted = false;
avgATR = iATR(Symbol(), PERIOD_H4, ATRPeriod, 1);
if(avgATR <= 0) avgATR = 800 * Point;
return(INIT_SUCCEEDED);
}

//+------------------------------------------------------------------+
//| EA退出函数 |
//+------------------------------------------------------------------+
void OnDeinit(const int reason)
{
Comment("");
}

//+------------------------------------------------------------------+
//| 基于波动率计算动态手数 |
//+------------------------------------------------------------------+
double GetDynamicLotSize()
{
double atr = iATR(Symbol(), PERIOD_H4, ATRPeriod, 1);
if(atr <= 0) atr = avgATR;

double lot = BaseLotSize;
// 波动率高于平均值时减小手数
if(atr > avgATR)
lot = BaseLotSize * (avgATR / atr);

lot = NormalizeDouble(lot, 2);
if(lot < 0.01) lot = 0.01;
if(lot > 1.0) lot = 1.0;
return lot;
}

//+------------------------------------------------------------------+
//| 检查三重EMA是否呈上涨排列 |
//+------------------------------------------------------------------+
bool IsTripleEMAUptrend()
{
double emaFast = iMA(Symbol(), PERIOD_H4, FastEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double emaMid = iMA(Symbol(), PERIOD_H4, MidEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double emaSlow = iMA(Symbol(), PERIOD_H4, SlowEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double close1 = iClose(Symbol(), PERIOD_H4, 1);

return (emaFast > emaMid && emaMid > emaSlow && close1 > emaSlow);
}

//+------------------------------------------------------------------+
//| 检查三重EMA是否呈下跌排列 |
//+------------------------------------------------------------------+
bool IsTripleEMADowntrend()
{
double emaFast = iMA(Symbol(), PERIOD_H4, FastEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double emaMid = iMA(Symbol(), PERIOD_H4, MidEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double emaSlow = iMA(Symbol(), PERIOD_H4, SlowEMAPeriod, 0, MODE_EMA, PRICE_CLOSE, 1);
double close1 = iClose(Symbol(), PERIOD_H4, 1);

return (emaFast < emaMid && emaMid < emaSlow && close1 < emaSlow);
}

//+------------------------------------------------------------------+
//| 检查ADX看涨确认 |
//+------------------------------------------------------------------+
bool IsADXBullish()
{
double adx = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_MAIN, 1);
double diPlus = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_PLUSDI, 1);
double diMinus = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_MINUSDI, 1);

return (adx > ADXThreshold && diPlus > diMinus);
}

bool IsADXBearish()
{
double adx = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_MAIN, 1);
double diPlus = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_PLUSDI, 1);
double diMinus = iADX(Symbol(), PERIOD_H4, ADXPeriod, PRICE_CLOSE, MODE_MINUSDI, 1);

return (adx > ADXThreshold && diMinus > diPlus);
}

//+------------------------------------------------------------------+
//| 管理持仓的追踪止损 |
//+------------------------------------------------------------------+
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())
{
OrderModify(OrderTicket(), OrderOpenPrice(), newSL, OrderTakeProfit(), 0, clrNONE);
}
}
}
else if(OrderType() == OP_SELL)
{
double profitPoints = (OrderOpenPrice() - Ask) / Point;
if(profitPoints >= trailTrigger / Point)
{
newSL = Ask + (atr * TrailingStep);
if(newSL < OrderStopLoss() || OrderStopLoss() == 0)
{
OrderModify(OrderTicket(), OrderOpenPrice(), newSL, OrderTakeProfit(), 0, clrNONE);
}
}
}
break;
}
}
}
}

//+------------------------------------------------------------------+
//| EA主循环函数(每Tick执行) |
//+------------------------------------------------------------------+
void OnTick()
{
// 每日及每周净值保护
double currentEquity = AccountEquity();
double dailyLoss = (dailyStartBalance - currentEquity) / dailyStartBalance * 100;
double weeklyLoss = (weeklyStartBalance - currentEquity) / weeklyStartBalance * 100;

if(dailyLoss >= DailyLossLimit)
{
Comment("已达每日亏损上限,停止开新仓");
return;
}
if(weeklyLoss >= WeeklyLossLimit)
{
Comment("已达每周亏损上限,停止开新仓");
return;
}

// 每日重置余额
datetime currentTime = TimeCurrent();
static datetime lastDayCheck = 0;
if(TimeDayOfYear(currentTime) != TimeDayOfYear(lastDayCheck))
{
dailyStartBalance = AccountEquity();
lastDayCheck = currentTime;
}

// 每周重置余额
static datetime lastWeekCheck = 0;
if(TimeDayOfWeek(currentTime) == 1 && TimeDayOfYear(currentTime) != TimeDayOfYear(lastWeekCheck))
{
weeklyStartBalance = AccountEquity();
lastWeekCheck = currentTime;
}

// 周日收盘前平仓(规避周末缺口)
if(UseSundayClose && !sundayCloseExecuted)
{
if(TimeDayOfWeek(currentTime) == 0 && TimeHour(currentTime) >= 22)
{
CloseAllOrders();
sundayCloseExecuted = true;
return;
}
if(TimeDayOfWeek(currentTime) != 0)
sundayCloseExecuted = 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 lot = GetDynamicLotSize();
int cmd = -1;
double sl = 0, tp = 0;
double ask = Ask;
double bid = Bid;

// 做多条件:三重EMA上涨排列 + ADX看涨确认
if(IsTripleEMAUptrend() && IsADXBullish())
{
cmd = OP_BUY;
sl = bid - (atr * ATRStopMultiplier);
tp = bid + (atr * ATRTakeMultiplier);
}
// 做空条件:三重EMA下跌排列 + ADX看跌确认
else if(IsTripleEMADowntrend() && IsADXBearish())
{
cmd = OP_SELL;
sl = ask + (atr * ATRStopMultiplier);
tp = ask - (atr * ATRTakeMultiplier);
}

if(cmd != -1)
{
int ticket = OrderSend(Symbol(), cmd, lot, (cmd==OP_BUY?ask:bid), 5, sl, tp, "Bitcoin TrendGuard", MagicNumber, 0, clrNONE);
if(ticket < 0)
Print("开仓失败,错误码:", GetLastError());
}
}

//+------------------------------------------------------------------+
//| 统计当前魔术号的持仓数量 |
//+------------------------------------------------------------------+
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个月。历史表现不代表未来结果。
```