算法示例

直接委托算法

直接发出新的委托(限价单、停止单、市价单)

  1. def on_tick(self, tick: TickData):
  2. """"""
  3. if not self.vt_orderid:
  4. if self.direction == Direction.LONG:
  5. self.vt_orderid = self.buy(
  6. self.vt_symbol,
  7. self.price,
  8. self.volume,
  9. self.order_type,
  10. self.offset
  11. )
  12. else:
  13. self.vt_orderid = self.sell(
  14. self.vt_symbol,
  15. self.price,
  16. self.volume,
  17. self.order_type,
  18. self.offset
  19. )
  20. self.put_variables_event()

时间加权平均算法

  • 将委托数量平均分布在某个时间区域内;

  • 每隔一段时间用指定的价格挂出买单(或者卖单)。

  • 买入情况:买一价低于目标价格时,发出委托,委托数量在剩余委托量与委托分割量中取最小值。

  • 卖出情况:卖一价高于目标价格时,发出委托,委托数量在剩余委托量与委托分割量中取最小值。

  1. def on_timer(self):
  2. """"""
  3. self.timer_count += 1
  4. self.total_count += 1
  5. self.put_variables_event()
  6. if self.total_count >= self.time:
  7. self.write_log("执行时间已结束,停止算法")
  8. self.stop()
  9. return
  10. if self.timer_count < self.interval:
  11. return
  12. self.timer_count = 0
  13. tick = self.get_tick(self.vt_symbol)
  14. if not tick:
  15. return
  16. self.cancel_all()
  17. left_volume = self.volume - self.traded
  18. order_volume = min(self.order_volume, left_volume)
  19. if self.direction == Direction.LONG:
  20. if tick.ask_price_1 <= self.price:
  21. self.buy(self.vt_symbol, self.price,
  22. order_volume, offset=self.offset)
  23. else:
  24. if tick.bid_price_1 >= self.price:
  25. self.sell(self.vt_symbol, self.price,
  26. order_volume, offset=self.offset)

冰山算法

  • 在某个价位挂单,但是只挂一部分,直到全部成交。

  • 买入情况:先检查撤单:最新Tick卖一价低于目标价格,执行撤单;若无活动委托,发出委托:委托数量在剩余委托量与挂出委托量中取最小值。

  • 卖出情况:先检查撤单:最新Tick买一价高于目标价格,执行撤单;若无活动委托,发出委托:委托数量在剩余委托量与挂出委托量中取最小值。

  1. def on_timer(self):
  2. """"""
  3. self.timer_count += 1
  4. if self.timer_count < self.interval:
  5. self.put_variables_event()
  6. return
  7. self.timer_count = 0
  8. contract = self.get_contract(self.vt_symbol)
  9. if not contract:
  10. return
  11. # If order already finished, just send new order
  12. if not self.vt_orderid:
  13. order_volume = self.volume - self.traded
  14. order_volume = min(order_volume, self.display_volume)
  15. if self.direction == Direction.LONG:
  16. self.vt_orderid = self.buy(
  17. self.vt_symbol,
  18. self.price,
  19. order_volume,
  20. offset=self.offset
  21. )
  22. else:
  23. self.vt_orderid = self.sell(
  24. self.vt_symbol,
  25. self.price,
  26. order_volume,
  27. offset=self.offset
  28. )
  29. # Otherwise check for cancel
  30. else:
  31. if self.direction == Direction.LONG:
  32. if self.last_tick.ask_price_1 <= self.price:
  33. self.cancel_order(self.vt_orderid)
  34. self.vt_orderid = ""
  35. self.write_log(u"最新Tick卖一价,低于买入委托价格,之前委托可能丢失,强制撤单")
  36. else:
  37. if self.last_tick.bid_price_1 >= self.price:
  38. self.cancel_order(self.vt_orderid)
  39. self.vt_orderid = ""
  40. self.write_log(u"最新Tick买一价,高于卖出委托价格,之前委托可能丢失,强制撤单")
  41. self.put_variables_event()

狙击手算法

  • 监控最新tick推送的行情,发现好的价格立刻报价成交。

  • 买入情况:最新Tick卖一价低于目标价格时,发出委托,委托数量在剩余委托量与卖一量中取最小值。

  • 卖出情况:最新Tick买一价高于目标价格时,发出委托,委托数量在剩余委托量与买一量中取最小值。

  1. def on_tick(self, tick: TickData):
  2. """"""
  3. if self.vt_orderid:
  4. self.cancel_all()
  5. return
  6. if self.direction == Direction.LONG:
  7. if tick.ask_price_1 <= self.price:
  8. order_volume = self.volume - self.traded
  9. order_volume = min(order_volume, tick.ask_volume_1)
  10. self.vt_orderid = self.buy(
  11. self.vt_symbol,
  12. self.price,
  13. order_volume,
  14. offset=self.offset
  15. )
  16. else:
  17. if tick.bid_price_1 >= self.price:
  18. order_volume = self.volume - self.traded
  19. order_volume = min(order_volume, tick.bid_volume_1)
  20. self.vt_orderid = self.sell(
  21. self.vt_symbol,
  22. self.price,
  23. order_volume,
  24. offset=self.offset
  25. )
  26. self.put_variables_event()

条件委托算法

  • 监控最新tick推送的行情,发现行情突破立刻报价成交。

  • 买入情况:Tick最新价高于目标价格时,发出委托,委托价为目标价格加上超价。

  • 卖出情况:Tick最新价低于目标价格时,发出委托,委托价为目标价格减去超价。

  1. def on_tick(self, tick: TickData):
  2. """"""
  3. if self.vt_orderid:
  4. return
  5. if self.direction == Direction.LONG:
  6. if tick.last_price >= self.stop_price:
  7. price = self.stop_price + self.price_add
  8. if tick.limit_up:
  9. price = min(price, tick.limit_up)
  10. self.vt_orderid = self.buy(
  11. self.vt_symbol,
  12. price,
  13. self.volume,
  14. offset=self.offset
  15. )
  16. self.write_log(f"停止单已触发,代码:{self.vt_symbol},方向:{self.direction}, 价格:{self.stop_price},数量:{self.volume},开平:{self.offset}")
  17. else:
  18. if tick.last_price <= self.stop_price:
  19. price = self.stop_price - self.price_add
  20. if tick.limit_down:
  21. price = max(price, tick.limit_down)
  22. self.vt_orderid = self.buy(
  23. self.vt_symbol,
  24. price,
  25. self.volume,
  26. offset=self.offset
  27. )
  28. self.write_log(f"停止单已触发,代码:{self.vt_symbol},方向:{self.direction}, 价格:{self.stop_price},数量:{self.volume},开平:{self.offset}")
  29. self.put_variables_event()

最优限价算法

  • 监控最新tick推送的行情,发现好的价格立刻报价成交。

  • 买入情况:先检查撤单:最新Tick买一价不等于目标价格时,执行撤单;若无活动委托,发出委托:委托价格为最新Tick买一价,委托数量为剩余委托量。

  • 卖出情况:先检查撤单:最新Tick买一价不等于目标价格时,执行撤单;若无活动委托,发出委托:委托价格为最新Tick卖一价,委托数量为剩余委托量。

  1. def on_tick(self, tick: TickData):
  2. """"""
  3. self.last_tick = tick
  4. if self.direction == Direction.LONG:
  5. if not self.vt_orderid:
  6. self.buy_best_limit()
  7. elif self.order_price != self.last_tick.bid_price_1:
  8. self.cancel_all()
  9. else:
  10. if not self.vt_orderid:
  11. self.sell_best_limit()
  12. elif self.order_price != self.last_tick.ask_price_1:
  13. self.cancel_all()
  14. self.put_variables_event()
  15. def buy_best_limit(self):
  16. """"""
  17. order_volume = self.volume - self.traded
  18. self.order_price = self.last_tick.bid_price_1
  19. self.vt_orderid = self.buy(
  20. self.vt_symbol,
  21. self.order_price,
  22. order_volume,
  23. offset=self.offset
  24. )
  25. def sell_best_limit(self):
  26. """"""
  27. order_volume = self.volume - self.traded
  28. self.order_price = self.last_tick.ask_price_1
  29. self.vt_orderid = self.sell(
  30. self.vt_symbol,
  31. self.order_price,
  32. order_volume,
  33. offset=self.offset
  34. )

网格算法

  • 每隔一段时间检查委托情况,若有委托则先清空。

  • 基于用户设置的价格步进(即网格)计算目标距离,目标距离=(目标价格- 当前价格)/价格步进,故当前价格低于目标价格,目标距离为正,方向为买入;当前价格高于目标价格,目标距离为负,方向为卖出。(高抛低吸概念)

  • 计算目标仓位,目标仓位= 取整后的目标距离 * 委托数量步进。注意卖卖方向取整的方式是不同的:买入方向要向下取整math.floor(),如目标距离为1.6,取1;卖出方向要向上取整,如目标距离为-1.6,取-1。

  • 计算具体委托仓位:若目标买入仓位大于当前仓位,执行买入操作;若目标卖出仓位低于当前仓位,执行卖出操作。

  • 为了能够快速成交,买入情况是基于ask price计算,卖出情况是基于bid price计算。

  1. def on_timer(self):
  2. """"""
  3. if not self.last_tick:
  4. return
  5. self.timer_count += 1
  6. if self.timer_count < self.interval:
  7. self.put_variables_event()
  8. return
  9. self.timer_count = 0
  10. if self.vt_orderid:
  11. self.cancel_all()
  12. # Calculate target volume to buy
  13. target_buy_distance = (self.price - self.last_tick.ask_price_1) / self.step_price
  14. target_buy_position = math.floor(target_buy_distance) * self.step_volume
  15. target_buy_volume = target_buy_position - self.last_pos
  16. # Buy when price dropping
  17. if target_buy_volume > 0:
  18. self.vt_orderid = self.buy(
  19. self.vt_symbol,
  20. self.last_tick.ask_price_1,
  21. min(target_buy_volume, self.last_tick.ask_volume_1)
  22. )
  23. # Calculate target volume to sell
  24. target_sell_distance = (self.price - self.last_tick.bid_price_1) / self.step_price
  25. target_sell_position = math.ceil(target_sell_distance) * self.step_volume
  26. target_sell_volume = self.last_pos - target_sell_position
  27. # Sell when price rising
  28. if target_sell_volume > 0:
  29. self.vt_orderid = self.sell(
  30. self.vt_symbol,
  31. self.last_tick.bid_price_1,
  32. min(target_sell_volume, self.last_tick.bid_volume_1)
  33. )

套利算法

  • 每隔一段时间检查委托情况,若有委托则先清空;若主动腿还持有净持仓,通过被动腿成交来对冲。

  • 计算价差spread_bid_price 和 spread_ask_price, 以及对应的委托数量

  • 卖出情况:主动腿价格相对被动腿上涨,其价差spread_bid_price大于spread_up时,触发买入信号

  • 买入情况:主动腿价格相对被动腿下跌,其价差spread_ask_price小于 - spread_down(spread_down默认设置为正数)时,触发卖出信号

  • 在买卖信号判断加入最大持仓的限制,其作用是避免持仓过多导致保证金不足或者直接被交易所惩罚性强平;而且随着价差持续波动,主动腿持仓可以从0 -> 10 -> 0 -> -10 -> 0,从而实现平仓获利离场。

  1. def on_timer(self):
  2. """"""
  3. self.timer_count += 1
  4. if self.timer_count < self.interval:
  5. self.put_variables_event()
  6. return
  7. self.timer_count = 0
  8. if self.active_vt_orderid or self.passive_vt_orderid:
  9. self.cancel_all()
  10. return
  11. if self.net_pos:
  12. self.hedge()
  13. return
  14. active_tick = self.get_tick(self.active_vt_symbol)
  15. passive_tick = self.get_tick(self.passive_vt_symbol)
  16. if not active_tick or not passive_tick:
  17. return
  18. # Calculate spread
  19. spread_bid_price = active_tick.bid_price_1 - passive_tick.ask_price_1
  20. spread_ask_price = active_tick.ask_price_1 - passive_tick.bid_price_1
  21. spread_bid_volume = min(active_tick.bid_volume_1, passive_tick.ask_volume_1)
  22. spread_ask_volume = min(active_tick.ask_volume_1, passive_tick.bid_volume_1)
  23. # Sell condition
  24. if spread_bid_price > self.spread_up:
  25. if self.acum_pos <= -self.max_pos:
  26. return
  27. else:
  28. self.active_vt_orderid = self.sell(
  29. self.active_vt_symbol,
  30. active_tick.bid_price_1,
  31. spread_bid_volume
  32. )
  33. # Buy condition
  34. elif spread_ask_price < -self.spread_down:
  35. if self.acum_pos >= self.max_pos:
  36. return
  37. else:
  38. self.active_vt_orderid = self.buy(
  39. self.active_vt_symbol,
  40. active_tick.ask_price_1,
  41. spread_ask_volume
  42. )
  43. self.put_variables_event()
  44. def hedge(self):
  45. """"""
  46. tick = self.get_tick(self.passive_vt_symbol)
  47. volume = abs(self.net_pos)
  48. if self.net_pos > 0:
  49. self.passive_vt_orderid = self.sell(
  50. self.passive_vt_symbol,
  51. tick.bid_price_5,
  52. volume
  53. )
  54. elif self.net_pos < 0:
  55. self.passive_vt_orderid = self.buy(
  56. self.passive_vt_symbol,
  57. tick.ask_price_5,
  58. volume
  59. )
  60. def on_trade(self, trade: TradeData):
  61. """"""
  62. # Update net position volume
  63. if trade.direction == Direction.LONG:
  64. self.net_pos += trade.volume
  65. else:
  66. self.net_pos -= trade.volume
  67. # Update active symbol position
  68. if trade.vt_symbol == self.active_vt_symbol:
  69. if trade.direction == Direction.LONG:
  70. self.acum_pos += trade.volume
  71. else:
  72. self.acum_pos -= trade.volume
  73. # Hedge if active symbol traded
  74. if trade.vt_symbol == self.active_vt_symbol:
  75. self.hedge()
  76. self.put_variables_event()