照搬的海龟交易系统,signma用来看短周期是否有开仓信号,signma始终为0,说明短周期未建仓,为何boLength的数值发生变化,测试的利润也会变化?理论上短周期未建仓,不管boLength多大(1-1000)对测试结果都没有影响。用的燃油1分钟。
Params
Numeric boLength(13); // 短周期 BreakOut Length
Numeric fsLength(2); // 长周期 FailSafe Length
Numeric teLength(1); // 离市周期 Trailing Exit Length
Vars
Numeric MinPoint; // 最小变动单位
Numeric TurtleUnits(1); // 交易单位
Series<Numeric> DonchianHi; // 唐奇安通道上轨,延后1个Bar
Series<Numeric> DonchianLo; // 唐奇安通道下轨,延后1个Bar
Series<Numeric> fsDonchianHi; // 唐奇安通道上轨,延后1个Bar,长周期
Series<Numeric> fsDonchianLo; // 唐奇安通道下轨,延后1个Bar,长周期
Numeric ExitHighestPrice; // 离市时判断需要的N周期最高价
Numeric ExitLowestPrice; // 离市时判断需要的N周期最低价
Numeric myEntryPrice; // 开仓价格
Numeric myExitPrice; // 平仓价格
Bool SendOrderThisBar(False); // 当前Bar有过交易
Series<Numeric> preEntryPrice(0); // 前一次开仓的价格
Series<Bool> PreBreakoutFailure(false); // 前一次突破是否失败
Bool LastProfitableTradeFilter(True); // 使用入市过滤条件
Series<Numeric> signma;
Events
OnBar(ArrayRef<Integer> indexs)
{
If(BarStatus == 0)
{
preEntryPrice = InvalidNumeric;
PreBreakoutFailure = false;
signma =0;
}
MinPoint = MinMove*PriceScale;
DonchianHi = HighestFC(High[1],boLength);
DonchianLo = LowestFC(Low[1],boLength);
fsDonchianHi = HighestFC(High[1],fsLength);
fsDonchianLo = LowestFC(Low[1],fsLength);
ExitLowestPrice = LowestFC(Low[1],teLength);
ExitHighestPrice = HighestFC(High[1],teLength);
//myma=AverageFC(Close, maLength);
Commentary("preEntryPrice="+Text(preEntryPrice));
Commentary("PreBreakoutFailure="+IIFString(PreBreakoutFailure,"True","False"));
// 当不使用过滤条件,或者使用过滤条件并且条件为PreBreakoutFailure为True进行后续操作
If(MarketPosition == 0 && ((!LastProfitableTradeFilter) Or (PreBreakoutFailure)))
{
// 突破开仓
If(High > DonchianHi )
{
// 开仓价格取突破上轨+一个价位和最高价之间的较小值,这样能更接近真实情况,并能尽量保证成交
myEntryPrice = min(high,DonchianHi + MinPoint);
myEntryPrice = IIF(myEntryPrice < Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替
preEntryPrice = myEntryPrice;
Buy(TurtleUnits,myEntryPrice);
SendOrderThisBar = True;
PreBreakoutFailure = False;
signma=signma[1]+1;
}
If(Low < DonchianLo )
{
// 开仓价格取突破下轨-一个价位和最低价之间的较大值,这样能更接近真实情况,并能尽量保证成交
myEntryPrice = max(low,DonchianLo - MinPoint);
myEntryPrice = IIF(myEntryPrice > Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替
preEntryPrice = myEntryPrice;
SellShort(TurtleUnits,myEntryPrice);
SendOrderThisBar = True;
PreBreakoutFailure = False;
signma=signma[1]+1;
}
}
// 长周期突破开仓 Failsafe Breakout point
If(MarketPosition == 0)
{
Commentary("fsDonchianHi="+Text(fsDonchianHi));
If(High > fsDonchianHi )
{
// 开仓价格取突破上轨+一个价位和最高价之间的较小值,这样能更接近真实情况,并能尽量保证成交
myEntryPrice = min(high,fsDonchianHi + MinPoint);
myEntryPrice = IIF(myEntryPrice < Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替
preEntryPrice = myEntryPrice;
Buy(TurtleUnits,myEntryPrice);
SendOrderThisBar = True;
PreBreakoutFailure = False;
//PlotNumeric("DonchianLo",DonchianLo);
}
Commentary("fsDonchianLo="+Text(fsDonchianLo));
If(Low < fsDonchianLo )
{
// 开仓价格取突破下轨-一个价位和最低价之间的较大值,这样能更接近真实情况,并能尽量保证成交
myEntryPrice = max(low,fsDonchianLo - MinPoint);
myEntryPrice = IIF(myEntryPrice > Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替
preEntryPrice = myEntryPrice;
SellShort(TurtleUnits,myEntryPrice);
SendOrderThisBar = True;
PreBreakoutFailure = False;
//PlotNumeric("DonchianLo",DonchianLo);
}
}
If(MarketPosition == 1 && BarsSinceEntry>1) // 有多仓的情况
{
Commentary("ExitLowestPrice="+Text(ExitLowestPrice));
If(Low < ExitLowestPrice)
{
myExitPrice = max(Low,ExitLowestPrice - MinPoint);
myExitPrice = IIF(myExitPrice > Open, Open,myExitPrice); // 大跳空的时候用开盘价代替
Sell(0,myExitPrice); // 数量用0的情况下将全部平仓
}
}
If(MarketPosition ==-1 && BarsSinceEntry>1) // 有空仓的情况
{
// 求出持空仓时离市的条件比较值
Commentary("ExitHighestPrice="+Text(ExitHighestPrice));
If(High > ExitHighestPrice)
{
myExitPrice = Min(High,ExitHighestPrice + MinPoint);
myExitPrice = IIF(myExitPrice < Open, Open,myExitPrice); // 大跳空的时候用开盘价代替
BuyToCover(0,myExitPrice); // 数量用0的情况下将全部平仓
}
}
//Commentary("signma="+IIFString(signma,"True","False"));
//PlotNumeric("DonchianHi",DonchianHi);
//PlotNumeric("DonchianLo",DonchianLo);
Commentary("signma="+Text(signma));
}
最大回溯值的问题
tb有一个最大回溯值机制。举个例子,如果你的代码出现了回溯调用10根bar之前的数据,那么图表前10根bar是不会作图也不会出信号的。因为前10根bar上无法回溯调用10根bar之前的数据,无法保证业务逻辑是否正确,信号是否正确。
我看你的参数范围从1到1000,这个范围还是很大的,很有可能你设置成1000的时候就把本来在前1000根bar上的信号抹掉了,导致最后测试报告不正确
我只是举个例子,数值设为1-10,出来的结果也不同,如图。boLength不管设多大,利润应该是一样的,错在哪?
我觉得你这个图已经说明问题了。
如果你按参数升序排列,你会发现交易次数是按降序排列的。
这就已经验证我上面说的问题了。当这个参数慢慢变大的时候,会逐个抹掉样本前端的信号导致测试报告变化。
实际上打开测试报告比较一下详细的信号记录就能弄明白这个问题,后面的信号都是一致的,差别就是前面的信号。而前面信号缺失的原因,就是我上面解释的原因。