/CTPSharp

CTP SDK的 C#绑定,当前支持的CTP SDK版本为6.7.2

Primary LanguageC#

CTPSharp

期货CTP接口API的C#绑定,基于代码生成的Window/Linux X64版本。 当前CTP版本为6.7.2

相较于其他一些封装库,本项目有以下优势:

  • 完全的原始SDK命名兼容
  • 增加了对原始callback形式的API的C#封装,可使用 Task和 event
  • 字符串使用GBK编码与SDK沟通,没有乱码
  • 最大限度避免了PInvoke调用过程中的内存问题

特别声明

此项目代码不得用于任何商业目的,若需要,请和我联系: 929968367@qq.com

项目结构:

CTPAPI 目录

此目录为原始 X64 C++ API,含动态库 .dll 和 连接库 .lib

Generator 目录

此项目为代码生成工具,用于将【CTPAPI】目录下的 .h文件分析后转换成CTPInvoke所需的CPP代码以及CTPSharp项目所需的C#代码。

得益于CTP SDK头文件的良好格式化,此工具不做任何词法语法分析,仅做字符串解析。

输出文件存放在 【generated】目录下

后续CTP SDK 版本更新,可直接放在【CTPAPI】目录下,重新生成

CTPInvoke 目录

此目录为 C# 代码需要调用的 PInvoke库, 其输出文件名需要是 CTPInvoke.dll(Linux下则需要命名为.so) 此项目的源代码为【generated】目录下的生成的 .cpp文件. 链接库需要加入 CTPAPI 中的两个 .lib。

Linux版本

Linux 编译需要cmake、g++。注意原始的so文件名需要保持lib前缀形势否则链接不过。

cd CTPInvoke
mkdir build
cd build
cmake .. -DCMAKE_BUILD_TYPE=RELEASE  # 构建 release 版本
或者
cmake .. -DCMAKE_BUILD_TYPE=DEBUG  # 构建 debug 版本
make

至此在 Publish/Linux64.NET8/Debug 或 Publish/Linux64.NET8/Release 目录下生成 libCTPInvoke.so 文件,对应于Windows平台的CTPInvoke.dll

CTPInvoke这个C库的作用是将C++接口转换成纯粹C形式的接口,因此它也可以直接被C#之外的其他语言调用。【generated】目录下有一个python脚本文件rust_bind.py,可用来将CTPInvoke这个C库转换成rust绑定文件 CTP.rs。该脚本依赖rust生态中的bindgen工具,运行时需要先安装它。

CTPSharp

此目录为可被 .NET 程序引用的动态库项目。其源码是 generated 目录下的全部 .cs文件 此项目为纯 .NET 项目,生成一个 CTPSharp.dll

客户.NET代码引用此绑定库的方法

  1. 拷贝CTPInvoke.dll、CTPSharp.dll和CTPAPI目录下的两个原始 _se.dll(Linux平台则是.so)
  2. 客户 .NET 项目添加对 CTPSharp.dll的引用
  3. 可完全参照原始 C++ API来使用CTPSharp.dll,所有类、方法的名称都和 C++ 版本保持一致
  4. 也可选择使用封装的两个类:XP.CTPSharp.TdAPI和XP.CTPSharp.MdAPI
  5. 注意使用 IDisposable 来管理对象生命周期

预制版本

如果不想编译,可以直接使用publish目录下的预制版本. 当前CTPInvoke及两个原生dll为 X64架构,CTPSharp.dll目标平台为 .NET 8

使用代码样例

class MDClient: IDisposable
{
    private MdAPI _api;
    private Config _config;
    public MDClient(string flowPath, Config config)
    {
        _config = config;
        _api = new(flowPath);
        _api.FrontConnected += OnFrontConnected;
        _api.FrontDisconnected += OnFrontDisconnected;
        _api.DepthMarketDataArrived += OnDepthMarketData;
        _api.RegisterFront(_config.DataServer);
         // 连接行情服务器;
        _api.Init();
    }

    public void Dispose()
    {
        _api.Dispose();
    }

    private void OnFrontConnected()
    {
        LogDebug("服务器连接成功");
        ReqLoginAsync();
    }

    private async void ReqLoginAsync()
    {
        var req = new CThostFtdcReqUserLoginField
        {
            UserID = _config.User,
            Password = _config.Password,
            BrokerID = _config.Broker
        };
        var resp = await _api.ReqUserLoginAsync(req);
        if (resp.NoError)
        {
            LogWarning("登录成功");
        }
        else
        {
            LogWarning($"登录失败,{resp.InfoField.ErrorMsg}, code={resp.InfoField.ErrorID}");
        }
    }

    private void OnFrontDisconnected(int nReason)
    {
        LogDebug($"连接断开,代码{nReason}");
    }

    private void LogDebug(string msg) {}
    private void LogWarning(string msg) {}
}

class Config
{
    public string User { get; }
    public string Password { get; }
    public string Broker { get; }
    public string DataServer { get; }
    public string TradeServer { get; }
    public string AuthCode { get; }
    public string AppId { get; }
    public string ProductInfo { get; }
}

详细可参考Example项目代码