C++连接CTP接口实现简单量化交易(行情、交易、k线、策略)

对于量化交易来说,量化策略和技术系统缺一不可,为了知其所以然,本文实现了一个C++连接CTP接口进行仿真交易的demo,从接收行情、下订单、数据处理到添加策略、挂载运行交易等多个环节来看一下量化交易的最简单流程,管中窥豹,一探究竟。大笑


准备工作

交易所接口

这里使用上期所提供的CTP接口API,通过CTP可以连接交易所进行行情接收交易。下载地址:CTP下载

本文使用的win32版本的,Linux版本用法类似。

CTP接口包含以下内容:


  • ThostFtdcTraderApi.h:C++头文件,包含交易相关的指令,如报单。
  • ThostFtdcMdApi.h:C++头文件,包含获取行情相关的指令。
  • ThostFtdcUserApiStruct.h:包含了所有用到的数据结构。
  • ThostFtdcUserApiDataType.h:包含了所有用到的数据类型。
  • thosttraderapi.lib、thosttraderapi.dll:交易部分的动态链接库和静态链接库。
  • thostmduserapi.lib、thostmduserapi.dll:行情部分的动态链接库和静态链接库。
  • error.dtd、error.xml:包含所有可能的错误信息。

整个开发包有2个核心头文件包括4个核心接口
CThostFtdcMdApi接口和CThostFtdcTraderApi两个头文件,一个处理行情,一个处理交易
(1)处理行情的CThostFtdcMdApi接口有两个类,分别是CThostFtdcMdApi和CThostFtdcMdSpi,以Api结尾的是用来下命令的,以Spi结尾的是用来响应命令的回调。
(2)处理交易的CThostFtdcTraderApi接口也有两个类,分别是CThostFtdcTraderApi和CThostFtdcTraderSpi,  通过CThostFtdcTraderApi向CTP发送操作请求,通过CThostFtdcTraderSpi接收CTP的操作响应。

期货账户

要连接期货交易所交易,需要开设自己的账户,实现期货交易、银期转账、保证金等功能,由于小白一般不会用实盘资金交易,所以此处推荐用上期所提供的simnow虚拟交易平台simnow申请一个虚拟账户。

SIMNOW提供两类数据前置地址:

(1)交易时段的地址,如09:00-15:00和21:00-02:30,使用第一套地址,这些数据是真实的行情数据,只是时间上比真实的行情会有延迟30秒左右(SIMNOW从交易所接收后转发出来的)。

(2)非交易时段地址,这时的数据是历史行情的播放,比如昨天的数据之类的,可以用来做程序调试。


建议选择申请那个7x24行情的账户,便于开发调试。


开发步骤

工程总览



其中,

  • CTP的API文件配置到工程
  • CustomMdSpi.h,CustomMdSpi.cpp是派生的行情回调类
  • CustomTradeSpi.h,CustomTradeSpi.cpp是派生的交易回调类
  • TickToKlineHelper.h,TickToKlineHelper.cpp是处理时序数据,转换成K线的类
  • StrategyTrade.h,StrategyTrade.cpp是策略类
  • main.cpp是程序的入口

一个简单的程序化交易系统需要完成的业务可以划分为:
1.基本操作,比如登录,订阅等;
2.行情操作,比如对行情数据的接收,存储等
3.订单操作,比如报单;对报单,成交状况的查询;报单,成交状况的私有回报等。
4.数据监听和处理操作,比如接收到新数据之后的统计处理,满足统计条件后的报单处理(其实这里就是我们的策略所在)


导入CTP接口库

visual studio创建工程后,首先需要将ctp的头文件以及链接库(lib和dll)目录配置到工程


  1. // 链接库  
  2. #pragma comment (lib, "thostmduserapi.lib")  
  3. #pragma comment (lib, "thosttraderapi.lib")  


全局参数

连接到交易所,需要配置经纪商代码、帐户名、密码以及订阅合约和买卖合约的相关参数
  1. // ---- 全局变量 ---- //  
  2. // 公共参数  
  3. TThostFtdcBrokerIDType gBrokerID = "9999";                         // 模拟经纪商代码  
  4. TThostFtdcInvestorIDType gInvesterID = "";                         // 投资者账户名  
  5. TThostFtdcPasswordType gInvesterPassword = "";                     // 投资者密码  
  6.   
  7. // 行情参数  
  8. CThostFtdcMdApi *g_pMdUserApi = nullptr;                           // 行情指针  
  9. char gMdFrontAddr[] = "tcp://180.168.146.187:10010";               // 模拟行情前置地址  
  10. char *g_pInstrumentID[] = {"TF1706""zn1705""cs1801""CF705"}; // 行情合约代码列表,中、上、大、郑交易所各选一种  
  11. int instrumentNum = 4;                                             // 行情合约订阅数量  
  12. unordered_map<string, TickToKlineHelper> g_KlineHash;              // 不同合约的k线存储表  
  13.   
  14. // 交易参数  
  15. CThostFtdcTraderApi *g_pTradeUserApi = nullptr;                    // 交易指针  
  16. char gTradeFrontAddr[] = "tcp://180.168.146.187:10001";            // 模拟交易前置地址  
  17. TThostFtdcInstrumentIDType g_pTradeInstrumentID = "m1709";         // 所交易的合约代码  
  18. TThostFtdcDirectionType gTradeDirection = THOST_FTDC_D_Sell;       // 买卖方向  
  19. TThostFtdcPriceType gLimitPrice = 2818;                            // 交易价格<span style="font-family:Microsoft YaHei;">  
  20. </span>  

这里只是简单的写一下,真实完整的交易系统中,一般用配置文件,有用户去定制

行情回调类

继承CThostFtdcMdSpi实现自己的行情回调类CustomMdSpi,在系统运行时这些重写的函数会被CTP的系统api回调从而实现个性化行情

CustomMdSpi头文件

  1. #pragma once  
  2. // ---- 派生的行情类 ---- //  
  3. #include <vector>  
  4. #include "CTP_API/ThostFtdcMdApi.h"  
  5.   
  6. class CustomMdSpi: public CThostFtdcMdSpi  
  7. {  
  8.     // ---- 继承自CTP父类的回调接口并实现 ---- //  
  9. public:  
  10.     ///当客户端与交易后台建立起通信连接时(还未登录前),该方法被调用。  
  11.     void OnFrontConnected();  
  12.   
  13.     ///当客户端与交易后台通信连接断开时,该方法被调用。当发生这个情况后,API会自动重新连接,客户端可不做处理。  
  14.     ///@param nReason 错误原因  
  15.     ///        0x1001 网络读失败  
  16.     ///        0x1002 网络写失败  
  17.     ///        0x2001 接收心跳超时  
  18.     ///        0x2002 发送心跳失败  
  19.     ///        0x2003 收到错误报文  
  20.     void OnFrontDisconnected(int nReason);  
  21.   
  22.     ///心跳超时警告。当长时间未收到报文时,该方法被调用。  
  23.     ///@param nTimeLapse 距离上次接收报文的时间  
  24.     void OnHeartBeatWarning(int nTimeLapse);  
  25.   
  26.     ///登录请求响应  
  27.     void OnRspUserLogin(CThostFtdcRspUserLoginField *pRspUserLogin, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  28.   
  29.     ///登出请求响应  
  30.     void OnRspUserLogout(CThostFtdcUserLogoutField *pUserLogout, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  31.   
  32.     ///错误应答  
  33.     void OnRspError(CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  34.   
  35.     ///订阅行情应答  
  36.     void OnRspSubMarketData(CThostFtdcSpecificInstrumentField *pSpecificInstrument, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  37.   
  38.     ///取消订阅行情应答  
  39.     void OnRspUnSubMarketData(CThostFtdcSpecificInstrumentField *pSpecificInstrument, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  40.   
  41.     ///订阅询价应答  
  42.     void OnRspSubForQuoteRsp(CThostFtdcSpecificInstrumentField *pSpecificInstrument, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  43.   
  44.     ///取消订阅询价应答  
  45.     void OnRspUnSubForQuoteRsp(CThostFtdcSpecificInstrumentField *pSpecificInstrument, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  46.   
  47.     ///深度行情通知  
  48.     void OnRtnDepthMarketData(CThostFtdcDepthMarketDataField *pDepthMarketData);  
  49.   
  50.     ///询价通知  
  51.     void OnRtnForQuoteRsp(CThostFtdcForQuoteRspField *pForQuoteRsp);  
  52. };  

都是重写回调函数

连接应答

  1. // 连接成功应答  
  2. void CustomMdSpi::OnFrontConnected()  
  3. {  
  4.     std::cout << "=====建立网络连接成功=====" << std::endl;  
  5.     // 开始登录  
  6.     CThostFtdcReqUserLoginField loginReq;  
  7.     memset(&loginReq, 0, sizeof(loginReq));  
  8.     strcpy(loginReq.BrokerID, gBrokerID);  
  9.     strcpy(loginReq.UserID, gInvesterID);  
  10.     strcpy(loginReq.Password, gInvesterPassword);  
  11.     static int requestID = 0; // 请求编号  
  12.     int rt = g_pMdUserApi->ReqUserLogin(&loginReq, requestID);  
  13.     if (!rt)  
  14.         std::cout << ">>>>>>发送登录请求成功" << std::endl;  
  15.     else  
  16.         std::cerr << "--->>>发送登录请求失败" << std::endl;  
  17. }  

登录应答

  1. // 登录应答  
  2. void CustomMdSpi::OnRspUserLogin(  
  3.     CThostFtdcRspUserLoginField *pRspUserLogin,   
  4.     CThostFtdcRspInfoField *pRspInfo,   
  5.     int nRequestID,   
  6.     bool bIsLast)  
  7. {  
  8.     bool bResult = pRspInfo && (pRspInfo->ErrorID != 0);  
  9.     if (!bResult)  
  10.     {  
  11.         std::cout << "=====账户登录成功=====" << std::endl;  
  12.         std::cout << "交易日: " << pRspUserLogin->TradingDay << std::endl;  
  13.         std::cout << "登录时间: " << pRspUserLogin->LoginTime << std::endl;  
  14.         std::cout << "经纪商: " << pRspUserLogin->BrokerID << std::endl;  
  15.         std::cout << "帐户名: " << pRspUserLogin->UserID << std::endl;  
  16.         // 开始订阅行情  
  17.         int rt = g_pMdUserApi->SubscribeMarketData(g_pInstrumentID, instrumentNum);  
  18.         if (!rt)  
  19.             std::cout << ">>>>>>发送订阅行情请求成功" << std::endl;  
  20.         else  
  21.             std::cerr << "--->>>发送订阅行情请求失败" << std::endl;  
  22.     }  
  23.     else  
  24.         std::cerr << "返回错误--->>> ErrorID=" << pRspInfo->ErrorID << ", ErrorMsg=" << pRspInfo->ErrorMsg << std::endl;  
  25. }<span style="font-family:Microsoft YaHei;">  
  26. </span>  
订阅行情应答
  1. // 订阅行情应答  
  2. void CustomMdSpi::OnRspSubMarketData(  
  3.     CThostFtdcSpecificInstrumentField *pSpecificInstrument,   
  4.     CThostFtdcRspInfoField *pRspInfo,   
  5.     int nRequestID,   
  6.     bool bIsLast)  
  7. {  
  8.     bool bResult = pRspInfo && (pRspInfo->ErrorID != 0);  
  9.     if (!bResult)  
  10.     {  
  11.         std::cout << "=====订阅行情成功=====" << std::endl;  
  12.         std::cout << "合约代码: " << pSpecificInstrument->InstrumentID << std::endl;  
  13.         // 如果需要存入文件或者数据库,在这里创建表头,不同的合约单独存储  
  14.         char filePath[100] = {'\0'};  
  15.         sprintf(filePath, "%s_market_data.csv", pSpecificInstrument->InstrumentID);  
  16.         std::ofstream outFile;  
  17.         outFile.open(filePath, std::ios::out); // 新开文件  
  18.         outFile << "合约代码" << ","  
  19.             << "更新时间" << ","  
  20.             << "最新价" << ","  
  21.             << "成交量" << ","  
  22.             << "买价一" << ","  
  23.             << "买量一" << ","  
  24.             << "卖价一" << ","  
  25.             << "卖量一" << ","  
  26.             << "持仓量" << ","  
  27.             << "换手率"  
  28.             << std::endl;  
  29.         outFile.close();  
  30.     }  
  31.     else  
  32.         std::cerr << "返回错误--->>> ErrorID=" << pRspInfo->ErrorID << ", ErrorMsg=" << pRspInfo->ErrorMsg << std::endl;  
  33. }  

  • 因为是异步接口,这里连接、登录、订阅行情是一步套一步来调用的,在运行过程中,会启动一个行情线程,交易所每500ms会推送一个订阅的行情tick数据,因此,某些接口会被连续间隔调用,直到连接关闭
  • 收到行情后除了存在内存,也可以用文本文件或者数据库等形式存储起来,在这里创建初始文件或者建库
深度行情通知
  1. // 行情详情通知  
  2. void CustomMdSpi::OnRtnDepthMarketData(CThostFtdcDepthMarketDataField *pDepthMarketData)  
  3. {  
  4.     // 打印行情,字段较多,截取部分  
  5.     std::cout << "=====获得深度行情=====" << std::endl;  
  6.     std::cout << "交易日: " << pDepthMarketData->TradingDay << std::endl;  
  7.     std::cout << "交易所代码: " << pDepthMarketData->ExchangeID << std::endl;  
  8.     std::cout << "合约代码: " << pDepthMarketData->InstrumentID << std::endl;  
  9.     std::cout << "合约在交易所的代码: " << pDepthMarketData->ExchangeInstID << std::endl;  
  10.     std::cout << "最新价: " << pDepthMarketData->LastPrice << std::endl;  
  11.     std::cout << "数量: " << pDepthMarketData->Volume << std::endl;  
  12.     // 如果只获取某一个合约行情,可以逐tick地存入文件或数据库  
  13.     char filePath[100] = {'\0'};  
  14.     sprintf(filePath, "%s_market_data.csv", pDepthMarketData->InstrumentID);  
  15.     std::ofstream outFile;  
  16.     outFile.open(filePath, std::ios::app); // 文件追加写入   
  17.     outFile << pDepthMarketData->InstrumentID << ","   
  18.         << pDepthMarketData->UpdateTime << "." << pDepthMarketData->UpdateMillisec << ","   
  19.         << pDepthMarketData->LastPrice << ","   
  20.         << pDepthMarketData->Volume << ","   
  21.         << pDepthMarketData->BidPrice1 << ","   
  22.         << pDepthMarketData->BidVolume1 << ","   
  23.         << pDepthMarketData->AskPrice1 << ","   
  24.         << pDepthMarketData->AskVolume1 << ","   
  25.         << pDepthMarketData->OpenInterest << ","   
  26.         << pDepthMarketData->Turnover << std::endl;  
  27.     outFile.close();  
  28.   
  29.     // 计算实时k线  
  30.     std::string instrumentKey = std::string(pDepthMarketData->InstrumentID);  
  31.     if (g_KlineHash.find(instrumentKey) == g_KlineHash.end())  
  32.         g_KlineHash[instrumentKey] = TickToKlineHelper();  
  33.     g_KlineHash[instrumentKey].KLineFromRealtimeData(pDepthMarketData);  
  34.   
  35.   
  36.     // 取消订阅行情  
  37.     //int rt = g_pMdUserApi->UnSubscribeMarketData(g_pInstrumentID, instrumentNum);  
  38.     //if (!rt)  
  39.     //  std::cout << ">>>>>>发送取消订阅行情请求成功" << std::endl;  
  40.     //else  
  41.     //  std::cerr << "--->>>发送取消订阅行情请求失败" << std::endl;  
  42. }  
  • 每个tick世间节点系统都会调用这个函数,推送具体的行情截面数据
  • 可以在此处将行情写到本地,或者做一些数据处理(例如实时K线计算,判断是否触发策略等)

交易回调类

同理,也需要继承CThostFtdcTraderSpi来实现自己的CustomTradeSpi类,用于交易下单、报单等操作的回调

CustomTradeSpi头文件

  1. #pragma once  
  2. // ---- 派生的交易类 ---- //  
  3. #include "CTP_API/ThostFtdcTraderApi.h"  
  4.   
  5. class CustomTradeSpi : public CThostFtdcTraderSpi  
  6. {  
  7. // ---- ctp_api部分回调接口 ---- //  
  8. public:  
  9.     ///当客户端与交易后台建立起通信连接时(还未登录前),该方法被调用。  
  10.     void OnFrontConnected();  
  11.   
  12.     ///登录请求响应  
  13.     void OnRspUserLogin(CThostFtdcRspUserLoginField *pRspUserLogin, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  14.   
  15.     ///错误应答  
  16.     void OnRspError(CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  17.   
  18.     ///当客户端与交易后台通信连接断开时,该方法被调用。当发生这个情况后,API会自动重新连接,客户端可不做处理。  
  19.     void OnFrontDisconnected(int nReason);  
  20.   
  21.     ///心跳超时警告。当长时间未收到报文时,该方法被调用。  
  22.     void OnHeartBeatWarning(int nTimeLapse);  
  23.   
  24.     ///登出请求响应  
  25.     void OnRspUserLogout(CThostFtdcUserLogoutField *pUserLogout, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  26.   
  27.     ///投资者结算结果确认响应  
  28.     void OnRspSettlementInfoConfirm(CThostFtdcSettlementInfoConfirmField *pSettlementInfoConfirm, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  29.   
  30.     ///请求查询合约响应  
  31.     void OnRspQryInstrument(CThostFtdcInstrumentField *pInstrument, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  32.   
  33.     ///请求查询资金账户响应  
  34.     void OnRspQryTradingAccount(CThostFtdcTradingAccountField *pTradingAccount, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  35.   
  36.     ///请求查询投资者持仓响应  
  37.     void OnRspQryInvestorPosition(CThostFtdcInvestorPositionField *pInvestorPosition, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  38.   
  39.     ///报单录入请求响应  
  40.     void OnRspOrderInsert(CThostFtdcInputOrderField *pInputOrder, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  41.   
  42.     ///报单操作请求响应  
  43.     void OnRspOrderAction(CThostFtdcInputOrderActionField *pInputOrderAction, CThostFtdcRspInfoField *pRspInfo, int nRequestID, bool bIsLast);  
  44.   
  45.     ///报单通知  
  46.     void OnRtnOrder(CThostFtdcOrderField *pOrder);  
  47.   
  48.     ///成交通知  
  49.     void OnRtnTrade(CThostFtdcTradeField *pTrade);  
  50.       
  51. // ---- 自定义函数 ---- //  
  52. public:  
  53.     bool loginFlag; // 登陆成功的标识  
  54.     void reqOrderInsert(  
  55.         TThostFtdcInstrumentIDType instrumentID,  
  56.         TThostFtdcPriceType price,  
  57.         TThostFtdcVolumeType volume,  
  58.         TThostFtdcDirectionType direction); // 个性化报单录入,外部调用  
  59. private:  
  60.     void reqUserLogin(); // 登录请求  
  61.     void reqUserLogout(); // 登出请求  
  62.     void reqSettlementInfoConfirm(); // 投资者结果确认  
  63.     void reqQueryInstrument(); // 请求查询合约  
  64.     void reqQueryTradingAccount(); // 请求查询资金帐户  
  65.     void reqQueryInvestorPosition(); // 请求查询投资者持仓  
  66.     void reqOrderInsert(); // 请求报单录入  
  67.       
  68.     void reqOrderAction(CThostFtdcOrderField *pOrder); // 请求报单操作  
  69.     bool isErrorRspInfo(CThostFtdcRspInfoField *pRspInfo); // 是否收到错误信息  
  70.     bool isMyOrder(CThostFtdcOrderField *pOrder); // 是否我的报单回报  
  71.     bool isTradingOrder(CThostFtdcOrderField *pOrder); // 是否正在交易的报单  
  72. };  
除了重写的基类函数,还自己封装一些主动调用的操作函数,比如登入登出、下单报单、查询报单等

登录应答

  1. void CustomTradeSpi::OnRspUserLogin(  
  2.     CThostFtdcRspUserLoginField *pRspUserLogin,  
  3.     CThostFtdcRspInfoField *pRspInfo,  
  4.     int nRequestID,  
  5.     bool bIsLast)  
  6. {  
  7.     if (!isErrorRspInfo(pRspInfo))  
  8.     {  
  9.         std::cout << "=====账户登录成功=====" << std::endl;  
  10.         loginFlag = true;  
  11.         std::cout << "交易日: " << pRspUserLogin->TradingDay << std::endl;  
  12.         std::cout << "登录时间: " << pRspUserLogin->LoginTime << std::endl;  
  13.         std::cout << "经纪商: " << pRspUserLogin->BrokerID << std::endl;  
  14.         std::cout << "帐户名: " << pRspUserLogin->UserID << std::endl;  
  15.         // 保存会话参数  
  16.         trade_front_id = pRspUserLogin->FrontID;  
  17.         session_id = pRspUserLogin->SessionID;  
  18.         strcpy(order_ref, pRspUserLogin->MaxOrderRef);  
  19.   
  20.         // 投资者结算结果确认  
  21.         reqSettlementInfoConfirm();  
  22.     }  
  23. }  
查询投资者结算结果应答

  1. void CustomTradeSpi::OnRspSettlementInfoConfirm(  
  2.     CThostFtdcSettlementInfoConfirmField *pSettlementInfoConfirm,  
  3.     CThostFtdcRspInfoField *pRspInfo,  
  4.     int nRequestID,  
  5.     bool bIsLast)  
  6. {  
  7.     if (!isErrorRspInfo(pRspInfo))  
  8.     {  
  9.         std::cout << "=====投资者结算结果确认成功=====" << std::endl;  
  10.         std::cout << "确认日期: " << pSettlementInfoConfirm->ConfirmDate << std::endl;  
  11.         std::cout << "确认时间: " << pSettlementInfoConfirm->ConfirmTime << std::endl;  
  12.         // 请求查询合约  
  13.         reqQueryInstrument();  
  14.     }  
  15. }  
查询合约应答

  1. void CustomTradeSpi::OnRspQryInstrument(  
  2.     CThostFtdcInstrumentField *pInstrument,  
  3.     CThostFtdcRspInfoField *pRspInfo,  
  4.     int nRequestID,  
  5.     bool bIsLast)  
  6. {  
  7.     if (!isErrorRspInfo(pRspInfo))  
  8.     {  
  9.         std::cout << "=====查询合约结果成功=====" << std::endl;  
  10.         std::cout << "交易所代码: " << pInstrument->ExchangeID << std::endl;  
  11.         std::cout << "合约代码: " << pInstrument->InstrumentID << std::endl;  
  12.         std::cout << "合约在交易所的代码: " << pInstrument->ExchangeInstID << std::endl;  
  13.         std::cout << "执行价: " << pInstrument->StrikePrice << std::endl;  
  14.         std::cout << "到期日: " << pInstrument->EndDelivDate << std::endl;  
  15.         std::cout << "当前交易状态: " << pInstrument->IsTrading << std::endl;  
  16.         // 请求查询投资者资金账户  
  17.         reqQueryTradingAccount();  
  18.     }  
  19. }  
查询投资者资金帐户应答

  1. void CustomTradeSpi::OnRspQryTradingAccount(  
  2.     CThostFtdcTradingAccountField *pTradingAccount,  
  3.     CThostFtdcRspInfoField *pRspInfo,  
  4.     int nRequestID,  
  5.     bool bIsLast)  
  6. {  
  7.     if (!isErrorRspInfo(pRspInfo))  
  8.     {  
  9.         std::cout << "=====查询投资者资金账户成功=====" << std::endl;  
  10.         std::cout << "投资者账号: " << pTradingAccount->AccountID << std::endl;  
  11.         std::cout << "可用资金: " << pTradingAccount->Available << std::endl;  
  12.         std::cout << "可取资金: " << pTradingAccount->WithdrawQuota << std::endl;  
  13.         std::cout << "当前保证金: " << pTradingAccount->CurrMargin << std::endl;  
  14.         std::cout << "平仓盈亏: " << pTradingAccount->CloseProfit << std::endl;  
  15.         // 请求查询投资者持仓  
  16.         reqQueryInvestorPosition();  
  17.     }  
  18. }  
查询投资者持仓应答

  1. void CustomTradeSpi::OnRspQryInvestorPosition(  
  2.     CThostFtdcInvestorPositionField *pInvestorPosition,  
  3.     CThostFtdcRspInfoField *pRspInfo,  
  4.     int nRequestID,  
  5.     bool bIsLast)  
  6. {  
  7.     if (!isErrorRspInfo(pRspInfo))  
  8.     {  
  9.         std::cout << "=====查询投资者持仓成功=====" << std::endl;  
  10.         if (pInvestorPosition)  
  11.         {  
  12.             std::cout << "合约代码: " << pInvestorPosition->InstrumentID << std::endl;  
  13.             std::cout << "开仓价格: " << pInvestorPosition->OpenAmount << std::endl;  
  14.             std::cout << "开仓量: " << pInvestorPosition->OpenVolume << std::endl;  
  15.             std::cout << "开仓方向: " << pInvestorPosition->PosiDirection << std::endl;  
  16.             std::cout << "占用保证金:" << pInvestorPosition->UseMargin << std::endl;  
  17.         }  
  18.         else  
  19.             std::cout << "----->该合约未持仓" << std::endl;  
  20.           
  21.         // 报单录入请求(这里是一部接口,此处是按顺序执行)  
  22.         /*if (loginFlag) 
  23.             reqOrderInsert();*/  
  24.         if (loginFlag)  
  25.             reqOrderInsert(g_pTradeInstrumentID, gLimitPrice, 1, gTradeDirection); // 自定义一笔交易  
  26.   
  27.         // 策略交易  
  28.         /*std::cout << "=====开始进入策略交易=====" << std::endl; 
  29.         while (loginFlag) 
  30.             StrategyCheckAndTrade(g_pTradeInstrumentID, this);*/  
  31.     }  
  32. }  

这里把下单录入的操作放在了持仓结果出来之后的回调里面,策略交易也简单的放在了这里,真实的情况下,应该是由行情触发某个策略条件开一个线程进行策略交易


下单操作

  1. void CustomTradeSpi::reqOrderInsert(  
  2.     TThostFtdcInstrumentIDType instrumentID,  
  3.     TThostFtdcPriceType price,  
  4.     TThostFtdcVolumeType volume,  
  5.     TThostFtdcDirectionType direction)  
  6. {  
  7.     CThostFtdcInputOrderField orderInsertReq;  
  8.     memset(&orderInsertReq, 0, sizeof(orderInsertReq));  
  9.     ///经纪公司代码  
  10.     strcpy(orderInsertReq.BrokerID, gBrokerID);  
  11.     ///投资者代码  
  12.     strcpy(orderInsertReq.InvestorID, gInvesterID);  
  13.     ///合约代码  
  14.     strcpy(orderInsertReq.InstrumentID, instrumentID);  
  15.     ///报单引用  
  16.     strcpy(orderInsertReq.OrderRef, order_ref);  
  17.     ///报单价格条件: 限价  
  18.     orderInsertReq.OrderPriceType = THOST_FTDC_OPT_LimitPrice;  
  19.     ///买卖方向:   
  20.     orderInsertReq.Direction = direction;  
  21.     ///组合开平标志: 开仓  
  22.     orderInsertReq.CombOffsetFlag[0] = THOST_FTDC_OF_Open;  
  23.     ///组合投机套保标志  
  24.     orderInsertReq.CombHedgeFlag[0] = THOST_FTDC_HF_Speculation;  
  25.     ///价格  
  26.     orderInsertReq.LimitPrice = price;  
  27.     ///数量:1  
  28.     orderInsertReq.VolumeTotalOriginal = volume;  
  29.     ///有效期类型: 当日有效  
  30.     orderInsertReq.TimeCondition = THOST_FTDC_TC_GFD;  
  31.     ///成交量类型: 任何数量  
  32.     orderInsertReq.VolumeCondition = THOST_FTDC_VC_AV;  
  33.     ///最小成交量: 1  
  34.     orderInsertReq.MinVolume = 1;  
  35.     ///触发条件: 立即  
  36.     orderInsertReq.ContingentCondition = THOST_FTDC_CC_Immediately;  
  37.     ///强平原因: 非强平  
  38.     orderInsertReq.ForceCloseReason = THOST_FTDC_FCC_NotForceClose;  
  39.     ///自动挂起标志: 否  
  40.     orderInsertReq.IsAutoSuspend = 0;  
  41.     ///用户强评标志: 否  
  42.     orderInsertReq.UserForceClose = 0;  
  43.   
  44.     static int requestID = 0; // 请求编号  
  45.     int rt = g_pTradeUserApi->ReqOrderInsert(&orderInsertReq, ++requestID);  
  46.     if (!rt)  
  47.         std::cout << ">>>>>>发送报单录入请求成功" << std::endl;  
  48.     else  
  49.         std::cerr << "--->>>发送报单录入请求失败" << std::endl;  
  50. }  
通过重载写了两个函数,一个是用默认参数下单,一个可以传参下单,比如设定合约代码、价格、数量等

报单操作

  1. void CustomTradeSpi::reqOrderAction(CThostFtdcOrderField *pOrder)  
  2. {  
  3.     static bool orderActionSentFlag = false// 是否发送了报单  
  4.     if (orderActionSentFlag)  
  5.         return;  
  6.   
  7.     CThostFtdcInputOrderActionField orderActionReq;  
  8.     memset(&orderActionReq, 0, sizeof(orderActionReq));  
  9.     ///经纪公司代码  
  10.     strcpy(orderActionReq.BrokerID, pOrder->BrokerID);  
  11.     ///投资者代码  
  12.     strcpy(orderActionReq.InvestorID, pOrder->InvestorID);  
  13.     ///报单操作引用  
  14.     //  TThostFtdcOrderActionRefType    OrderActionRef;  
  15.     ///报单引用  
  16.     strcpy(orderActionReq.OrderRef, pOrder->OrderRef);  
  17.     ///请求编号  
  18.     //  TThostFtdcRequestIDType RequestID;  
  19.     ///前置编号  
  20.     orderActionReq.FrontID = trade_front_id;  
  21.     ///会话编号  
  22.     orderActionReq.SessionID = session_id;  
  23.     ///交易所代码  
  24.     //  TThostFtdcExchangeIDType    ExchangeID;  
  25.     ///报单编号  
  26.     //  TThostFtdcOrderSysIDType    OrderSysID;  
  27.     ///操作标志  
  28.     orderActionReq.ActionFlag = THOST_FTDC_AF_Delete;  
  29.     ///价格  
  30.     //  TThostFtdcPriceType LimitPrice;  
  31.     ///数量变化  
  32.     //  TThostFtdcVolumeType    VolumeChange;  
  33.     ///用户代码  
  34.     //  TThostFtdcUserIDType    UserID;  
  35.     ///合约代码  
  36.     strcpy(orderActionReq.InstrumentID, pOrder->InstrumentID);  
  37.     static int requestID = 0; // 请求编号  
  38.     int rt = g_pTradeUserApi->ReqOrderAction(&orderActionReq, ++requestID);  
  39.     if (!rt)  
  40.         std::cout << ">>>>>>发送报单操作请求成功" << std::endl;  
  41.     else  
  42.         std::cerr << "--->>>发送报单操作请求失败" << std::endl;  
  43.     orderActionSentFlag = true;  
  44. }  
主要是对于未成交的订单进行编辑或者撤销操作

报单应答

  1. void CustomTradeSpi::OnRtnOrder(CThostFtdcOrderField *pOrder)  
  2. {  
  3.     char str[10];  
  4.     sprintf(str, "%d", pOrder->OrderSubmitStatus);  
  5.     int orderState = atoi(str) - 48;    //报单状态0=已经提交,3=已经接受  
  6.   
  7.     std::cout << "=====收到报单应答=====" << std::endl;  
  8.   
  9.     if (isMyOrder(pOrder))  
  10.     {  
  11.         if (isTradingOrder(pOrder))  
  12.         {  
  13.             std::cout << "--->>> 等待成交中!" << std::endl;  
  14.             //reqOrderAction(pOrder); // 这里可以撤单  
  15.             //reqUserLogout(); // 登出测试  
  16.         }  
  17.         else if (pOrder->OrderStatus == THOST_FTDC_OST_Canceled)  
  18.             std::cout << "--->>> 撤单成功!" << std::endl;  
  19.     }  
  20. }  
  21.   
  22. void CustomTradeSpi::OnRtnTrade(CThostFtdcTradeField *pTrade)  
  23. {  
  24.     std::cout << "=====报单成功成交=====" << std::endl;  
  25.     std::cout << "成交时间: " << pTrade->TradeTime << std::endl;  
  26.     std::cout << "合约代码: " << pTrade->InstrumentID << std::endl;  
  27.     std::cout << "成交价格: " << pTrade->Price << std::endl;  
  28.     std::cout << "成交量: " << pTrade->Volume << std::endl;  
  29.     std::cout << "开平仓方向: " << pTrade->Direction << std::endl;  
  30. }  

等待成交进行轮询可以选择报单操作,成交完成后的应答


时间序列转K线

从交易拿到的tick数据是时间序列数据,在证券交易中其实还需要根据时间序列算出一些技术指标数据,例如MACD,KDJ、K线等,这里简单地对数据做一下处理,写一个TickToKlineHelper将时间序列专程K线


K线数据结构

  1. // k线数据结构  
  2. struct KLineDataType  
  3. {  
  4.     double open_price;   // 开  
  5.     double high_price;   // 高  
  6.     double low_price;    // 低  
  7.     double close_price;  // 收  
  8.     int volume;          // 量  
  9. };  

转换函数

  1. void TickToKlineHelper::KLineFromLocalData(const std::string &sFilePath, const std::string &dFilePath)  
  2. {  
  3.     // 先清理残留数据  
  4.     m_priceVec.clear();  
  5.     m_volumeVec.clear();  
  6.     m_KLineDataArray.clear();  
  7.   
  8.     std::cout << "开始转换tick到k线..." << std::endl;  
  9.     // 默认读取的tick数据表有4个字段:合约代码、更新时间、最新价、成交量  
  10.     std::ifstream srcInFile;  
  11.     std::ofstream dstOutFile;  
  12.     srcInFile.open(sFilePath, std::ios::in);  
  13.     dstOutFile.open(dFilePath, std::ios::out);  
  14.     dstOutFile << "开盘价" << ','  
  15.         << "最高价" << ','  
  16.         << "最低价" << ','  
  17.         << "收盘价" << ','   
  18.         << "成交量" << std::endl;  
  19.   
  20.     // 一遍解析文件一边计算k线数据,1分钟k线每次读取60 * 2 = 120行数据  
  21.     std::string lineStr;  
  22.     bool isFirstLine = true;  
  23.     while (std::getline(srcInFile, lineStr))  
  24.     {  
  25.         if (isFirstLine)  
  26.         {  
  27.             // 跳过第一行表头  
  28.             isFirstLine = false;  
  29.             continue;  
  30.         }  
  31.         std::istringstream ss(lineStr);  
  32.         std::string fieldStr;  
  33.         int count = 4;  
  34.         while (std::getline(ss, fieldStr, ','))  
  35.         {  
  36.             count--;  
  37.             if (count == 1)  
  38.                 m_priceVec.push_back(std::atof(fieldStr.c_str()));  
  39.             else if (count == 0)  
  40.             {  
  41.                 m_volumeVec.push_back(std::atoi(fieldStr.c_str()));  
  42.                 break;  
  43.             }  
  44.         }  
  45.   
  46.         // 计算k线  
  47.   
  48.         if (m_priceVec.size() == kDataLineNum)  
  49.         {  
  50.             KLineDataType k_line_data;  
  51.             k_line_data.open_price = m_priceVec.front();  
  52.             k_line_data.high_price = *std::max_element(m_priceVec.cbegin(), m_priceVec.cend());  
  53.             k_line_data.low_price = *std::min_element(m_priceVec.cbegin(), m_priceVec.cend());  
  54.             k_line_data.close_price = m_priceVec.back();  
  55.             // 成交量的真实的算法是当前区间最后一个成交量减去上去一个区间最后一个成交量  
  56.             k_line_data.volume = m_volumeVec.back() - m_volumeVec.front();   
  57.             //m_KLineDataArray.push_back(k_line_data); // 此处可以存到内存  
  58.               
  59.             dstOutFile << k_line_data.open_price << ','  
  60.                 << k_line_data.high_price << ','  
  61.                 << k_line_data.low_price << ','  
  62.                 << k_line_data.close_price << ','  
  63.                 << k_line_data.volume << std::endl;  
  64.   
  65.             m_priceVec.clear();  
  66.             m_volumeVec.clear();  
  67.         }  
  68.     }  
  69.   
  70.     srcInFile.close();  
  71.     dstOutFile.close();  
  72.   
  73.     std::cout << "k线生成成功" << std::endl;  
  74. }  
  75.   
  76. void TickToKlineHelper::KLineFromRealtimeData(CThostFtdcDepthMarketDataField *pDepthMarketData)  
  77. {  
  78.     m_priceVec.push_back(pDepthMarketData->LastPrice);  
  79.     m_volumeVec.push_back(pDepthMarketData->Volume);  
  80.     if (m_priceVec.size() == kDataLineNum)  
  81.     {  
  82.         KLineDataType k_line_data;  
  83.         k_line_data.open_price = m_priceVec.front();  
  84.         k_line_data.high_price = *std::max_element(m_priceVec.cbegin(), m_priceVec.cend());  
  85.         k_line_data.low_price = *std::min_element(m_priceVec.cbegin(), m_priceVec.cend());  
  86.         k_line_data.close_price = m_priceVec.back();  
  87.         // 成交量的真实的算法是当前区间最后一个成交量减去上去一个区间最后一个成交量  
  88.         k_line_data.volume = m_volumeVec.back() - m_volumeVec.front();  
  89.         m_KLineDataArray.push_back(k_line_data); // 此处可以存到内存  
  90.   
  91.         m_priceVec.clear();  
  92.         m_volumeVec.clear();  
  93.     }  
  94. }<span style="font-family:Microsoft YaHei;">  
  95. </span>  

  • 可以从本地文件中读取行情数据,进行离线转换,也可以在接受到行情时进行实时计算
  • 基本思想是,针对每个合约代码,建立字典,维持一个行情数组,当时间间隔达到要求(例如分钟、分时、分日)时计算该时段的开、高、低、收、成交量等数据存入K线数组
  • 最低时间单位的K线计算出来之后,高时间间隔的K线数据可以根据低时间间隔的K线计算出来(例如,算出了分钟K,那么分时K就根据分钟K来算)
  • 本例子中只是实现了一个大概的原理,非常不精确,仅供参考

策略交易

量化交易系统最终是需要将编写的策略代码挂载到系统中进行策略交易的,这里做了一个简单的实现
StrategyTrade.h
  1. #pragma once  
  2. // ---- 简单策略交易的类 ---- //  
  3.   
  4. #include <functional>  
  5. #include "CTP_API/ThostFtdcUserApiStruct.h"  
  6. #include "TickToKlineHelper.h"  
  7. #include "CustomTradeSpi.h"  
  8.   
  9. typedef void(*reqOrderInsertFun)(  
  10.     TThostFtdcInstrumentIDType instrumentID,  
  11.     TThostFtdcPriceType price,  
  12.     TThostFtdcVolumeType volume,  
  13.     TThostFtdcDirectionType direction);  
  14.   
  15. using ReqOrderInsertFunctionType = std::function<  
  16.     void(TThostFtdcInstrumentIDType instrumentID,  
  17.     TThostFtdcPriceType price,  
  18.     TThostFtdcVolumeType volume,  
  19.     TThostFtdcDirectionType direction)>;  
  20.   
  21. void StrategyCheckAndTrade(TThostFtdcInstrumentIDType instrumentID, CustomTradeSpi *customTradeSpi);<span style="font-family:Microsoft YaHei;">  
  22. </span>  

StrategyTrade.cpp
  1. #include <vector>  
  2. #include <string>  
  3. #include <unordered_map>  
  4. #include <thread>  
  5. #include <mutex>  
  6. #include "StrategyTrade.h"  
  7. #include "CustomTradeSpi.h"  
  8.   
  9. extern std::unordered_map<std::string, TickToKlineHelper> g_KlineHash;  
  10.   
  11. // 线程互斥量  
  12. std::mutex marketDataMutex;  
  13.   
  14. void StrategyCheckAndTrade(TThostFtdcInstrumentIDType instrumentID, CustomTradeSpi *customTradeSpi)  
  15. {  
  16.     // 加锁  
  17.     std::lock_guard<std::mutex> lk(marketDataMutex);  
  18.     TickToKlineHelper tickToKlineObject = g_KlineHash.at(std::string(instrumentID));  
  19.     // 策略  
  20.     std::vector<double> priceVec = tickToKlineObject.m_priceVec;  
  21.     if (priceVec.size() >= 3)  
  22.     {  
  23.         int len = priceVec.size();  
  24.         // 最后连续三个上涨就买开仓,反之就卖开仓,这里暂时用最后一个价格下单  
  25.         if (priceVec[len - 1] > priceVec[len - 2] && priceVec[len - 2] > priceVec[len - 3])  
  26.             customTradeSpi->reqOrderInsert(instrumentID, priceVec[len - 1], 1, THOST_FTDC_D_Buy);  
  27.         else if (priceVec[len - 1] < priceVec[len - 2] && priceVec[len - 2] < priceVec[len - 3])  
  28.             customTradeSpi->reqOrderInsert(instrumentID, priceVec[len - 1], 1, THOST_FTDC_D_Buy);  
  29.     }  
  30. }<span style="font-family:Microsoft YaHei;">  
  31. </span>  
  • 基本思想,针对指定合约,判断如果连续三个上涨就买开仓,连续三个下跌就卖开仓,价格都是用最新价
  • 因为行情和交易是分开的线程,涉及到线程竞争,所以在实际下单时需要加入互斥锁,线程同步
  • 策略如何被行情触发然后交易其实需要用事件驱动来做的,这里没有实现T_T

入口

main.cpp
  1. int main()  
  2. {  
  3.     // 账号密码  
  4.     cout << "请输入账号: ";  
  5.     scanf("%s", gInvesterID);  
  6.     cout << "请输入密码: ";  
  7.     scanf("%s", gInvesterPassword);  
  8.   
  9.     // 初始化行情线程  
  10.     cout << "初始化行情..." << endl;  
  11.     g_pMdUserApi = CThostFtdcMdApi::CreateFtdcMdApi();   // 创建行情实例  
  12.     CThostFtdcMdSpi *pMdUserSpi = new CustomMdSpi;       // 创建行情回调实例  
  13.     g_pMdUserApi->RegisterSpi(pMdUserSpi);               // 注册事件类  
  14.     g_pMdUserApi->RegisterFront(gMdFrontAddr);           // 设置行情前置地址  
  15.     g_pMdUserApi->Init();                                // 连接运行  
  16.       
  17.   
  18.   
  19.     // 初始化交易线程  
  20.     cout << "初始化交易..." << endl;  
  21.     g_pTradeUserApi = CThostFtdcTraderApi::CreateFtdcTraderApi(); // 创建交易实例  
  22.     //CThostFtdcTraderSpi *pTradeSpi = new CustomTradeSpi;  
  23.     CustomTradeSpi *pTradeSpi = new CustomTradeSpi;               // 创建交易回调实例  
  24.     g_pTradeUserApi->RegisterSpi(pTradeSpi);                      // 注册事件类  
  25.     g_pTradeUserApi->SubscribePublicTopic(THOST_TERT_RESTART);    // 订阅公共流  
  26.     g_pTradeUserApi->SubscribePrivateTopic(THOST_TERT_RESTART);   // 订阅私有流  
  27.     g_pTradeUserApi->RegisterFront(gTradeFrontAddr);              // 设置交易前置地址  
  28.     g_pTradeUserApi->Init();                                      // 连接运行  
  29.           
  30.   
  31.     // 等到线程退出  
  32.     g_pMdUserApi->Join();  
  33.     delete pMdUserSpi;  
  34.     g_pMdUserApi->Release();  
  35.   
  36.     g_pTradeUserApi->Join();  
  37.     delete pTradeSpi;  
  38.     g_pTradeUserApi->Release();  
  39.   
  40.     // 转换本地k线数据  
  41.     //TickToKlineHelper tickToKlineHelper;  
  42.     //tickToKlineHelper.KLineFromLocalData("market_data.csv", "K_line_data.csv");  
  43.       
  44.     getchar();  
  45.     return 0;  
  46. }  
  • CThostFtdcMdApi跟CustomMdSpi要建立关联,CThostFtdcTraderApi跟CustomTradeSpi建立关联,其实就是类似于函数注册
  • 配置行情和交易地址
  • 行情和交易分别是不同的线程,注意线程同步
  • 记得内存回收

运行结果

行情

应答日志


存成csv表格



交易

应答日志



K线数据




报单情况

用上期所的快期软件,登录上自己的账号之后,从过程序下单,在这个界面里能看到实时的报单成交状况


特别声明:标价仅为视频价格(为避免不必要的纠纷,请详细了解清楚后再拍!!!)特别声明:标价仅为视频价格(为避免不必要的纠纷,请详细了解清楚后再拍!!!)特别声明:标价仅为视频价格(为避免不必要的纠纷,请详细了解清楚后再拍!!!) 获取文档和源码请加作者vx:X_Trader_Lab适合人群X-Trader:从CTP-API 到期货日内策略,适合对期货期权日内交易感兴趣的同学。 学习目标本课程体系分成三个子课程(本子课程为课程体系的第一部分):本课程体系分成三个子课程(本子课程为课程体系的第一部分):本课程体系分成三个子课程(本子课程为课程体系的第一部分):1.CTP-API交易接口深度解析2.深度探索X-Trader交易框架3.期货日内策略的原理及框架(本子课程为课程体系的第一部分) 详细介绍X-Trader 是一个基于C++ 的,适应全市场全品种交易的跨平台的极简量化交易框架,X-Trader 支持用户使用C++ 构 建各种类型的量化交易策略程序, 并提供包含历史数据-实时数据-开发调试-模拟交易-实盘交易-运行监控-风险 管理的一站式解决方案。本课程旨在帮助同学们从0 到1,掌握期货日内交易策略框架的核心技术。 课程大纲1.1 CTP1.2 API1.3 开启程序化交易之旅1.4 CTP-API 的基本架构及初始化1.5 行情接口开发1.6 行情数据处理1.7 看穿式监管评测1.8 交易接口开发1.9 报单1.10 报单回报1.11 撤单1.12 成交回报1.13 回调规则1.14 查询持仓1.15 更新持仓1.16 查询保证金率和手续费率1.17 规避自成交1.18 报单流控、查询流控和会话数控制
在Linux环境下使用C语言来完成ctp期货量化交易系统,首先需要安装相应的开发工具和环境,例如gcc编译器和相关的开发库。然后,可以通过ctp官方提供的API来进行开发。 接下来,需要编写C语言程序来连接ctp交易接口,包括登录行情服务器、连接交易服务器、订阅行情数据、下单交易等相关功能。在编写程序时,需要充分了解ctp交易接口的相关文档和示例代码,以便正确地调用接口函数。 在交易系统的开发过程中,需要考虑到错误处理、数据处理、交易策略实现等方面。对于错误处理,可以通过编写日志来记录程序的运行情况,以便排查错误。对于数据处理,可以通过编写算法来对行情数据进行分析和处理,以支持量化交易策略实现。 在编写交易策略时,需要根据具体的量化交易策略实现相应的买卖逻辑,可以通过编写条件判断语句和相关算法来实现交易决策。 最后,在完成ctp期货量化交易系统的开发后,还需要进行充分的测试和优化。通过模拟交易和回测来验证交易系统的稳定性和盈利性,通过优化代码和算法来提高系统的性能和效率。 总之,在Linux环境下使用C语言完成ctp期货量化交易系统的开发,需要充分的了解ctp接口和API,编写对应的功能程序,实现量化交易策略,并进行测试和优化,以确保系统的稳定性和盈利性。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值