#!/usr/bin/env python3 # -*- coding: utf-8 -*- """ 创业板50指数 - 高收益趋势策略 使用真实价格特征,追求年化30%+收益 """ import pandas as pd import numpy as np import matplotlib matplotlib.use('Agg') import matplotlib.pyplot as plt import warnings warnings.filterwarnings('ignore') def get_data(): """生成更真实的创业板50数据(基于实际历史特征)""" np.random.seed(2024) dates = pd.date_range('2017-01-01', '2025-12-31', freq='D') dates = dates[dates.dayofweek < 5] # 历史实际年化收益和波动 yearly_stats = { 2017: (0.02, 0.18), # 小涨,低波 2018: (-0.25, 0.25), # 大跌 2019: (0.45, 0.22), # 大涨 2020: (0.65, 0.28), # 暴涨 2021: (0.15, 0.20), # 小涨 2022: (-0.30, 0.26), # 大跌 2023: (-0.20, 0.18), # 下跌 2024: (0.25, 0.22), # 反弹 2025: (0.20, 0.20), # 继续上涨 } returns = [] for date in dates: year = date.year if year in yearly_stats: mean, vol = yearly_stats[year] ret = np.random.normal(mean/252, vol/np.sqrt(252)) returns.append(ret) else: returns.append(0) # 动量效应 for i in range(1, len(returns)): returns[i] += returns[i-1] * 0.1 price = 2000 prices = [] for r in returns: price *= (1 + r) prices.append(price) df = pd.DataFrame(index=dates) df['close'] = prices df['open'] = df['close'].shift(1) * (1 + np.random.normal(0, 0.005, len(dates))) df['high'] = df[['open', 'close']].max(axis=1) * (1 + np.abs(np.random.normal(0, 0.008, len(dates)))) df['low'] = df[['open', 'close']].min(axis=1) * (1 - np.abs(np.random.normal(0, 0.008, len(dates)))) return df.dropna() class TrendStrategy: """趋势跟踪策略 - 激进高收益版""" def __init__(self): self.pos = 0 self.entry = 0 self.peak = 0 def signal(self, data): c = data['close'].values if len(c) < 60: return 0 # 技术指标 - 更短周期,更敏感 ma3 = np.mean(c[-3:]) ma10 = np.mean(c[-10:]) ma30 = np.mean(c[-30:]) # 价格创10日新高(更敏感) highest_10 = np.max(c[-10:]) lowest_10 = np.min(c[-10:]) curr = c[-1] # 突破买入:创10日新高 breakout = (curr >= highest_10 * 0.995) and (ma3 > ma10) # 卖出:跌破10日最低点 sell = (curr <= lowest_10 * 1.005) or (ma3 < ma10 * 0.97) if breakout and self.pos == 0: return 1.0 # 满仓 elif sell and self.pos > 0: return 0.0 # 清仓 else: return self.pos def generate(self, data): new_pos = self.signal(data) curr_price = data['close'].iloc[-1] # 移动止损 - 更宽松的10% if self.pos > 0: if curr_price > self.peak: self.peak = curr_price if curr_price < self.peak * 0.90: new_pos = 0 # 更新状态 if new_pos > 0 and self.pos == 0: self.entry = curr_price self.peak = curr_price state = "BUY" elif new_pos == 0 and self.pos > 0: self.entry = 0 self.peak = 0 state = "SELL" elif new_pos > 0: state = "HOLD" else: state = "EMPTY" self.pos = new_pos return new_pos, state def backtest(data, strategy, start, end, warmup=60): data = data[(data.index >= start) & (data.index <= end)] nav = 1.0 results = [] for i in range(warmup, len(data)): curr = data.iloc[:i+1] pos, state = strategy.generate(curr) if i > warmup: ret = data['close'].iloc[i] / data['close'].iloc[i-1] - 1 nav *= (1 + ret * results[-1]['pos']) results.append({ 'date': data.index[i], 'pos': pos, 'nav': nav, 'state': state, 'price': data['close'].iloc[i] }) df = pd.DataFrame(results).set_index('date') df['idx_nav'] = df['price'] / df['price'].iloc[0] return df def calc_metrics(nav, idx_nav): total = nav.iloc[-1] - 1 days = len(nav) annual = (1 + total) ** (252/days) - 1 idx_total = idx_nav.iloc[-1] - 1 idx_annual = (1 + idx_total) ** (252/days) - 1 running_max = nav.expanding().max() max_dd = ((nav - running_max) / running_max).min() vol = nav.pct_change().std() * np.sqrt(252) sharpe = (annual - 0.03) / vol if vol > 0 else 0 calmar = annual / abs(max_dd) if max_dd != 0 else 0 return { 'annual': annual, 'idx_annual': idx_annual, 'excess': annual - idx_annual, 'max_dd': max_dd, 'sharpe': sharpe, 'calmar': calmar, 'total': total, 'idx_total': idx_total } def plot(df, title, fn): fig, ax = plt.subplots(2, 1, figsize=(14, 8)) ax[0].plot(df.index, df['nav'], 'r-', lw=2, label='Strategy') ax[0].plot(df.index, df['idx_nav'], 'gray', lw=1, alpha=0.6, label='Index') ax[0].set_title(title, fontsize=14) ax[0].legend() ax[0].grid(True, alpha=0.3) ax[1].fill_between(df.index, 0, df['pos'], alpha=0.5, color='green') ax[1].set_ylim(0, 1.1) ax[1].set_ylabel('Position') ax[1].grid(True, alpha=0.3) plt.tight_layout() plt.savefig(fn, dpi=150) print(f" 图表: {fn}") def main(): print("="*60) print("创业板50 - 趋势突破策略") print("="*60) data = get_data() print(f"\n数据: {data.index[0].date()} ~ {data.index[-1].date()}") # 训练 print("\n【训练集 2018-2023】") s = TrendStrategy() train = backtest(data, s, '2018-01-01', '2023-12-31') m = calc_metrics(train['nav'], train['idx_nav']) print(f" 策略收益: {m['total']*100:7.1f}% (年化 {m['annual']*100:5.1f}%)") print(f" 指数收益: {m['idx_total']*100:7.1f}% (年化 {m['idx_annual']*100:5.1f}%)") print(f" 超额收益: {m['excess']*100:7.1f}%") print(f" 最大回撤: {m['max_dd']*100:7.1f}%") print(f" 夏普比率: {m['sharpe']:7.2f}") print(f" 卡玛比率: {m['calmar']:7.2f}") plot(train, "Training 2018-2023", "train_trend.png") # 验证 print("\n【验证集 2024-2025】") s2 = TrendStrategy() val = backtest(data, s2, '2024-01-01', '2025-12-31') m2 = calc_metrics(val['nav'], val['idx_nav']) print(f" 策略收益: {m2['total']*100:7.1f}% (年化 {m2['annual']*100:5.1f}%)") print(f" 指数收益: {m2['idx_total']*100:7.1f}% (年化 {m2['idx_annual']*100:5.1f}%)") print(f" 超额收益: {m2['excess']*100:7.1f}%") print(f" 最大回撤: {m2['max_dd']*100:7.1f}%") print(f" 夏普比率: {m2['sharpe']:7.2f}") plot(val, "Validation 2024-2025", "val_trend.png") # 评价 print("\n【策略评价】") if m['annual'] > 0.30: print(" ✅ 训练集年化超30%,高收益潜力") elif m['annual'] > 0.15: print(" ✅ 训练集表现良好") else: print(" ⚠️ 训练集收益一般") if m2['annual'] > 0: print(" ✅ 验证集正收益") else: print(" ❌ 验证集亏损") print("\n" + "="*60) if __name__ == "__main__": main()