资源加载中... loading...

使用数字货币交易所聚合行情接口构建多品种策略

Author: 发明者量化-小小梦, Created: 2021-03-17 18:38:47, Updated: 2023-09-26 20:58:11

img

使用数字货币交易所聚合行情接口构建多品种策略

FMZ量化交易平台策略围观板块里面经常见到一些多品种策略,同时检测几十个甚至一个交易所全市场的行情。是如何做到的呢?并且需要如何设计呢?本篇文章带你一起来探讨如何使用交易所聚合行情接口构建多品种策略。

列举币安和火币这两个交易所,查看交易所API文档,发现都有聚合行情接口:

行情接口

  • 币安合约: https://fapi.binance.com/fapi/v1/ticker/bookTicker 接口返回数据

    [
        {
            "symbol": "BTCUSDT", // 交易对
            "bidPrice": "4.00000000", //最优买单价
            "bidQty": "431.00000000", //挂单量
            "askPrice": "4.00000200", //最优卖单价
            "askQty": "9.00000000", //挂单量
            "time": 1589437530011   // 撮合引擎时间
        }
        ...
    ]
    
  • 火币现货币币: https://api.huobi.pro/market/tickers 接口返回数据

    [  
        {  
            "open":0.044297,      // 开盘价
            "close":0.042178,     // 收盘价
            "low":0.040110,       // 最低价
            "high":0.045255,      // 最高价
            "amount":12880.8510,  
            "count":12838,
            "vol":563.0388715740,
            "symbol":"ethbtc",
            "bid":0.007545,
            "bidSize":0.008,
            "ask":0.008088,
            "askSize":0.009
        }, 
        ...
    ]
    

    但是,实际不是这样的,火币接口实际返回的结构是:

    {
        "status": "ok",
        "ts": 1616032188422,
        "data": [{
      	  "symbol": "hbcbtc",
      	  "open": 0.00024813,
      	  "high": 0.00024927,
      	  "low": 0.00022871,
      	  "close": 0.00023495,
      	  "amount": 2124.32,
      	  "vol": 0.517656218,
      	  "count": 1715,
      	  "bid": 0.00023427,
      	  "bidSize": 2.3,
      	  "ask": 0.00023665,
      	  "askSize": 2.93
        }, ...]
    }
    

    在处理接口返回的数据时需要注意。

构建策略程序框架

如何在策略中封装这两个接口,又如何处理数据呢? 一起慢慢来看。

先来写一个构造函数,用于构造控制对象。

// 参数e用于传入exchange交易所对象,参数subscribeList是需要处理的交易对列表,例如["BTCUSDT", "ETHUSDT", "EOSUSDT", "LTCUSDT", "ETCUSDT", "XRPUSDT"]
function createManager(e, subscribeList) {           
    var self = {}
    self.supportList = ["Futures_Binance", "Huobi"]  // 支持的交易所的

    // 对象属性
    self.e = e
    self.name = e.GetName()
    self.type = self.name.includes("Futures_") ? "Futures" : "Spot"
    self.label = e.GetLabel()
    self.quoteCurrency = ""  
    self.subscribeList = subscribeList   // subscribeList : [strSymbol1, strSymbol2, ...]
    self.tickers = []                    // 接口获取的所有行情数据,定义数据格式:{bid1: 123, ask1: 123, symbol: "xxx"}}
    self.subscribeTickers = []           // 需要的行情数据,定义数据格式:{bid1: 123, ask1: 123, symbol: "xxx"}}
    self.accData = null                  // 用于记录账户资产数据

    // 初始化函数
    self.init = function() { 
    	// 判断是否支持该交易所
        if (!_.contains(self.supportList, self.name)) {        	
        	throw "not support"
        }
    }

    // 判断数据精度
    self.judgePrecision = function (p) {
        var arr = p.toString().split(".")
        if (arr.length != 2) {
            if (arr.length == 1) {
                return 0
            }
            throw "judgePrecision error, p:" + String(p)
        }
        
        return arr[1].length
    }

    // 更新资产
    self.updateAcc = function(callBackFuncGetAcc) {
        var ret = callBackFuncGetAcc(self)
        if (!ret) {
        	return false 
        }
        self.accData = ret 
        return true 
    }

    // 更新行情数据
    self.updateTicker = function(url, callBackFuncGetArr, callBackFuncGetTicker) {
    	var tickers = []
    	var subscribeTickers = []
    	var ret = self.httpQuery(url)
    	if (!ret) {
    		return false 
    	}
    	try {
            _.each(callBackFuncGetArr(ret), function(ele) {
            	var ticker = callBackFuncGetTicker(ele)
            	tickers.push(ticker)
            	for (var i = 0 ; i < self.subscribeList.length ; i++) {
            		if (self.subscribeList[i] == ele.symbol) {
            			subscribeTickers.push(ticker)
            		}
            	}
            })
        } catch(err) {
        	Log("错误:", err)
        	return false 
        }

        self.tickers = tickers
        self.subscribeTickers = subscribeTickers
        return true 
    }

    self.httpQuery = function(url) {
    	var ret = null
        try {
            var retHttpQuery = HttpQuery(url)
            ret = JSON.parse(retHttpQuery)
        } catch (err) {
            // Log("错误:", err)
            ret = null
        }
        return ret 
    }

    self.returnTickersTbl = function() {
        var tickersTbl = {
        	type : "table", 
        	title : "tickers",
        	cols : ["symbol", "ask1", "bid1"], 
        	rows : []
        }
        _.each(self.subscribeTickers, function(ticker) {        
        	tickersTbl.rows.push([ticker.symbol, ticker.ask1, ticker.bid1])
        })
        return tickersTbl
    }

    // 初始化
    self.init()
	return self 
}

使用FMZ的API函数HttpQuery函数发出请求,访问交易所接口。使用HttpQuery时需要使用异常处理try...catch处理接口返回失败等异常情况。 看到这里有的同学可能会问:“交易所接口返回的数据结构各不相同,要怎么处理呢?用同样的处理方式肯定不行吧。” 确实如此,不仅交易所接口返回的数据结构不同,就连返回的数据字段命名也不同。同样的一个意思可能是不同的命名。例如以上我们列举的接口。同样表达的意思是买一价格,在币安称为:bidPrice,在火币称为bid

我们这里使用回调函数解决,把这些特殊处理的部分独立出来。 所以上面这个对象初始化后,具体使用时就成了这样: (以下代码省略了构造函数createManager) 以币安期货监控这些合约:["BTCUSDT", "ETHUSDT", "EOSUSDT", "LTCUSDT", "ETCUSDT", "XRPUSDT"] 火币现货监控这些币币交易对:["btcusdt", "ethusdt", "eosusdt", "etcusdt", "ltcusdt", "xrpusdt"]为例子。

function main() {
    var manager1 = createManager(exchanges[0], ["BTCUSDT", "ETHUSDT", "EOSUSDT", "LTCUSDT", "ETCUSDT", "XRPUSDT"])
    var manager2 = createManager(exchanges[1], ["btcusdt", "ethusdt", "eosusdt", "etcusdt", "ltcusdt", "xrpusdt"])   

    while (true) {
    	// 更新行情数据
    	var ticker1GetSucc = manager1.updateTicker("https://fapi.binance.com/fapi/v1/ticker/bookTicker", 
    		function(data) {return data}, 
    		function (ele) {return {bid1: ele.bidPrice, ask1: ele.askPrice, symbol: ele.symbol}})
    	var ticker2GetSucc = manager2.updateTicker("https://api.huobi.pro/market/tickers", 
    		function(data) {return data.data}, 
    		function(ele) {return {bid1: ele.bid, ask1: ele.ask, symbol: ele.symbol}})
        if (!ticker1GetSucc || !ticker2GetSucc) {
        	Sleep(1000)
        	continue
        }
        
        var tbl1 = {
        	type : "table", 
        	title : "期货行情数据",
        	cols : ["期货合约", "期货买一", "期货卖一"], 
        	rows : []
        }
        _.each(manager1.subscribeTickers, function(ticker) {
        	tbl1.rows.push([ticker.symbol, ticker.bid1, ticker.ask1])
        })
        var tbl2 = {
        	type : "table", 
        	title : "现货行情数据",
        	cols : ["现货合约", "现货买一", "现货卖一"], 
        	rows : []
        }
        _.each(manager2.subscribeTickers, function(ticker) {
        	tbl2.rows.push([ticker.symbol, ticker.bid1, ticker.ask1])
        })
        LogStatus(_D(), "\n`" + JSON.stringify(tbl1) + "`", "\n`" + JSON.stringify(tbl2) + "`")
        Sleep(10000)
    }
}

运行测试: 第一个交易所对象添加币安期货,第二个交易所对象添加火币现货 img

img

可以看到,这里把如何取接口返回的数据等操作使用回调函数进行不同交易所的特异化处理。

    	var ticker1GetSucc = manager1.updateTicker("https://fapi.binance.com/fapi/v1/ticker/bookTicker", 
    		function(data) {return data}, 
    		function (ele) {return {bid1: ele.bidPrice, ask1: ele.askPrice, symbol: ele.symbol}})
    	var ticker2GetSucc = manager2.updateTicker("https://api.huobi.pro/market/tickers", 
    		function(data) {return data.data}, 
    		function(ele) {return {bid1: ele.bid, ask1: ele.ask, symbol: ele.symbol}})

制定了行情获取,接下来可以制定账户资产获取,因为多品种策略,账户资产数据同样也要是多个的。好在交易所账户资产接口一般都是返回全资产数据。

在构造函数createManager中添加获取资产的方法

    // 更新资产
    self.updateAcc = function(callBackFuncGetAcc) {
        var ret = callBackFuncGetAcc(self)
        if (!ret) {
        	return false 
        }
        self.accData = ret 
        return true 
    }

同样由于交易所接口返回的格式,字段命名各不相同,这里也需要使用回调函数特异化处理。

以火币现货,币安期货为例子,可以这样写回调函数:

    // 获取账户资产的回调函数
    var callBackFuncGetHuobiAcc = function(self) {
        var account = self.e.GetAccount()
        var ret = []
        if (!account) {
        	return false 
        }
        // 构造资产的数组结构
        var list = account.Info.data.list
        _.each(self.subscribeList, function(symbol) {
            var coinName = symbol.split("usdt")[0]
            var acc = {symbol: symbol}
            for (var i = 0 ; i < list.length ; i++) {
            	if (coinName == list[i].currency) {
                    if (list[i].type == "trade") {
                        acc.Stocks = parseFloat(list[i].balance)
                    } else if (list[i].type == "frozen") {
                    	acc.FrozenStocks = parseFloat(list[i].balance)
                    }
                } else if (list[i].currency == "usdt") {
                	if (list[i].type == "trade") {
                		acc.Balance = parseFloat(list[i].balance)
                	} else if (list[i].type == "frozen") {
                		acc.FrozenBalance = parseFloat(list[i].balance)
                	}
                }
            }
            ret.push(acc)
        })
        return ret 
    }

    var callBackFuncGetFutures_BinanceAcc = function(self) {
    	self.e.SetCurrency("BTC_USDT")   // 设置为U本位合约的交易对
        self.e.SetContractType("swap")   // 合约都是永续合约
        var account = self.e.GetAccount() 
        var ret = []
        if (!account) {
        	return false 
        }
        var balance = account.Balance
        var frozenBalance = account.FrozenBalance
        // 构造资产数据结构
        _.each(self.subscribeList, function(symbol) {
            var acc = {symbol: symbol}
            acc.Balance = balance
            acc.FrozenBalance = frozenBalance
            ret.push(acc)
        })
        return ret 
    }

运行具有获取行情、资产功能的策略框架

行情: img

资产: img

可以看到获取到行情数据后,可以处理数据计算各个品种的差价,监控多个交易对的期现差价。 进而可以设计一个多品种的期现对冲策略。

根据这样的设计方式,还可以扩展其它的交易所,有兴趣的同学可以动手试下~


Related

More

爱Jimmy 更新资产里面的callBackFuncGetAcc是个参数,还是个函数呀?另外获取资产账户的两个回调函数callBackFuncGetHuobiAcc和callBackFuncGetFutures_BinanceAcc与callBackFuncGetAcc什么关系?我感觉是函数里面套函数,这块的知识哪里学习?

爱Jimmy 请问期现差价时,如何同时提取一个交易对的现货Bid1与期货Ask1的价格?

zltim 厉害

发明者量化-小小梦 百度回调函数。

发明者量化-小小梦 就是两个循环套着,遍历。

发明者量化-小小梦 我给你说的就是现货和期货的差价,文章上不是都截图了,期现差价。 数据都有了,具体怎么算,你减一下不就有了么。

爱Jimmy 我的思路你看对不对?假如要获取BTCUSDT现货买一价和BTCUSDT期货卖一价的差价:首先遍历期货行情数组manager1.subscribeTickers,找出BTCUSDT的卖一价manager1.subscribeTickers[i].ask1,问题是如何在期货遍历数组进行时如何提取现货BTCUSDT的bid1价格,难道再间套遍历下现货行情数组manager1.subscribeTickers?不知道老师您明白我的问题了吗?

爱Jimmy 我的意思是期货与现货的差价,不是期货或现货买一卖一的差价;碰到的问题是,不能同时获取同一交易对期货买一价和现货的卖一价。比如:如何获取BTCUSDT现货买一价和BTCUSDT期货卖一价的差价。我写了下,下面代码不知道问题出在哪里? function GetBAspot(syboml,tickerspot,BA){ for (var i = 0; i < tickerspot.length; i++) { if(tickerspot[i].syboml!==syboml){ continue }else if(tickerspot[i].syboml===syboml){ var bidspot=tickerspot[i].bid1 var askspot=tickerspot[i].ask1 } } if(BA==="bid")return bidspot if(BA==="ask")return askspot } function main(){ _.each(manager1.subscribeTickers, function(ticker) { var symb=ticker.symbol var symb1=manager1.symFuturesToSpot(symb) tbl1.rows.push([ticker.symbol, ticker.bid1, ticker.ask1,manager1.Getfundingrate(symb),manager1.Getrealrate(symb,50),GetBAspot(symb1,SpotTickers,"ask")]) }) }

发明者量化-小小梦 这个具体编写就行了,买一卖一都有了,差价不就是相互减一下。

爱Jimmy 文章中获取期现差价的图怎么没公布源码呀?

发明者量化-小小梦 没明白你的意思。

爱Jimmy 我看到了,但是没有统一指定同一交易对

发明者量化-小小梦 你看代码里买一、卖一都有数据的。