第三帖,关于SetBasePeriod(\\\\\\\\\\\\\\\"5m\\\\\\\\\\\\\\\")

请教一个问题,上一贴发现的奇怪现象,我找到问题了,但是不知道是什么原因,还请老师们赐教,感谢!

比如在系统自带的海龟策略中进行改写时,为了控制开平仓交易,不允许在已经开平仓的BAR上继续操作,在每一个交易条件上都加了控制 && SendOrderThisBar==false,为了跨函数使用,这个控制变量改为了全局变量,

在onbaropen中使用了SetBasePeriod(1m),并在其后赋值控制SendOrderThisBar = false;,就导致了很多交易无法执行,而且出现大量闪烁,但是图上不会有任何交易。

现在我试了一下有两种情形可以不发生以上的情况:

(1)如果把SendOrderThisBar = false;这一句放在SetBasePeriod(1m)之前,交易就能发生了,但是这样控制变量就失去了意义,因为在1分钟周期上每一次都被重置,无法实现大周期的BAR控制。

(2)如果删掉Global, 把SendOrderThisBar改为局部变量Bool SendOrderThisBar(False),也能够交易了,但是这样的话,就失去了跨函数矫正SendOrderThisBar的意义,同时局部变量是否会在1分钟小周期上不断被赋值为False,是否也同样会失去对大周期的交易控制的意义呢?

还请老师指教这是为什么呢? 感谢拉!

截图如下:

data-href=

代码如下:

//------------------------------------------------------------------------

// 简称: TurtleTrader

// 名称: 海龟交易系统

// 类别: 策略应用

// 类型: 内建应用

//------------------------------------------------------------------------

Params

   Numeric nEntries(3);                    // 最大建仓次数

   Numeric RiskRatio(1);                    // % Risk Per N ( 0 - 100)

   Numeric ATRLength(20);                    // 平均波动周期 ATR Length

   Numeric boLength(20);                    // 短周期 BreakOut Length

   Numeric fsLength(55);                    // 长周期 FailSafe Length

   Numeric teLength(10);                    // 离市周期 Trailing Exit Length

   Bool LastProfitableTradeFilter(True);    // 使用入市过滤条件

Vars

   Numeric MinPoint;                        // 最小变动单位

   Series<Numeric> AvgTR;                    // ATR

   Numeric N;                                // N 值

   Numeric TotalEquity;                    // 按最新收盘价计算出的总资产

   Numeric TurtleUnits;                    // 交易单位

   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;                    // 平仓价格

   Global Bool SendOrderThisBar(False);            // 当前Bar有过交易

   Series<Numeric> preEntryPrice(0);        // 前一次开仓的价格

   Series<Bool> PreBreakoutFailure(false);    // 前一次突破是否失败

Events

   OnInit()

   {


      SetBasePeriod(1m);


   }

   OnBarOpen(ArrayRef<Integer> indexs)

   {

       

       If(GetArraySize(indexs) == 0) Return;

       SendOrderThisBar = false;

   }

   OnBar(ArrayRef<Integer> indexs)

   {

       

       If(BarStatus == 0)

       {

           preEntryPrice = InvalidNumeric;

           PreBreakoutFailure = false;

       }

       MinPoint = MinMove*PriceScale;

       AvgTR = XAverage(TrueRange,ATRLength);

       N = AvgTR[1];

       TotalEquity = Portfolio_CurrentCapital() + Portfolio_UsedMargin();

       TurtleUnits = (TotalEquity*RiskRatio/100) /(N * ContractUnit()*BigPointValue());

       TurtleUnits = IntPart(TurtleUnits); // 对小数取整

       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);

       Commentary(N=+Text(N)+,barnum=+Text(CurrentBar));

       Commentary(preEntryPrice=+Text(preEntryPrice));

       Commentary(PreBreakoutFailure=+IIFString(PreBreakoutFailure,True,False));

       // 当不使用过滤条件,或者使用过滤条件并且条件为PreBreakoutFailure为True进行后续操作

       If(MarketPosition == 0 && ((!LastProfitableTradeFilter) Or (PreBreakoutFailure)))

       {

           

           // 突破开仓

           If(High > DonchianHi && TurtleUnits >= 1 && SendOrderThisBar==false)

           {

               // 开仓价格取突破上轨+一个价位和最高价之间的较小值,这样能更接近真实情况,并能尽量保证成交

               myEntryPrice = min(high,DonchianHi + MinPoint);

               myEntryPrice = IIF(myEntryPrice < Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替

               preEntryPrice = myEntryPrice;

               Buy(TurtleUnits,myEntryPrice);

               SendOrderThisBar = True;

               PreBreakoutFailure = False;

           }

           If(Low < DonchianLo && TurtleUnits >= 1 && SendOrderThisBar==false)

           {

               // 开仓价格取突破下轨-一个价位和最低价之间的较大值,这样能更接近真实情况,并能尽量保证成交

               myEntryPrice = max(low,DonchianLo - MinPoint);

               myEntryPrice = IIF(myEntryPrice > Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替

               preEntryPrice = myEntryPrice;

               SellShort(TurtleUnits,myEntryPrice);

               SendOrderThisBar = True;

               PreBreakoutFailure = False;

           }

       }

       // 长周期突破开仓 Failsafe Breakout point

       If(MarketPosition == 0)

       {

           Commentary(fsDonchianHi=+Text(fsDonchianHi));

           If(High > fsDonchianHi && TurtleUnits >= 1 && SendOrderThisBar==false)

           {

               // 开仓价格取突破上轨+一个价位和最高价之间的较小值,这样能更接近真实情况,并能尽量保证成交

               myEntryPrice = min(high,fsDonchianHi + MinPoint);

               myEntryPrice = IIF(myEntryPrice < Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替

               preEntryPrice = myEntryPrice;

               Buy(TurtleUnits,myEntryPrice);

               SendOrderThisBar = True;

               PreBreakoutFailure = False;

           }

           Commentary(fsDonchianLo=+Text(fsDonchianLo));

           If(Low < fsDonchianLo && TurtleUnits >= 1 && SendOrderThisBar==false)

           {

               // 开仓价格取突破下轨-一个价位和最低价之间的较大值,这样能更接近真实情况,并能尽量保证成交

               myEntryPrice = max(low,fsDonchianLo - MinPoint);

               myEntryPrice = IIF(myEntryPrice > Open, Open,myEntryPrice); // 大跳空的时候用开盘价代替

               preEntryPrice = myEntryPrice;

               SellShort(TurtleUnits,myEntryPrice);

               SendOrderThisBar = True;

               PreBreakoutFailure = False;

           }

       }

       If(MarketPosition == 1) // 有多仓的情况

       {

           Commentary(ExitLowestPrice=+Text(ExitLowestPrice));

           If(Low < ExitLowestPrice && SendOrderThisBar==false)

           {

               myExitPrice = max(Low,ExitLowestPrice - MinPoint);

               myExitPrice = IIF(myExitPrice > Open, Open,myExitPrice); // 大跳空的时候用开盘价代替

               Sell(0,myExitPrice);    // 数量用0的情况下将全部平仓

           }Else

           {

               If(preEntryPrice!=InvalidNumeric && TurtleUnits >= 1)

               {

                   If(Open >= preEntryPrice + 0.5*N && CurrentEntries < nEntries && SendOrderThisBar==false) // 如果开盘就超过设定的1/2N,则直接用开盘价增仓。

                   {

                       myEntryPrice = Open;

                       preEntryPrice = myEntryPrice;

                       Buy(TurtleUnits,myEntryPrice);

                       SendOrderThisBar = True;

                   }

                   while(High >= preEntryPrice + 0.5*N && CurrentEntries < nEntries && SendOrderThisBar==false) // 以最高价为标准,判断能进行几次增仓

                   {

                       myEntryPrice = preEntryPrice + 0.5 * N;

                       preEntryPrice = myEntryPrice;

                       if(False == Buy(TurtleUnits,myEntryPrice))

                       {

                           break;

                       }

                       SendOrderThisBar = True;

                   }

               }

               // 止损指令

               If(Low <= preEntryPrice - 2 * N && SendOrderThisBar == false) // 加仓Bar不止损

               {

                   myExitPrice = preEntryPrice - 2 * N;

                   myExitPrice = IIF(myExitPrice > Open, Open,myExitPrice); // 大跳空的时候用开盘价代替

                   Sell(0,myExitPrice); // 数量用0的情况下将全部平仓

                   PreBreakoutFailure = True;

               }

           }

       }Else If(MarketPosition ==-1) // 有空仓的情况

       {

           // 求出持空仓时离市的条件比较值

           Commentary(ExitHighestPrice=+Text(ExitHighestPrice));

           If(High > ExitHighestPrice && SendOrderThisBar==false)

           {

               myExitPrice = Min(High,ExitHighestPrice + MinPoint);

               myExitPrice = IIF(myExitPrice < Open, Open,myExitPrice); // 大跳空的时候用开盘价代替

               BuyToCover(0,myExitPrice);    // 数量用0的情况下将全部平仓

           }Else

           {

               If(preEntryPrice!=InvalidNumeric && TurtleUnits >= 1)

               {

                   If(Open <= preEntryPrice - 0.5*N && CurrentEntries < nEntries && SendOrderThisBar==false) // 如果开盘就超过设定的1/2N,则直接用开盘价增仓。

                   {

                       myEntryPrice = Open;

                       preEntryPrice = myEntryPrice;

                       SellShort(TurtleUnits,myEntryPrice);

                       SendOrderThisBar = True;

                   }

                   while(Low <= preEntryPrice - 0.5*N && CurrentEntries < nEntries && SendOrderThisBar==false) // 以最低价为标准,判断能进行几次增仓

                   {

                       myEntryPrice = preEntryPrice - 0.5 * N;

                       preEntryPrice = myEntryPrice;

                       if(False == SellShort(TurtleUnits,myEntryPrice))

                       {

                           break;

                       }

                       SendOrderThisBar = True;

                   }

               }

               // 止损指令

               If(High >= preEntryPrice + 2 * N && SendOrderThisBar==false) // 加仓Bar不止损

               {

                   myExitPrice = preEntryPrice + 2 * N;

                   myExitPrice = IIF(myExitPrice < Open, Open,myExitPrice); // 大跳空的时候用开盘价代替

                   BuyToCover(0,myExitPrice); // 数量用0的情况下将全部平仓

                   PreBreakoutFailure = True;

               }

           }

       }

       Commentary(CurrentEntries = + Text(CurrentEntries));

   }

//------------------------------------------------------------------------

// 编译版本 GS2010.12.08

// 版权所有    TradeBlazer Software 2003-2025

// 更改声明 TradeBlazer Software保留对TradeBlazer平

//          台每一版本的TradeBlazer公式修改和重写的权利

//------------------------------------------------------------------------




关于SetBasePeriod(\\\"5m\\\")
第二帖,关于SetBasePeriod(\\\\\\\"5m\\\\\\\")
关于setbaseperiod的几个疑问
求助帖
setbaseperiod只是用来做图表回测的吧?
使用SetBasePeriod函数的K线显示与使用问题
SetBasePeriod多周期回测固定时间点发生信号闪烁
第三方tick行情如何接入tb
setbaseperiod函数没有被声明
如何编写5分钟k线上穿20均线后,第三根开多仓?然后开仓后,第三根平仓 or 收盘前一分钟平仓?

SetBasePeriod应该用在大周期数据源上

设置一个全局变量

这个全局变量对应大周期的bar周期范围内的一次性交易


买卖信号放在小周期数据源上

买卖开平时候置为True

整个代码中

注意对非全局变量的数据源区分使用

谢谢老师的解答,但是我还是不晓得怎么写。可否帮忙给个示例?或者帮忙在我那段代码上修改一下?谢谢了。

跨周期是否要用SubscribeBar,先把一分钟和1小时的bar数据引用进来?

SubscribeBar引用1分钟数据是否会与SetBasePeriod(1m);相冲突?

另外,在小周期的data1进行买卖操作后,修改控制信号为True,是否也要用data1.SendOderThisBar=True;

感谢老师!