Viva Las Vegas 策略
概述
Viva Las Vegas 是一个充满娱乐性的资金管理专家顾问,它会在每次机会随机选择做多或做空,然后交由五种经典的倍投系统之一来决定下一笔仓位的大小。StockSharp 版本完整保留了原始 MetaTrader 行为:
- 每次尝试都会通过伪随机掷硬币选择方向。
- 立即按照点数距离放置对称的止损和止盈保护。
- 一旦上一笔仓位平仓,就立刻更新资金管理序列并打开新的仓位。
因此策略始终保持单笔持仓,便于观察不同投注系统在 StockSharp 交易框架中的表现。
资金管理模块
MoneyManagement 参数从以下倍投模型中进行选择,所有模型都以 BaseVolume 作为基准手数:
- Martingale(马丁格尔) – 每次亏损后将手数加倍,盈利后恢复到基准手数。
- Negative Pyramid(反金字塔) – 亏损后手数加倍,盈利后手数减半,但不会低于基准手数。
- Labouchere(拉布歇尔) – 维护一个数字序列(默认
1-2-3),下注时取首尾数字之和;盈利后移除首尾,亏损后将两者之和追加到序列末尾。 - Oscar’s Grind(奥斯卡推进) – 盈利时按基准手数逐步提高仓位,直到累计盈利达到一个基准手数后重置;亏损只会减小运行中的盈利值。
- 31 System(31 系统) – 在序列
1,1,1,2,2,4,4,8,8中循环,第一次盈利会把当前数值翻倍,第二次连续盈利后重置到序列开头。
所有模块都遵循原始 MQL 实现的细节,包括将收益为零的交易视为亏损。
交易流程
- 启动时根据
Seed参数初始化伪随机数生成器(Seed = 0时使用时间作为种子),并通过StartProtection打开对称的止损和止盈。 - 当没有持仓且没有挂单时,策略向当前资金管理模块请求下一笔手数,将结果按标的物的
VolumeStep四舍五入,然后抛硬币决定是BuyMarket还是SellMarket。 - 仓位建立后,由保护模块根据设定的点数距离管理退出。
- 仓位回到零时,比较新的
PnL与上一笔交易时的基准值:- 利润 > 0 → 发送 win(盈利)通知。
- 利润 ≤ 0 → 发送 loss(亏损)通知。
- 周期立即重新开始,因此策略要么持有单笔仓位,要么正等待新的随机入场。
由于任意时刻只有一笔仓位,策略在图表上的表现与原始 EA 的单票模式完全一致。
参数
| 名称 | 类型 | 默认值 | 说明 |
|---|---|---|---|
StopTakePips |
int |
50 |
以点为单位的止损/止盈距离,通过 StartProtection 同步应用到保护单。 |
BaseVolume |
decimal |
1 |
资金管理算法使用的基准手数。 |
MoneyManagement |
MoneyManagementMode |
Martingale |
控制下一笔下单手数的倍投模型。 |
Seed |
int |
0 |
伪随机数种子,设置为 0 时采用时间戳保证每次运行结果不同。 |
实现说明
- 手数会按
VolumeStep对齐,并检查MinVolume/MaxVolume限制,避免因手数非法而被交易所拒单。 - 止损/止盈距离采用 MetaTrader 的传统换算方式:报价小数位为 3 或 5 时,一个点等于 10 个最小跳动。
- 盈亏通过策略的
PnL属性计算,从而让保护单和平仓动作与原始 EA 一样影响资金管理序列。 - 代码中的英文注释标明了关键决策节点,便于学习或修改成其它实验策略。
使用建议
- 建议在模拟账户或行情回放环境中运行,此类倍投策略风险极高,仅适合实验用途。
- 启动前请将
BaseVolume调整到符合标的物的合约最小单位。 - 搭配 StockSharp 图表可以直观观察不同资金管理模型如何扩张或收缩仓位规模。
namespace StockSharp.Samples.Strategies;
using System;
using System.Linq;
using System.Collections.Generic;
using Ecng.Common;
using Ecng.Collections;
using Ecng.Serialization;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
using StockSharp.Algo;
public class VivaLasVegasStrategy : Strategy
{
private readonly StrategyParam<int> _stopTakePips;
private readonly StrategyParam<decimal> _baseVolume;
private readonly StrategyParam<MoneyManagementModes> _moneyManagementMode;
private readonly StrategyParam<int> _seed;
private int _activeSeed;
private IMoneyManagement _management;
private decimal _previousPosition;
private decimal _lastRealizedPnL;
private bool _orderInFlight;
public enum MoneyManagementModes
{
Martingale,
NegativePyramid,
Labouchere,
OscarsGrind,
System31,
}
public VivaLasVegasStrategy()
{
_stopTakePips = Param(nameof(StopTakePips), 50)
.SetGreaterThanZero()
.SetDisplay("Stop/take distance", "Protective distance expressed in pips for both stop-loss and take-profit.", "Risk")
.SetOptimize(10, 200, 10);
_baseVolume = Param(nameof(BaseVolume), 1m)
.SetGreaterThanZero()
.SetDisplay("Base volume", "Initial lot size used as the anchor for all money management progressions.", "Risk")
.SetOptimize(0.1m, 5m, 0.1m);
_moneyManagementMode = Param(nameof(MoneyManagement), MoneyManagementModes.Martingale)
.SetDisplay("Money management", "Progression model that decides the next order volume.", "General");
_seed = Param(nameof(Seed), 0)
.SetDisplay("Random seed", "Seed for the pseudo-random trade direction. Zero switches to time-based seeding.", "General");
}
public int StopTakePips
{
get => _stopTakePips.Value;
set => _stopTakePips.Value = value;
}
public decimal BaseVolume
{
get => _baseVolume.Value;
set => _baseVolume.Value = value;
}
public MoneyManagementModes MoneyManagement
{
get => _moneyManagementMode.Value;
set
{
_moneyManagementMode.Value = value;
InitializeMoneyManagement();
}
}
public int Seed
{
get => _seed.Value;
set => _seed.Value = value;
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_activeSeed = 0;
_management = null;
_previousPosition = 0m;
_lastRealizedPnL = 0m;
_orderInFlight = false;
}
/// <inheritdoc />
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
Volume = BaseVolume;
InitializeMoneyManagement();
_activeSeed = Seed == 0 ? System.Environment.TickCount : Seed;
var steps = StopTakePips * GetPipMultiplier();
if (StopTakePips > 0 && steps > 0m)
{
var unit = new Unit(steps, UnitTypes.Absolute);
StartProtection(unit, unit);
}
else
{
StartProtection(null, null);
}
// Use candle subscription to pace trades
var subscription = SubscribeCandles(TimeSpan.FromMinutes(5).TimeFrame());
subscription.Bind(ProcessCandle).Start();
}
private void ProcessCandle(ICandleMessage candle)
{
if (candle.State != CandleStates.Finished)
return;
if (Position == 0m && !_orderInFlight)
TryOpenPosition();
}
/// <inheritdoc />
protected override void OnOwnTradeReceived(MyTrade trade)
{
base.OnOwnTradeReceived(trade);
if (Position != 0m)
{
// A fill confirmed our open position, so further market orders must wait.
_orderInFlight = false;
}
}
/// <inheritdoc />
protected override void OnPositionReceived(Position position)
{
base.OnPositionReceived(position);
if (_previousPosition == 0m && Position != 0m)
{
// A new position was just established; capture the realized PnL baseline.
_lastRealizedPnL = PnL;
_orderInFlight = false;
}
else if (_previousPosition != 0m && Position == 0m)
{
var tradePnL = PnL - _lastRealizedPnL;
_lastRealizedPnL = PnL;
var closedVolume = Math.Abs(_previousPosition);
if (closedVolume > 0m && _management != null)
{
var result = tradePnL > 0m ? TradeResults.Win : TradeResults.Loss;
_management.Update(result, closedVolume, BaseVolume);
}
_orderInFlight = false;
}
_previousPosition = Position;
}
private void TryOpenPosition()
{
if (ProcessState != ProcessStates.Started)
return;
if (Position != 0m || _orderInFlight)
return;
var volume = _management?.GetVolume(BaseVolume) ?? BaseVolume;
volume = AdjustVolume(volume);
if (volume <= 0m)
return;
_activeSeed = _activeSeed * 1103515245 + 12345;
var isBuy = ((_activeSeed >> 16) & 1) == 0;
_orderInFlight = true;
if (isBuy)
{
// Coin toss favoured the bullish side.
BuyMarket(volume);
}
else
{
// Bearish outcome - sell into the market.
SellMarket(volume);
}
}
private decimal AdjustVolume(decimal volume)
{
var security = Security;
if (security == null)
return volume;
var step = security.VolumeStep ?? 0m;
if (step > 0m)
{
var steps = Math.Max(1m, Math.Round(volume / step, MidpointRounding.AwayFromZero));
volume = steps * step;
}
var minVolume = security.MinVolume ?? 0m;
if (minVolume > 0m && volume < minVolume)
volume = minVolume;
var maxVolume = security.MaxVolume ?? 0m;
if (maxVolume > 0m && volume > maxVolume)
volume = maxVolume;
return volume;
}
private decimal GetPipMultiplier()
{
var security = Security;
if (security == null)
return 1m;
return security.Decimals is 3 or 5 ? 10m : 1m;
}
private void InitializeMoneyManagement()
{
_management = CreateMoneyManagement(MoneyManagement);
_management.Reset(BaseVolume);
}
private IMoneyManagement CreateMoneyManagement(MoneyManagementModes mode)
{
return mode switch
{
MoneyManagementModes.Martingale => new MartingaleManagement(),
MoneyManagementModes.NegativePyramid => new NegativePyramidManagement(),
MoneyManagementModes.Labouchere => new LabouchereManagement(),
MoneyManagementModes.OscarsGrind => new OscarsGrindManagement(),
MoneyManagementModes.System31 => new System31Management(),
_ => new MartingaleManagement(),
};
}
private enum TradeResults
{
Win,
Loss,
}
private interface IMoneyManagement
{
decimal GetVolume(decimal baseVolume);
void Update(TradeResults result, decimal closedVolume, decimal baseVolume);
void Reset(decimal baseVolume);
}
private sealed class MartingaleManagement : IMoneyManagement
{
private decimal _nextVolume;
public decimal GetVolume(decimal baseVolume)
{
if (_nextVolume <= 0m)
_nextVolume = baseVolume;
return _nextVolume;
}
public void Update(TradeResults result, decimal closedVolume, decimal baseVolume)
{
_nextVolume = result == TradeResults.Win ? baseVolume : _nextVolume * 2m;
}
public void Reset(decimal baseVolume)
{
_nextVolume = 0m;
}
}
private sealed class NegativePyramidManagement : IMoneyManagement
{
private decimal _nextVolume;
public decimal GetVolume(decimal baseVolume)
{
if (_nextVolume <= 0m)
_nextVolume = baseVolume;
return _nextVolume;
}
public void Update(TradeResults result, decimal closedVolume, decimal baseVolume)
{
if (result == TradeResults.Win)
{
_nextVolume /= 2m;
if (_nextVolume < baseVolume)
_nextVolume = baseVolume;
}
else
{
_nextVolume *= 2m;
}
}
public void Reset(decimal baseVolume)
{
_nextVolume = 0m;
}
}
private sealed class LabouchereManagement : IMoneyManagement
{
private static readonly int[] _baseSeries = { 1, 2, 3 };
private readonly List<decimal> _series = new();
public decimal GetVolume(decimal baseVolume)
{
if (_series.Count == 0)
Reset(baseVolume);
if (_series.Count > 1)
return (_series[0] + _series[^1]) * baseVolume;
return _series[0] * baseVolume;
}
public void Update(TradeResults result, decimal closedVolume, decimal baseVolume)
{
if (_series.Count == 0)
Reset(baseVolume);
if (result == TradeResults.Win)
{
if (_series.Count > 2)
{
_series.RemoveAt(_series.Count - 1);
_series.RemoveAt(0);
}
else
{
Reset(baseVolume);
}
}
else
{
var first = _series[0];
var last = _series[^1];
_series.Add(first + last);
}
}
public void Reset(decimal baseVolume)
{
_series.Clear();
foreach (var value in _baseSeries)
_series.Add(value);
}
}
private sealed class OscarsGrindManagement : IMoneyManagement
{
private decimal _nextVolume;
private decimal _currentResult;
public decimal GetVolume(decimal baseVolume)
{
if (_nextVolume <= 0m)
_nextVolume = baseVolume;
return _nextVolume;
}
public void Update(TradeResults result, decimal closedVolume, decimal baseVolume)
{
if (result == TradeResults.Win)
{
_currentResult += closedVolume;
if (_currentResult >= baseVolume)
{
_nextVolume = baseVolume;
_currentResult = 0m;
return;
}
_nextVolume += baseVolume;
var cap = baseVolume + Math.Abs(_currentResult);
if (_nextVolume > cap)
_nextVolume = cap;
}
else
{
_currentResult -= closedVolume;
}
}
public void Reset(decimal baseVolume)
{
_nextVolume = 0m;
_currentResult = 0m;
}
}
private sealed class System31Management : IMoneyManagement
{
private static readonly int[] _series = { 1, 1, 1, 2, 2, 4, 4, 8, 8 };
private int _index;
private bool _doubleUp;
public decimal GetVolume(decimal baseVolume)
{
var multiplier = _series[_index];
if (_doubleUp)
multiplier *= 2;
return multiplier * baseVolume;
}
public void Update(TradeResults result, decimal closedVolume, decimal baseVolume)
{
if (result == TradeResults.Win)
{
if (!_doubleUp)
{
_doubleUp = true;
}
else
{
_doubleUp = false;
_index = 0;
}
}
else
{
if (!_doubleUp)
{
_index = (_index + 1) % _series.Length;
}
else
{
_doubleUp = false;
}
}
}
public void Reset(decimal baseVolume)
{
_index = 0;
_doubleUp = false;
}
}
}
import clr
clr.AddReference("StockSharp.Messages")
clr.AddReference("StockSharp.Algo")
clr.AddReference("StockSharp.Algo.Indicators")
clr.AddReference("StockSharp.Algo.Strategies")
import math
from System import TimeSpan
from StockSharp.Messages import DataType, CandleStates
from StockSharp.Algo.Strategies import Strategy
class viva_las_vegas_strategy(Strategy):
def __init__(self):
super(viva_las_vegas_strategy, self).__init__()
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromMinutes(5))) \
.SetDisplay("Candle type", "Timeframe for pacing trades.", "General")
self._stop_take_pips = self.Param("StopTakePips", 50) \
.SetDisplay("Stop/take distance", "Protective distance in pips.", "Risk")
self._base_volume = self.Param("BaseVolume", 1.0) \
.SetDisplay("Base volume", "Initial lot size.", "Risk")
self._seed = self.Param("Seed", 0) \
.SetDisplay("Random seed", "Seed for pseudo-random direction. Zero = time-based.", "General")
self._active_seed = 0
self._prev_position = 0.0
self._last_pnl = 0.0
self._next_volume = 0.0
self._entry_price = 0.0
self._best_price = 0.0
@property
def CandleType(self):
return self._candle_type.Value
@property
def StopTakePips(self):
return self._stop_take_pips.Value
@property
def BaseVolume(self):
return float(self._base_volume.Value)
@property
def SeedValue(self):
return self._seed.Value
def OnStarted2(self, time):
super(viva_las_vegas_strategy, self).OnStarted2(time)
import time as _time
self._active_seed = int(_time.time() * 1000) if self.SeedValue == 0 else self.SeedValue
self._prev_position = 0.0
self._last_pnl = 0.0
self._next_volume = self.BaseVolume
self._entry_price = 0.0
self._best_price = 0.0
subscription = self.SubscribeCandles(self.CandleType)
subscription.Bind(self.ProcessCandle).Start()
def _get_pip_size(self):
sec = self.Security
if sec is not None:
ps = sec.PriceStep
if ps is not None and float(ps) > 0:
return float(ps)
return 0.0001
def ProcessCandle(self, candle):
if candle.State != CandleStates.Finished:
return
close = float(candle.ClosePrice)
high = float(candle.HighPrice)
low = float(candle.LowPrice)
pip = self._get_pip_size()
stop_dist = self.StopTakePips * pip
# manage position exit
if self.Position > 0:
if close > self._best_price:
self._best_price = close
if stop_dist > 0 and (close <= self._entry_price - stop_dist or close >= self._entry_price + stop_dist):
won = close >= self._entry_price + stop_dist
self.SellMarket()
self._update_volume(won)
self._entry_price = 0.0
self._best_price = 0.0
return
elif self.Position < 0:
if close < self._best_price:
self._best_price = close
if stop_dist > 0 and (close >= self._entry_price + stop_dist or close <= self._entry_price - stop_dist):
won = close <= self._entry_price - stop_dist
self.BuyMarket()
self._update_volume(won)
self._entry_price = 0.0
self._best_price = 0.0
return
if self.Position == 0:
self._active_seed = (self._active_seed * 1103515245 + 12345) & 0x7FFFFFFF
is_buy = ((self._active_seed >> 16) & 1) == 0
self._entry_price = close
self._best_price = close
if is_buy:
self.BuyMarket()
else:
self.SellMarket()
def _update_volume(self, won):
if won:
self._next_volume = self.BaseVolume
else:
self._next_volume = self._next_volume * 2.0
def OnReseted(self):
super(viva_las_vegas_strategy, self).OnReseted()
self._active_seed = 0
self._prev_position = 0.0
self._last_pnl = 0.0
self._next_volume = 0.0
self._entry_price = 0.0
self._best_price = 0.0
def CreateClone(self):
return viva_las_vegas_strategy()