t1_converter_v2.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. CYB50 T+1 回测引擎 - 正确的资金管理版本 V2
  5. 核心逻辑:
  6. 1. 按时间顺序处理所有信号
  7. 2. 维护持仓状态(是否持仓、持仓期间)
  8. 3. 当新信号的开仓时间落在已有持仓期间时,跳过该信号
  9. 4. 正确处理T+1延期平仓
  10. """
  11. import pandas as pd
  12. import numpy as np
  13. from datetime import datetime, timedelta
  14. import sys
  15. sys.path.insert(0, '/root/.openclaw/workspace/cat-fly')
  16. from cyb50_30min_dual_direction import (
  17. ConfigManager, IntradayDataFetcher,
  18. DualDirectionSignalGenerator, DualDirectionExecutor
  19. )
  20. def simulate_t1_trades_v2(data_df, long_trades_df, initial_capital=1000000):
  21. """
  22. 正确的T+1回测 V2 - 正确处理时间重叠
  23. 参数:
  24. data_df: 包含所有价格数据的DataFrame
  25. long_trades_df: 原始做多交易列表(作为信号源)
  26. initial_capital: 初始资金
  27. 返回:
  28. t1_trades_df: T+1规则下实际执行的交易
  29. """
  30. print("\n" + "="*80)
  31. print("T+1回测引擎 V2 - 正确处理时间重叠")
  32. print("="*80)
  33. if len(long_trades_df) == 0:
  34. print("没有做多交易记录")
  35. return pd.DataFrame()
  36. # 准备信号列表
  37. signals = []
  38. for idx, trade in long_trades_df.iterrows():
  39. signals.append({
  40. 'signal_id': idx + 1,
  41. 'entry_time': trade['开仓时间'],
  42. 'entry_price': trade['开仓价格'],
  43. 'exit_time': trade['平仓时间'],
  44. 'exit_price': trade['平仓价格'],
  45. 'exit_reason': trade['退出原因'],
  46. 'entry_signals': trade.get('入场信号', '')
  47. })
  48. # 按开仓时间排序
  49. signals = sorted(signals, key=lambda x: x['entry_time'])
  50. t1_trades = []
  51. capital = initial_capital
  52. # 当前持仓状态
  53. current_position = None # None 或 {'entry_time', 'entry_price', 'can_sell_time', ...}
  54. print(f"\n初始资金: {initial_capital:,.0f}元")
  55. print(f"总信号数: {len(signals)}笔")
  56. print("\n" + "-"*80)
  57. for signal in signals:
  58. sig_entry = signal['entry_time']
  59. sig_exit = signal['exit_time']
  60. # 检查是否有持仓
  61. if current_position is not None:
  62. # 检查新信号是否在持仓期间
  63. hold_end = current_position['actual_exit_time']
  64. if sig_entry < hold_end:
  65. # 新信号在持仓期间,跳过
  66. print(f"\n[跳过] 信号 #{signal['signal_id']}: {sig_entry.strftime('%m-%d %H:%M')}")
  67. print(f" 原因: 当前持仓中 (持仓期: {current_position['entry_time'].strftime('%m-%d %H:%M')} → {hold_end.strftime('%m-%d %H:%M')})")
  68. continue
  69. else:
  70. # 新信号在持仓结束后,先结算当前持仓
  71. print(f"\n[结算前持仓] 信号 #{signal['signal_id']} 到来时,前持仓已结束")
  72. # 持仓已经结束,可以开新仓
  73. current_position = None
  74. # 现在可以开仓
  75. entry_time = sig_entry
  76. entry_price = signal['entry_price']
  77. # 计算可卖时间(T+1规则)
  78. can_sell_time = entry_time + timedelta(days=1)
  79. can_sell_time = can_sell_time.replace(hour=9, minute=30)
  80. # 确定实际平仓时间
  81. # 原始平仓时间 vs T+1最早可卖时间,取较晚者
  82. actual_exit_time = max(sig_exit, can_sell_time)
  83. # 查找实际平仓价格
  84. if actual_exit_time > sig_exit:
  85. # 需要延期,查找数据
  86. future_data = data_df[data_df.index >= actual_exit_time]
  87. if len(future_data) > 0:
  88. actual_exit_time = future_data.index[0]
  89. actual_exit_price = future_data.iloc[0]['Open']
  90. t1_adjusted = True
  91. else:
  92. # 无后续数据,使用原始
  93. actual_exit_time = sig_exit
  94. actual_exit_price = signal['exit_price']
  95. t1_adjusted = False
  96. else:
  97. actual_exit_price = signal['exit_price']
  98. t1_adjusted = False
  99. # 计算仓位(使用全部资金)
  100. position_size = int(capital / entry_price)
  101. if position_size <= 0:
  102. print(f"\n[跳过] 信号 #{signal['signal_id']}: {entry_time.strftime('%m-%d %H:%M')} - 资金不足")
  103. continue
  104. # 记录持仓
  105. current_position = {
  106. 'entry_time': entry_time,
  107. 'entry_price': entry_price,
  108. 'position_size': position_size,
  109. 'actual_exit_time': actual_exit_time,
  110. 'actual_exit_price': actual_exit_price,
  111. 't1_adjusted': t1_adjusted,
  112. 'original_exit_time': sig_exit,
  113. 'original_exit_price': signal['exit_price'],
  114. 'exit_reason': signal['exit_reason'],
  115. 'entry_signals': signal['entry_signals']
  116. }
  117. # 计算盈亏
  118. gross_pnl = (actual_exit_price - entry_price) * position_size
  119. pnl_pct = (actual_exit_price - entry_price) / entry_price * 100
  120. # 更新资金
  121. capital += gross_pnl
  122. # 记录交易
  123. trade_record = {
  124. '交易方向': '做多',
  125. '开仓时间': entry_time,
  126. '平仓时间': actual_exit_time,
  127. '开仓价格': entry_price,
  128. '平仓价格': actual_exit_price,
  129. '仓位': position_size,
  130. '盈亏金额': gross_pnl,
  131. '盈亏百分比': pnl_pct,
  132. '退出原因': signal['exit_reason'] if not t1_adjusted else f"T+1延期-{signal['exit_reason']}",
  133. '持仓周期数': int((actual_exit_time - entry_time).total_seconds() / 1800),
  134. '持仓小时数': (actual_exit_time - entry_time).total_seconds() / 3600,
  135. 'T+1调整': '是(T0→T1)' if t1_adjusted else '否',
  136. '原平仓时间': sig_exit,
  137. '原平仓价格': signal['exit_price'],
  138. '入场信号': signal['entry_signals'],
  139. '平仓时资金': capital,
  140. }
  141. t1_trades.append(trade_record)
  142. # 打印
  143. status = "✅盈利" if gross_pnl > 0 else "❌亏损"
  144. adj_str = "[T+1调整] " if t1_adjusted else ""
  145. print(f"\n[执行] 信号 #{signal['signal_id']}: {entry_time.strftime('%m-%d %H:%M')} → {actual_exit_time.strftime('%m-%d %H:%M')}")
  146. print(f" {adj_str}价格: {entry_price:.2f} → {actual_exit_price:.2f}")
  147. print(f" 盈亏: {gross_pnl:+,.0f}元 ({pnl_pct:+.2f}%) {status}")
  148. print(f" 资金: {capital:,.0f}元")
  149. # 处理最后一笔持仓(如果数据结束前未平仓)
  150. if current_position is not None and current_position['actual_exit_time'] > data_df.index[-1]:
  151. final_price = data_df.iloc[-1]['Close']
  152. final_time = data_df.index[-1]
  153. entry_time = current_position['entry_time']
  154. entry_price = current_position['entry_price']
  155. position_size = current_position['position_size']
  156. gross_pnl = (final_price - entry_price) * position_size
  157. pnl_pct = (final_price - entry_price) / entry_price * 100
  158. capital += gross_pnl
  159. trade_record = {
  160. '交易方向': '做多',
  161. '开仓时间': entry_time,
  162. '平仓时间': final_time,
  163. '开仓价格': entry_price,
  164. '平仓价格': final_price,
  165. '仓位': position_size,
  166. '盈亏金额': gross_pnl,
  167. '盈亏百分比': pnl_pct,
  168. '退出原因': '回测强制平仓',
  169. '持仓周期数': int((final_time - entry_time).total_seconds() / 1800),
  170. '持仓小时数': (final_time - entry_time).total_seconds() / 3600,
  171. 'T+1调整': '否',
  172. '原平仓时间': final_time,
  173. '原平仓价格': final_price,
  174. '入场信号': current_position['entry_signals'],
  175. '平仓时资金': capital,
  176. }
  177. t1_trades.append(trade_record)
  178. print(f"\n[强制平仓] {final_time.strftime('%m-%d %H:%M')} @ {final_price:.2f}")
  179. print(f" 盈亏: {gross_pnl:+,.0f}元")
  180. # 生成结果
  181. t1_trades_df = pd.DataFrame(t1_trades)
  182. print("\n" + "="*80)
  183. print("T+1回测 V2 完成")
  184. print("="*80)
  185. print(f"原始信号数: {len(signals)}笔")
  186. print(f"实际执行: {len(t1_trades)}笔")
  187. print(f"跳过信号: {len(signals) - len(t1_trades)}笔")
  188. print(f"最终资金: {capital:,.0f}元")
  189. print(f"总收益率: {(capital/initial_capital-1)*100:+.2f}%")
  190. return t1_trades_df
  191. def main():
  192. """主程序 - 测试V2版本"""
  193. print("="*80)
  194. print("CYB50 T+1 回测引擎 V2")
  195. print("="*80)
  196. initial_capital = 1000000
  197. # 获取数据和原始交易
  198. print("\n【步骤1】获取数据...")
  199. config_manager = ConfigManager('config.json')
  200. fetcher = IntradayDataFetcher(config_manager)
  201. end_date = datetime.now()
  202. start_date = end_date - timedelta(days=70)
  203. raw_data = fetcher.fetch_30min_data(start_date, end_date)
  204. data_with_indicators = fetcher.calculate_intraday_indicators(raw_data)
  205. signal_generator = DualDirectionSignalGenerator()
  206. signals_df = signal_generator.generate_dual_direction_signals(data_with_indicators)
  207. executor = DualDirectionExecutor(initial_capital=initial_capital)
  208. results_df, trades_df = executor.execute_dual_direction_trades(signals_df)
  209. long_trades = trades_df[trades_df['交易方向'] == '做多'].copy()
  210. print(f"✅ 原始做多信号: {len(long_trades)}笔")
  211. # 运行V2回测
  212. print("\n【步骤2】运行T+1回测 V2...")
  213. t1_trades = simulate_t1_trades_v2(data_with_indicators, long_trades, initial_capital)
  214. # 对比
  215. print("\n【步骤3】对比...")
  216. orig_pnl = long_trades['盈亏金额'].sum()
  217. t1_pnl = t1_trades['盈亏金额'].sum() if len(t1_trades) > 0 else 0
  218. print(f"原始交易: {len(long_trades)}笔, 盈亏 {orig_pnl:+,.0f}元")
  219. print(f"T+1交易: {len(t1_trades)}笔, 盈亏 {t1_pnl:+,.0f}元")
  220. print(f"差异: {t1_pnl - orig_pnl:+,.0f}元")
  221. # 检查时间重叠
  222. if len(t1_trades) > 0:
  223. print("\n【步骤4】验证时间无重叠...")
  224. prev_exit = None
  225. overlap = 0
  226. for _, row in t1_trades.iterrows():
  227. if prev_exit is not None and row['开仓时间'] < prev_exit:
  228. overlap += 1
  229. print(f" ⚠️ 重叠: {row['开仓时间'].strftime('%m-%d %H:%M')} 早于前笔平仓 {prev_exit.strftime('%m-%d %H:%M')}")
  230. prev_exit = row['平仓时间']
  231. if overlap == 0:
  232. print(" ✅ 无时间重叠")
  233. else:
  234. print(f" ❌ 发现 {overlap} 处重叠")
  235. # 保存
  236. if len(t1_trades) > 0:
  237. timestamp = datetime.now().strftime('%Y%m%d_%H%M%S')
  238. output_file = f'cyb50_t1_v2_{timestamp}.csv'
  239. export_df = t1_trades.copy()
  240. for col in ['开仓时间', '平仓时间']:
  241. export_df[col] = export_df[col].dt.strftime('%Y-%m-%d %H:%M:%S')
  242. export_df.to_csv(output_file, index=False, encoding='utf-8-sig')
  243. print(f"\n✅ 已保存: {output_file}")
  244. print("\n" + "="*80)
  245. print("完成!")
  246. print("="*80)
  247. if __name__ == "__main__":
  248. main()