java实现网格交易回测

以下是一个基于Java实现的简单网格交易回测程序框架,以证券ETF(512880)为例。代码包含历史数据加载、网格策略逻辑和基础统计指标:

import java.io.BufferedReader;
import java.io.FileReader;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.*;// K线数据对象
class KData {Date date;double open;double high;double low;double close;long volume;public KData(String[] data) throws ParseException {SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");this.date = sdf.parse(data[0]);this.open = Double.parseDouble(data[1]);this.high = Double.parseDouble(data[2]);this.low = Double.parseDouble(data[3]);this.close = Double.parseDouble(data[4]);this.volume = Long.parseLong(data[5]);}
}// 网格策略回测引擎
class GridStrategyBacktest {List<KData> historicalData;double initialCapital = 100000;  // 初始资金10万元double cash = initialCapital;int position = 0;  // 持仓数量double gridStep = 0.05;  // 网格间距5%int gridLevels = 10;     // 网格层数double basePrice;        // 基准价(网格中心)List<String> trades = new ArrayList<>();  // 交易记录double totalReturn = 0;int winCount = 0;int tradeCount = 0;public GridStrategyBacktest(List<KData> data, double basePrice) {this.historicalData = data;this.basePrice = basePrice;}// 执行回测public void runBacktest() {for (KData bar : historicalData) {double price = bar.close;// 计算当前价格对应的网格层级int targetLevel = (int) ((price - basePrice) / (basePrice * gridStep));// 计算应该持有的仓位int targetPosition = Math.min(gridLevels, Math.max(-gridLevels, targetLevel)) * 100;// 执行交易int volumeToTrade = targetPosition - position;if (volumeToTrade != 0) {executeTrade(bar.date, price, volumeToTrade);}}// 计算最终收益totalReturn = (cash + position * historicalData.get(historicalData.size()-1).close - initialCapital) / initialCapital;}private void executeTrade(Date date, double price, int volume) {double cost = Math.abs(volume) * price * 1.0003;  // 包含0.03%的交易费用if (volume > 0) {  // 买入if (cost > cash) return;cash -= cost;position += volume;trades.add(String.format("%tF - 买入 %d股 @ %.4f", date, volume, price));} else {          // 卖出if (-volume > position) return;cash += (-volume) * price * 0.9997;  // 扣除卖出手续费position += volume;trades.add(String.format("%tF - 卖出 %d股 @ %.4f", date, -volume, price));}tradeCount++;if ((volume > 0 && price < basePrice) || (volume < 0 && price > basePrice)) {winCount++;}}// 输出统计结果public void printResults() {System.out.println("===== 回测结果 =====");System.out.printf("初始资金: %.2f\n", initialCapital);System.out.printf("最终资产: %.2f\n", cash + position * historicalData.get(historicalData.size()-1).close);System.out.printf("总收益率: %.2f%%\n", totalReturn * 100);System.out.printf("交易次数: %d\n胜率: %.2f%%\n", tradeCount, (double)winCount/tradeCount*100);System.out.println("\n最后5笔交易记录:");trades.subList(Math.max(0, trades.size()-5), trades.size()).forEach(System.out::println);}
}public class ETFBacktester {public static void main(String[] args) {// 1. 加载历史数据(示例数据格式:日期,开盘价,最高价,最低价,收盘价,成交量)List<KData> history = loadCSV("512880.csv");  // 需替换为实际数据文件路径// 2. 设置网格参数double basePrice = 1.0;  // 根据历史中位数设定网格基准价// 3. 运行回测GridStrategyBacktest backtester = new GridStrategyBacktest(history, basePrice);backtester.runBacktest();// 4. 输出结果backtester.printResults();}private static List<KData> loadCSV(String filename) {List<KData> data = new ArrayList<>();try (BufferedReader br = new BufferedReader(new FileReader(filename))) {String line;br.readLine();  // 跳过标题行while ((line = br.readLine()) != null) {String[] values = line.split(",");data.add(new KData(values));}} catch (Exception e) {e.printStackTrace();}return data;}
}

代码说明及使用步骤:

  1. 数据准备

    • 需要准备CSV格式的历史数据文件(示例文件名:512880.csv)
    • 数据格式(按日期排序):
      日期,开盘价,最高价,最低价,收盘价,成交量
      2023-01-03,1.052,1.068,1.050,1.065,12345678
      ...
      
  2. 核心逻辑

    • 网格生成:以基准价(basePrice)为中心,上下各生成gridLevels层网格
    • 交易触发:当价格突破网格层级时调整仓位
    • 费用计算:包含0.03%的买卖双边交易费用
  3. 关键参数

    double initialCapital = 100000;  // 初始资金
    double gridStep = 0.05;         // 5%网格间距
    int gridLevels = 10;            // 网格层数
    
  4. 输出指标

    • 总收益率
    • 交易次数
    • 胜率(盈利交易占比)
    • 详细交易记录

扩展建议(可根据需求添加):

  1. 增强统计指标

    // 在GridStrategyBacktest类中添加:
    double maxDrawdown = 0;     // 最大回撤
    double peak = initialCapital;// 在每次交易后更新:
    double currentValue = cash + position * price;
    if (currentValue > peak) {peak = currentValue;
    } else {double dd = (peak - currentValue)/peak;if (dd > maxDrawdown) maxDrawdown = dd;
    }
    
  2. 参数优化功能

    public void optimizeParameters() {for (double step = 0.03; step < 0.08; step += 0.01) {for (int levels = 5; levels <= 15; levels += 2) {GridStrategyBacktest test = new GridStrategyBacktest(history, basePrice);test.gridStep = step;test.gridLevels = levels;test.runBacktest();System.out.printf("步长:%.2f 层数:%d 收益:%.2f%%\n", step, levels, test.totalReturn*100);}}
    }
    
  3. 可视化输出

    // 使用JFreeChart库生成收益曲线图
    XYSeries series = new XYSeries("净值曲线");
    for (int i = 0; i < historicalData.size(); i++) {double value = cash + position * historicalData.get(i).close;series.add(i, value / initialCapital);
    }
    

注意事项:

  1. 需要复权价格数据(建议使用后复权)
  2. 实际交易需考虑最小交易单位(A股ETF为100股整数倍)
  3. 可增加止盈止损逻辑:
    // 在executeTrade方法中添加:
    if (totalReturn > 0.3) {  // 收益率超过30%时清仓int sellVolume = position;executeTrade(date, price, -sellVolume);
    }
    

如需完整实现,建议结合第三方库(如Ta4j用于技术指标计算)和数据库(存储历史数据)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/902852.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

探秘 3D 展厅之卓越优势,解锁沉浸式体验新境界

&#xff08;一&#xff09;打破时空枷锁&#xff0c;全球触达​ 3D 展厅的首要优势便是打破了时空限制。在传统展厅中&#xff0c;观众需要亲临现场&#xff0c;且必须在展厅开放的特定时间内参观。而 3D 展厅依托互联网&#xff0c;让观众无论身处世界哪个角落&#xff0c;只…

第十二届蓝桥杯 2021 C/C++组 直线

目录 题目&#xff1a; 题目描述&#xff1a; 题目链接&#xff1a; 思路&#xff1a; 核心思路&#xff1a; 两点确定一条直线&#xff1a; 思路详解&#xff1a; 代码&#xff1a; 第一种方式代码详解&#xff1a; 第二种方式代码详解&#xff1a; 题目&#xff1a;…

微信小程序蓝牙连接打印机打印单据完整Demo【蓝牙小票打印】

文章目录 一、准备工作1. 硬件准备2. 开发环境 二、小程序配置1. 修改app.json 三、完整代码实现1. pages/index/index.wxml2. pages/index/index.wxss3. pages/index/index.js 四、ESC/POS指令说明五、测试流程六、常见问题解决七、进一步优化建议 下面我将提供一个完整的微信…

ubuntu opencv 安装

1.ubuntu opencv 安装 在Ubuntu系统中安装OpenCV&#xff0c;可以通过多种方式进行&#xff0c;以下是一种常用的安装方法&#xff0c;包括从源代码编译安装。请注意&#xff0c;安装步骤可能会因OpenCV的版本和Ubuntu系统的具体版本而略有不同。 一、安装准备 更新系统&…

【C++】class静态常量

Usage: static const T 1 background static const成员属于类&#xff0c;而不是类的实例&#xff0c;所以它们的初始化需要在类外进行(或者在C17之后可以用inline初始化)。 使用中可能遇到的情况&#xff1a; 在头文件中声明一个static const成员&#xff0c;然后在多个cpp…

Java 安全:如何防止 DDoS 攻击?

一、DDoS 攻击简介 DDoS&#xff08;分布式拒绝服务&#xff09;攻击是一种常见的网络攻击手段&#xff0c;攻击者通过控制大量的僵尸主机向目标服务器发送海量请求&#xff0c;致使服务器资源耗尽&#xff0c;无法正常响应合法用户请求。在 Java 应用开发中&#xff0c;了解 …

统计文件中单词出现的次数并累计

# 统计单词出现次数 fileopen("E:\Dasktape/python_test.txt","r",encoding"UTF-8") f1file.read() # 读取文件 countf1.count("is") # 统计文件中is 单词出现的次数 print(f"此文件中单词is出现了{count}次")# 2.判断单词出…

C语言实现贪心算法

一、贪心算法核心思想 特征&#xff1a;在每一步选择中都采取当前状态下最优&#xff08;局部最优&#xff09;的选择&#xff0c;从而希望导致全局最优解 适用场景&#xff1a;需要满足贪心选择性质和最优子结构性质 二、经典贪心算法示例 1. 活动选择问题 目标&#xff1a…

《一文读懂Transformers库:开启自然语言处理新世界的大门》

《一文读懂Transformers库:开启自然语言处理新世界的大门》 GitHub - huggingface/transformers: 🤗 Transformers: State-of-the-art Machine Learning for Pytorch, TensorFlow, and JAX. HF-Mirror Hello! Transformers快速入门 pip install transformers -i https:/…

Vue里面elementUi-aside 和el-main不垂直排列

先说解决方法 main.js少导包 import element-ui/lib/theme-chalk/index.css; //加入此行即可 问题复现 排查了一个小时终于找出来问题了&#xff0c;建议导包去看官方的文档&#xff0c;作者就是因为看了别人的导包流程导致的问题 导包官网地址Element UI导包快速入门

MYSQL 常用字符串函数 和 时间函数详解

一、字符串函数 1、​CONCAT(str1, str2, …) 拼接多个字符串。 SELECT CONCAT(Hello, , World); -- 输出 Hello World2、SUBSTRING(str, start, length)​​ 或 ​SUBSTR() 截取字符串。 SELECT SUBSTRING(MySQL, 3, 2); -- 输出 SQ3、LENGTH(str)​​ 与 ​CHAR_LENGTH…

Python-Agent调用多个Server-FastAPI版本

Python-Agent调用多个Server-FastAPI版本 Agent调用多个McpServer进行工具调用 1-核心知识点 fastAPI的快速使用agent调用多个server 2-思路整理 1&#xff09;先把每个子服务搭建起来2&#xff09;再暴露一个Agent 3-参考网址 VSCode配置Python开发环境&#xff1a;https:/…

Drools+自定义规则库

文章目录 前言一、创建规则库二、SpringBootDrools程序1.Maven依赖2.application.yml3.Mapper.xml4.Drools配置类5.Service6.Contoller7.测试接口 前言 公司的技术方案想搭建Drools自定义规则库配合大模型进行数据的校验。本篇用来记录使用SpringBoot配合Drools开发Demo程序。…

潮了 低配电脑6G显存生成60秒AI视频 本地部署/一键包/云算力部署/批量生成

最近发现了一个让人眼前一亮的工具——FramePack&#xff0c;它能用一块普通的6GB显存笔记本GPU&#xff0c;生成60秒电影级的高清视频画面&#xff0c;效果堪称炸裂&#xff01;那么我们就把他本地部署起来玩一玩、下载离线一键整合包&#xff0c;或者是用云算力快速上手。接下…

【蓝桥杯选拔赛真题104】Scratch回文数 第十五届蓝桥杯scratch图形化编程 少儿编程创意编程选拔赛真题解析

目录 scratch回文数 一、题目要求 1、准备工作 2、功能实现 二、案例分析 1、角色分析 2、背景分析 3、前期准备 三、解题思路 四、程序编写 五、考点分析 六、推荐资料 1、scratch资料 2、python资料 3、C++资料 scratch回文数 第十五届青少年蓝桥杯scratch编…

大厂面试-框架篇

前言 本章内容来自B站黑马程序员java大厂面试题和小林coding 博主学习笔记&#xff0c;如果有不对的地方&#xff0c;海涵。 如果这篇文章对你有帮助&#xff0c;可以点点关注&#xff0c;点点赞&#xff0c;谢谢你&#xff01; 1.Spring 1.1 Spring框架中的单例bean是线程…

【AI 加持下的 Python 编程实战 2_10】DIY 拓展:从扫雷小游戏开发再探问题分解与 AI 代码调试能力(中)

文章目录 DIY 实战&#xff1a;从扫雷小游戏开发再探问题分解能力3 问题分解实战&#xff08;自顶向下&#xff09;3.2 页面渲染逻辑3.3 事件绑定逻辑 4 代码实现&#xff08;自底向上&#xff09;4.1 页面渲染部分4.2 事件绑定部分 写在前面 本篇将利用《Learn AI-assisted Py…

微信小程序开发1------微信小程序中的消息提示框总结

微信小程序中的消息提示框主要分为以下几种&#xff1a; 1. wx.showToast(Object object) 功能&#xff1a; 显示消息提示框&#xff0c;一般用于显示操作结果、状态等。 特点&#xff1a; 提示框显示在屏幕中间&#xff0c;持续一段时间后自动消失&#xff08;默认1.5秒&…

AI 场景落地:API 接口服务 VS 本地部署,哪种更适合?

在当前 AI 技术迅猛发展的背景下&#xff0c;企业在实现 AI 场景落地时&#xff0c;面临着一个关键抉择&#xff1a;是选择各大厂商提供的 API 接口服务&#xff0c;还是进行本地化部署&#xff1f;这不仅关乎成本、性能和安全性&#xff0c;还涉及到技术架构、数据治理和长期战…

Android 加壳应用运行流程 与 生命周期类处理方案

版权归作者所有&#xff0c;如有转发&#xff0c;请注明文章出处&#xff1a;https://cyrus-studio.github.io/blog/ DexClassLoader DexClassLoader 可以加载任意路径下的 dex&#xff0c;或者 jar、apk、zip 文件&#xff08;包含classes.dex&#xff09;。常用于插件化、热…