策略编写指南¶
本文档旨在帮助策略开发者快速掌握 AKQuant 的策略编写方法。
1. 核心概念 (Glossary)¶
对于量化交易的新手,这里有一些基础术语的解释:
- Bar (K线): 包含了某一段时间(如1分钟、1天)内的市场行情,主要包含 5 个数据:
- Open: 开盘价
- High: 最高价
- Low: 最低价
- Close: 收盘价
- Volume: 成交量
- Strategy (策略): 你的交易机器人。它的核心工作就是不断地看行情 (on_bar),然后决定买 (buy) 还是卖 (sell)。
- Context (上下文): 机器人的“记事本”和“工具箱”。里面记录了当前有多少钱 (cash)、有多少股票 (positions),也提供了下单的工具。
- Position (持仓): 你当前持有的股票或期货数量。正数表示多头(买入持有),负数表示空头(借券卖出)。
- Backtest (回测): 历史模拟。用过去的数据来测试你的策略,看看如果过去这么做,能赚多少钱。
2. 策略生命周期¶
一个策略从开始到结束,会经历以下几个阶段:
__init__: Python 对象初始化,适合定义参数。on_start: 策略启动时调用,必须在此处使用self.subscribe()订阅数据,也可在此注册指标。如果是热启动,需注意不要覆盖已恢复的状态。on_resume: 仅在热启动时调用(在on_start之前)。用于处理从快照恢复后的特殊逻辑。on_bar: 每一根 K 线闭合时触发 (核心交易逻辑)。on_tick: 每一个 Tick 到达时触发 (高频/盘口策略)。on_order: 订单状态变化时触发 (如提交、成交、取消)。on_trade: 收到成交回报时触发。on_reject: 订单进入Rejected状态时触发。on_session_start/on_session_end: 会话切换时触发。before_trading/after_trading: 交易日级钩子。on_daily_rebalance: 交易日调仓钩子(每天最多一次,适合横截面轮动)。on_portfolio_update: 账户快照变化时触发。on_error: 用户回调抛异常时触发,默认触发后继续抛出异常。on_timer: 定时器触发时调用 (需手动注册)。on_stop: 策略停止时调用,适合进行资源清理或结果统计。on_train_signal: 滚动训练触发信号 (仅在 ML 模式下触发)。
2.1 回调触发契约¶
对于每个 bar/tick/timer 事件,框架按以下顺序分发回调:
on_order/on_trade(若拒单则额外触发on_reject)- 框架钩子(
on_session_*、before_trading/after_trading、on_portfolio_update) - 用户事件回调(
on_bar/on_tick/on_timer)
说明:
on_reject对同一订单 id 只触发一次。- 回测中已终态拒单会通过上下文快照
recent_rejected_orders在下一次事件分发时补发,避免因清理活跃订单导致漏触发。 before_trading在本地交易日首次进入 Normal 会话时触发一次。on_daily_rebalance与before_trading同一阶段触发,每个交易日最多触发一次。after_trading在离开 Normal 会话时触发;若先跨日再收到事件,会在下一事件补发上一交易日的after_trading。- 若需要更精确的交易日边界触发,可在策略中设置
self.enable_precise_day_boundary_hooks = True。 on_portfolio_update采用增量触发:初始化时触发一次,后续仅在订单/成交或持仓相关价格变化时触发。- 可通过
self.portfolio_update_eps过滤微小资产波动(默认0.0,即不过滤)。 - 停止阶段会在
on_stop之前补发待触发的on_session_end/after_trading。 on_error参数为(error, source, payload),推荐通过self.error_mode = "raise" | "continue"控制行为(默认raise)。self.re_raise_on_error仍兼容,作为兜底开关。- 推荐使用
self.runtime_config = StrategyRuntimeConfig(...)统一配置上述行为开关。 - 旧别名字段与
runtime_config会自动保持同步。
3. 风险管理 (Risk Management)¶
AKQuant 内置了强大的预交易风控模块,支持在 Engine 层面拦截不合规的订单。
你可以在回测脚本中(Engine 初始化后)配置这些规则:
from akquant import Engine
engine = Engine()
# ... 添加数据 ...
# 获取风控管理器 (注意:PyO3 返回的是副本,修改后需赋值回去)
rm = engine.risk_manager
# 1. 单标的持仓上限 (例如 10%)
# 如果买入导致某标的持仓市值占总权益超过 10%,则拒绝订单
rm.add_max_position_percent_rule(0.10)
# 2. 行业集中度限制 (例如科技股不超过 20%)
# 需要提供 Symbol -> Sector 的映射字典
sector_map = {"AAPL": "Tech", "MSFT": "Tech", "XOM": "Energy"}
rm.add_sector_concentration_rule(0.20, sector_map)
# 3. 总杠杆率熔断 (例如 1.5倍)
# 总敞口 / 总权益 > 1.5 时拒绝开仓
# 对于高杠杆策略,建议关闭默认的现金检查 (check_cash=False)
rm.config.check_cash = False
rm.add_max_leverage_rule(1.5)
# 4. 账户最大回撤限制 (例如 20%)
# 当前权益相对历史峰值回撤超过阈值时,拒绝新订单
rm.add_max_drawdown_rule(0.20)
# 5. 单日亏损限制 (例如 5%)
# 当日权益相对当日首个风控检查时点下跌超过阈值时,拒绝新订单
rm.add_max_daily_loss_rule(0.05)
# 6. 账户净值止损阈值 (例如 80%)
# 当前权益低于“规则首次生效时权益 * 阈值”时,拒绝新订单
rm.add_stop_loss_rule(0.80)
# 应用配置
engine.risk_manager = rm
# 运行回测
engine.run(strategy=MyStrategy)
通过 run_backtest 统一入口也可直接配置账户级风控:
from akquant import run_backtest
from akquant.config import RiskConfig
result = run_backtest(
data=data,
strategy=MyStrategy,
risk_config=RiskConfig(
max_account_drawdown=0.20,
max_daily_loss=0.05,
stop_loss_threshold=0.80,
),
)
账户级参数建议(可作为起步值):
| 风格 | max_account_drawdown |
max_daily_loss |
stop_loss_threshold |
|---|---|---|---|
| 保守 | 0.10 |
0.02 |
0.90 |
| 中性 | 0.20 |
0.05 |
0.80 |
| 激进 | 0.30 |
0.08 |
0.70 |
建议先从“中性”起步,再根据策略波动与换手逐步收紧或放宽。
3.1 信用账户回测(融资/融券)¶
若策略需要在回测中启用融资买入或融券卖出,可在 RiskConfig 中显式切换到账户模式 margin:
from akquant.config import RiskConfig
risk_config = RiskConfig(
account_mode="margin",
enable_short_sell=True,
initial_margin_ratio=0.5,
maintenance_margin_ratio=0.3,
financing_rate_annual=0.08,
borrow_rate_annual=0.10,
allow_force_liquidation=True,
liquidation_priority="short_first",
)
常用字段说明:
account_mode:"cash"/"margin"。enable_short_sell:True时允许股票开空。initial_margin_ratio: 初始保证金比例(影响可开仓规模)。maintenance_margin_ratio: 维持担保比例阈值。allow_force_liquidation: 维持担保比例跌破阈值时是否触发强平。liquidation_priority:"short_first"或"long_first"。
策略中可通过 get_account() 读取信用账户扩展字段:
snap = self.get_account()
print(
snap["account_mode"],
snap["borrowed_cash"],
snap["short_market_value"],
snap["maintenance_ratio"],
snap["accrued_interest"],
snap["daily_interest"],
)
4. 常用工具 (Utilities)¶
AKQuant 提供了一系列便捷工具来简化策略开发。
3.1 日志记录 (Logging)¶
使用 self.log() 可以输出带有当前回测时间戳的日志,方便调试和记录。
def on_bar(self, bar):
# 自动添加时间戳,例如: [2023-01-01 09:30:00] 信号触发: 买入
self.log("信号触发: 买入")
# 支持指定日志级别
import logging
self.log("资金不足", level=logging.WARNING)
3.2 便捷数据访问 (Data Access)¶
为了减少代码冗余,Strategy 类提供了当前 Bar/Tick 数据的快捷访问属性:
| 属性 | 说明 | 对应原始代码 |
|---|---|---|
self.symbol |
当前标的代码 | bar.symbol / tick.symbol |
self.close |
当前最新价 | bar.close / tick.price |
self.open |
当前开盘价 | bar.open (Tick 模式为 0) |
self.high |
当前最高价 | bar.high (Tick 模式为 0) |
self.low |
当前最低价 | bar.low (Tick 模式为 0) |
self.volume |
当前成交量 | bar.volume / tick.volume |
示例:
def on_bar(self, bar):
# 旧写法
if bar.close > bar.open: ...
# 新写法 (更简洁)
if self.close > self.open:
self.buy(self.symbol, 100)
3.3 定时器 (Timer)¶
除了底层的 schedule 方法,AKQuant 提供了更便捷的定时任务注册方式:
add_daily_timer(time_str, payload): 每天在指定时间触发。- 支持实盘: 在回测模式下预生成所有触发时间;在实盘模式下,每日自动调度下一次触发。
schedule(trigger_time, payload): 在指定时间点(一次性)触发。
def on_start(self):
# 每天 14:55:00 触发收盘检查
self.add_daily_timer("14:55:00", "daily_check")
# 在特定日期时间触发
self.schedule("2023-01-01 09:30:00", "special_event")
def on_timer(self, payload):
if payload == "daily_check":
self.log("Running daily check...")
3.4 横截面策略推荐范式 (Cross-Section Pattern)¶
AKQuant 的 on_bar 按“单事件流”逐条触发。若你要做多标的横截面比较(轮动、排序、打分),推荐优先使用 on_daily_rebalance,由框架保证“每天最多一次”的触发语义。
推荐步骤:
- 在
on_start中定义universe并订阅标的。 - 在
on_daily_rebalance中遍历universe计算分数。 - 在
on_daily_rebalance中统一选股与调仓。
class CrossSectionStrategy(Strategy):
def __init__(self, lookback=20):
self.lookback = lookback
self.universe = ["sh600519", "sz000858", "sh601318"]
self.warmup_period = lookback + 1
def on_start(self):
for symbol in self.universe:
self.subscribe(symbol)
def on_daily_rebalance(self, trading_date, timestamp):
history_map = self.get_history_map(
count=self.lookback,
symbols=self.universe,
field="close",
)
scores = {}
for symbol, closes in history_map.items():
if len(closes) < self.lookback:
continue
scores[symbol] = (closes[-1] - closes[0]) / closes[0]
if not scores:
return
self.rebalance_to_topn(
scores=scores,
top_n=2,
weight_mode="score",
long_only=False,
)
完整示例见:examples/strategies/05_stock_momentum_rotation_timer.py(on_daily_rebalance)与 examples/strategies/07_stock_momentum_rotation_on_timer.py(on_timer 固定时点)。
3.5 横截面方案 B:收齐同 timestamp 后执行¶
当策略没有固定调仓时点(不方便用 on_timer)时,可在 on_bar 中先缓存同一时间片的标的,收齐后再执行一次横截面逻辑。
from collections import defaultdict
class CrossSectionBucketStrategy(Strategy):
def __init__(self, lookback=20):
self.lookback = lookback
self.universe = ["sh600519", "sz000858", "sh601318"]
self.warmup_period = lookback + 1
self.pending = defaultdict(set)
def on_bar(self, bar):
self.pending[bar.timestamp].add(bar.symbol)
if len(self.pending[bar.timestamp]) < len(self.universe):
return
self.pending.pop(bar.timestamp, None)
scores = {}
for symbol in self.universe:
closes = self.get_history(count=self.lookback, symbol=symbol, field="close")
if len(closes) < self.lookback:
return
scores[symbol] = (closes[-1] - closes[0]) / closes[0]
best = max(scores, key=lambda s: scores[s])
self.order_target_percent(target_percent=0.95, symbol=best)
完整示例见:examples/strategies/06_stock_momentum_rotation_bucket.py。
3.6 方案选型对照 (A vs B)¶
| 维度 | 方案 A:on_timer 统一执行 |
方案 B:收齐 timestamp 后执行 |
|---|---|---|
| 触发方式 | 固定时点触发(如 14:55) | 事件驱动,时间片收齐触发 |
| 稳健性 | 高,不依赖到达顺序 | 中,需维护缓存并处理缺失 |
| 实现复杂度 | 低,逻辑集中 | 中,需管理 timestamp -> symbols |
| 适用场景 | 日频/定时调仓、生产默认 | 无固定调仓时点的横截面策略 |
| 常见风险 | 定时器时间与数据频率不匹配 | 某些标的缺失导致不触发 |
建议:优先使用方案 A;只有在无法定义稳定调仓时点时再采用方案 B。
3.7 横截面常见坑位清单¶
- 停牌/缺失数据:某些标的当日无 Bar 时,方案 B 可能不触发;可设置超时降级,或允许“有效样本数达阈值”即执行。
- Universe 漂移:成分股调整后若仍用旧列表,会出现权重与真实池不一致;建议定期刷新并记录生效日期。
- 调仓时点与成交策略错配:例如
fill_policy={"price_basis":"open","bar_offset":1}时,收盘时点信号会在下一根撮合;应结合fill_policy.temporal明确 timer 是否当期成交。 - 历史长度不足:新上市或停牌恢复标的数据窗口不完整;评分前统一做
len(closes)检查并跳过不足样本。 - 仓位未收敛:多标的先卖后买若资金未及时释放,可能导致买入不足;可采用目标仓位 API 并在下一时点二次收敛。
完整上线检查可参考:横截面策略实战清单。
5. 策略风格选择¶
AKQuant 提供了两种风格的策略开发接口:
风格选择建议可参考:策略风格决策指南。
| 特性 | 类风格 (推荐) | 函数风格 |
|---|---|---|
| 定义方式 | 继承 akquant.Strategy |
定义 initialize + on_bar(必选),可选 on_start / on_stop / on_tick / on_order / on_trade / on_timer |
| 适用场景 | 复杂策略、需要维护内部状态、生产环境 | 快速原型验证、迁移 Zipline/Backtrader 策略 |
| 代码结构 | 面向对象,逻辑封装性好 | 脚本化,简单直观 |
| API 调用 | self.buy(), self.ctx |
ctx.buy(), ctx 作为参数传递 |
5.1 函数式回调触发前提¶
| 回调 | 触发前提 | 说明 |
|---|---|---|
on_bar(ctx, bar) |
回测数据流产生 Bar 事件 | 函数式策略的必选主回调 |
on_start(ctx) |
回测启动时触发 | 对齐类策略 on_start 生命周期 |
on_stop(ctx) |
回测结束时触发 | 对齐类策略 on_stop 生命周期 |
on_tick(ctx, tick) |
回测数据流产生 Tick 事件 | 仅 Bar 数据集不会触发 Tick 回调 |
on_order(ctx, order) |
策略上下文中观察到订单状态变化 | 每轮事件循环中先于主事件回调触发 |
on_trade(ctx, trade) |
recent_trades 中出现成交回报 |
框架会进行成交去重,避免重复触发 |
on_timer(ctx, payload) |
已注册的定时器到点触发 | 支持单次定时与每日定时 payload |
5.2 相关示例¶
- 函数式回调基础示例:
examples/23_functional_callbacks_demo.py - 函数式 Tick 回调模拟示例:
examples/24_functional_tick_simulation_demo.py - LiveRunner 支持函数式入口与多 slot 编排:
LiveRunner(strategy_cls=on_bar, strategy_id="alpha", strategies_by_slot={"beta": OtherStrategy}, initialize=..., on_tick=..., on_order=..., on_trade=..., on_timer=...) - 回测多 slot 与策略级风控映射建议使用集中式
BacktestConfig(strategy_config=StrategyConfig(...)):docs/zh/advanced/multi_strategy_guide.md - broker_live 函数式下单示例:
examples/39_live_broker_submit_order_demo.py - broker_live 默认执行语义为
strict,可通过gateway_options={"execution_semantics_mode": "strict"}显式声明 - 函数式多策略 slot + 风控示例:
examples/40_functional_multi_slot_risk_demo.py - LiveRunner 多策略 slot 编排示例:
examples/41_live_multi_slot_orchestration_demo.py - 运行后可分别观察输出标记:
done_functional_callbacks_demodone_functional_tick_simulation_demo
6. 编写类风格策略 (Class-based)¶
这是 AKQuant 推荐的策略编写方式,结构清晰,易于扩展。
6.1 数据预热 (Warmup Period)¶
在计算技术指标(如 MA, RSI)时,需要一定长度的历史数据。AKQuant 提供了 warmup_period 机制来自动处理数据预加载。
- 静态设置 (推荐): 在类中定义
warmup_period = N。 - 动态设置: 在
__init__中设置self.warmup_period = N。 - 自动推断: 如果使用内置指标,框架会尝试自动计算所需长度(但显式设置更安全)。
6.2 历史数据获取¶
self.get_history(count, ...): 返回numpy.ndarray,性能最高,适合计算指标。self.get_history_df(count, ...): 返回pandas.DataFrame,包含 OHLCV,适合复杂分析。
6.3 完整示例¶
from akquant import Strategy, Bar
import numpy as np
class MyStrategy(Strategy):
# 声明需要的预热数据长度 (例如 20日均线需要至少 20 根 Bar)
warmup_period = 20
def __init__(self, ma_window=20):
# 注意: Strategy 类使用了 __new__ 进行初始化,子类不再需要调用 super().__init__()
self.ma_window = ma_window
# 如果参数影响预热长度,可以动态覆盖
self.warmup_period = ma_window + 5
def on_start(self):
# 显式订阅数据
self.subscribe("600000")
def on_bar(self, bar: Bar):
# 1. 获取历史数据
# 返回 numpy array: [close_t-N, ..., close_t-1, close_t]
history = self.get_history(count=self.ma_window, symbol=bar.symbol, field="close")
# 检查数据是否足够 (虽然 warmup_period 会保证,但防御性编程是好习惯)
if len(history) < self.ma_window:
return
# 计算均线
ma_value = np.mean(history)
# 2. 交易逻辑
# 获取当前持仓 (使用 Position Helper 或 get_position)
pos = self.get_position(bar.symbol)
if bar.close > ma_value and pos == 0:
self.buy(symbol=bar.symbol, quantity=100)
elif bar.close < ma_value and pos > 0:
self.sell(symbol=bar.symbol, quantity=100)
7. 订单与交易详解 (Orders & Execution)¶
7.1 订单生命周期¶
在 AKQuant 中,订单状态流转如下:
- New: 订单对象被创建。
- Submitted: 订单已发送给交易所/仿真撮合引擎。
- Accepted: (实盘模式) 交易所确认接收订单。
- Filled: 订单全部成交。
- PartiallyFilled: 订单部分成交(
filled_quantity < quantity)。
- PartiallyFilled: 订单部分成交(
- Cancelled: 订单已取消。
- Rejected: 订单被风控或交易所拒绝 (如资金不足、超出涨跌停)。
broker_live + CTP 默认使用严格执行语义:终态以 OnRtnOrder 为准。也就是说,发送撤单请求后,只有收到 OnRtnOrder(Cancelled) 才会进入 Cancelled;收到错误回报后,也需要后续订单回报来最终确认 Rejected。
7.2 常用交易指令¶
-
市价单 (Market Order):
self.buy(symbol, quantity)self.sell(symbol, quantity)- 以当前市场最优价格立即成交,保证成交速度,不保证价格。
-
限价单 (Limit Order):
self.buy(symbol, quantity, price=10.5)- 只有当市场价格 <= 10.5 时才买入。
-
目标仓位 (Target Order):
self.order_target(target=100, symbol="AAPL"): 调整持仓数量至 100 股。self.order_target_percent(target_percent=0.5, symbol="AAPL"): 调整持仓至总资产的 50%。self.order_target_value(target_value=10000, symbol="AAPL"): 调整持仓至 10000 元市值。self.order_target_weights(target_weights={"AAPL":0.4,"MSFT":0.3}, liquidate_unmentioned=True, rebalance_tolerance=0.01): 按多标的权重统一调仓。- 默认权重和不超过
1.0,如需超过请设置allow_leverage=True。 - 执行顺序为先卖后买,减少现金占用导致的买入失败。
- 默认权重和不超过
def on_timer(self, payload: str):
weights = {"sh600519": 0.35, "sz000858": 0.25, "sh601318": 0.20}
self.order_target_weights(
target_weights=weights,
liquidate_unmentioned=True,
rebalance_tolerance=0.01,
)
- 撤单 (Cancel Order):
self.cancel_order(order_id): 撤销指定订单。self.cancel_all_orders(): 撤销当前所有未成交订单。
7.3 OCO 与 Bracket 助手¶
AKQuant 提供了两组交易助手,减少策略中手写订单联动逻辑:
self.create_oco_order_group(first_order_id, second_order_id, group_id=None)- 把两个订单绑定为 OCO(One-Cancels-the-Other)。
- 任一订单成交后,另一订单会自动撤销。
self.place_bracket_order(symbol, quantity, entry_price=None, stop_trigger_price=None, take_profit_price=None, ...)- 一次性提交 Bracket 结构。
- 进场单成交后,自动挂出止损/止盈;当止损与止盈同时存在时自动绑定 OCO。
from akquant import OrderStatus, Strategy
class BracketHelperStrategy(Strategy):
def __init__(self):
self.entry_order_id = ""
def on_bar(self, bar):
if self.get_position(bar.symbol) > 0 or self.entry_order_id:
return
self.entry_order_id = self.place_bracket_order(
symbol=bar.symbol,
quantity=100,
stop_trigger_price=bar.close * 0.98,
take_profit_price=bar.close * 1.04,
entry_tag="entry",
stop_tag="stop",
take_profit_tag="take",
)
def on_order(self, order):
if order.id == self.entry_order_id and order.status in (
OrderStatus.Cancelled,
OrderStatus.Rejected,
):
self.entry_order_id = ""
7.4 Trailing Stop 助手¶
如果你需要在策略里直接表达“随价格移动的止损线”,可以使用以下助手:
self.place_trailing_stop(symbol, quantity, trail_offset, side="Sell", trail_reference_price=None, ...)- 触发后按市价执行(
StopTrail -> Market)。
- 触发后按市价执行(
self.place_trailing_stop_limit(symbol, quantity, price, trail_offset, side="Sell", trail_reference_price=None, ...)- 触发后按限价执行(
StopTrailLimit -> Limit)。
- 触发后按限价执行(
from akquant import Strategy
class TrailingHelperStrategy(Strategy):
def __init__(self):
self.trailing_order_id = ""
def on_bar(self, bar):
if self.get_position(bar.symbol) == 0:
self.buy(bar.symbol, 100)
self.trailing_order_id = self.place_trailing_stop(
symbol=bar.symbol,
quantity=100,
trail_offset=1.5,
side="Sell",
trail_reference_price=bar.close,
tag="trail-stop",
)
完整可运行脚本见:examples/36_trailing_orders.py。
6.3 市场规则与 T+1 (Market Rules)¶
在 A 股市场回测中,T+1 交易规则是一个非常重要的限制:当天买入的股票,第二个交易日才能卖出。
启用 T+1¶
默认情况下,AKQuant 使用 T+0 规则(便于美股或期货回测)。如需启用 T+1,请在 run_backtest 中设置:
# 启用 T+1 规则 (适用于 A 股)
akquant.run_backtest(
...,
t_plus_one=True,
commission_rate=0.0003,
stamp_tax_rate=0.001 # 配合印花税设置
)
对策略逻辑的影响¶
启用 T+1 后,你需要区分总持仓和可用持仓:
self.get_position(symbol): 返回总持仓(包含今日买入未解锁的部分)。self.ctx.get_available_position(symbol): 返回可用持仓(即今日可卖出的数量)。 > 推荐使用Position辅助类: >
示例代码:
def on_bar(self, bar: Bar):
# 使用 Position Helper
pos = self.position
# 卖出逻辑:必须检查可用持仓
if signal_sell and pos.available > 0:
self.sell(bar.symbol, pos.available)
注意:如果你在 T+1 模式下尝试卖出超过
available的数量,订单会被风控模块(Risk Manager)拒绝 (Rejected),并提示 "Insufficient available position"。
6.4 账户与持仓查询¶
除了 get_position,你还可以查询更多账户信息:
self.ctx.cash: 当前账户可用资金。self.ctx.equity: 当前账户总权益(现金 + 持仓市值)。self.get_trades(): 获取历史所有已平仓交易记录(Closed Trades)。self.get_open_orders(): 获取当前未成交订单。
on_trade 与 get_trades() 的语义不同:
on_trade(self, trade)接收的是当前事件步内的增量成交回报(适合做实时响应)。self.get_trades()返回的是累计“已平仓”交易(Closed Trades),未平仓时不会出现在这个列表里。
推荐模式:
class MyStrategy(Strategy):
def __init__(self):
self.recent_exec_count = 0
def on_trade(self, trade):
self.recent_exec_count += 1
print("incremental trade:", trade.order_id, trade.symbol, trade.quantity)
def on_stop(self):
closed = self.get_trades()
print("closed trades:", len(closed))
6.5 标的静态属性查询(推荐)¶
当你需要在策略中读取行权价、到期日、合约乘数、期权类型、标的代码等静态属性时,优先使用策略 API,而不是依赖 bar.extra。
可用接口:
self.get_instrument(symbol): 返回InstrumentSnapshot。self.get_instrument_field(symbol, field): 返回单字段值。self.get_instrument_config(symbol, fields=None): 兼容接口;支持单字段或多字段批量读取。self.get_instruments(symbols=None): 返回多个标的的快照字典。
这些接口在 on_start 即可使用(回测启动阶段已注入快照)。
import akquant
from akquant import Bar, Strategy
class MetaAwareStrategy(Strategy):
def on_start(self):
self.subscribe("OPTION_A")
expiry = self.get_instrument_field("OPTION_A", "expiry_date")
strike = self.get_instrument_field("OPTION_A", "strike_price")
print("meta:", expiry, strike)
def on_bar(self, bar: Bar):
meta = self.get_instrument_config(
bar.symbol, fields=["asset_type", "option_type", "multiplier"]
)
if meta["asset_type"] == "OPTION" and meta["option_type"] == "CALL":
pass
7. 进阶功能¶
7.1 事件回调¶
除了 on_bar,你还可以重写其他回调函数来处理更精细的逻辑:
on_order(self, order): 订单状态更新时触发。on_trade(self, trade): 订单成交时触发。
7.2 指标 (Indicators)¶
AKQuant 采用“平台双主流、策略单主流”模式。每个策略需要显式设置 indicator_mode,并使用对应注册接口:
indicator_mode="precompute"+register_precomputed_indicator(...)indicator_mode="incremental"+register_incremental_indicator(...)
from akquant import Bar, SMA, Strategy
class IndicatorStrategy(Strategy):
def __init__(self):
self.indicator_mode = "precompute"
self.sma20 = SMA(20)
self.register_precomputed_indicator("sma20", self.sma20)
def on_start(self):
self.subscribe("AAPL")
def on_bar(self, bar: Bar):
val = self.sma20.get_value(bar.symbol, bar.timestamp)
if bar.close > val:
self.buy(bar.symbol, 100)
from akquant import Bar, SMA, Strategy
class IncrementalIndicatorStrategy(Strategy):
def __init__(self):
self.indicator_mode = "incremental"
self.sma20 = SMA(20)
self.register_incremental_indicator(
"sma20",
self.sma20,
source="close",
symbols=["AAPL"],
)
def on_bar(self, bar: Bar):
if bar.symbol != "AAPL":
return
val = self.sma20.value
if val is None:
return
if bar.close > val:
self.buy(bar.symbol, 100)
## 8. 高级特性:热启动 (Warm Start)
AKQuant 支持**热启动 (Warm Start)** 功能,允许你保存回测状态并在未来恢复。这对于长周期分段回测、滚动训练或模拟实盘环境非常有用。
### 8.1 核心机制
* **保存快照**: 使用 `save_snapshot` 将引擎状态保存到文件。
* **恢复运行**: 使用 `run_warm_start` 从快照恢复并继续运行。
### 8.2 策略适配
为了支持热启动,策略类提供了 `on_resume` 生命周期钩子和 `is_restored` 属性。
* **`on_resume()`**: 仅在从快照恢复时调用(在 `on_start` 之前)。
* **`self.is_restored`**: 布尔值,指示当前策略实例是否是从快照恢复的。
**示例代码**:
```python
def on_start(self):
# 1. 初始化指标 (仅在冷启动时)
if not self.is_restored:
self.sma = SMA(30)
else:
self.log("Resumed from snapshot. Indicators retained.")
# 2. 注册指标 (必须执行)
self.register_indicator("sma", self.sma)
# 3. 订阅行情 (必须执行)
self.subscribe(self.symbol)
更多详细信息,请参阅 热启动指南。 ```