TA-Lib簡介 作為一套被業(yè)界廣泛應(yīng)用的開源技術(shù)分析庫(包含技術(shù)指標(biāo)計(jì)算和K線模式識(shí)別等),TA-Lib自2001年發(fā)布以來已經(jīng)有了十多年的歷史。TA-Lib中一共包含大約125個(gè)技術(shù)指標(biāo)的計(jì)算函數(shù),同時(shí)提供了包括C/C++、java、Perl、Python等多種語言的API。
有什么用 簡單來說TA-Lib就是提供了一堆經(jīng)過長期實(shí)踐檢驗(yàn)的技術(shù)指標(biāo)計(jì)算函數(shù)。基于現(xiàn)成的計(jì)算函數(shù),開發(fā)新策略雛形、快速驗(yàn)證某個(gè)靈感的時(shí)間可以大幅縮短,否則想象一下每開發(fā)個(gè)策略都要自己實(shí)現(xiàn)要用的技術(shù)指標(biāo),未免太浪費(fèi)時(shí)間。
但是除此以外,TA-Lib還可以有一些其他的用法,舉兩個(gè)例子。
百科全書
堅(jiān)持每天收盤后選一個(gè)自己沒用過的指標(biāo),輸入數(shù)據(jù),畫個(gè)圖、跑個(gè)回測,開發(fā)量化策略很很多其他的技術(shù)一樣都是熟能生巧。
另外,所有的技術(shù)指標(biāo)在被開發(fā)出來的時(shí)候,背后都有一定的金融邏輯原理(行為金融學(xué))的支撐,生搬硬套固然不可取,但是放著前人經(jīng)驗(yàn)完全不看,整天憑自己的空想就弄個(gè)機(jī)器學(xué)習(xí)算法在數(shù)據(jù)上瞎折騰豈不是更浪費(fèi)時(shí)間?
Alpha庫
很大一部分CTA類的策略可以總結(jié)為幾個(gè)簡單的邏輯框架,比如趨勢策略通常可以分解成以下部分:趨勢信號(hào)(通常是基于某幾個(gè)參數(shù)計(jì)算出來的指標(biāo)值超過某個(gè)閾值)、信號(hào)過濾(和趨勢信號(hào)類似)、出場方案(固定點(diǎn)數(shù)/百分比的止盈和止損,移動(dòng)止損)。
因此把邏輯框架的代碼搭好后,就可以通過機(jī)器學(xué)習(xí)算法來實(shí)現(xiàn)一種自動(dòng)的策略開發(fā)方式:
從TA-Lib中選取兩個(gè)指標(biāo)分別作為趨勢信號(hào)和信號(hào)過濾,結(jié)合止損、止盈方案,生成一個(gè)策略;
基于某一組歷史數(shù)據(jù)(如股指的1分鐘行情),通過遺傳算法來對(duì)以上的參數(shù)進(jìn)行光與優(yōu)化;
兩個(gè)指標(biāo)的參數(shù)加起來通常不會(huì)超過10個(gè),再加上止盈、止損、移動(dòng)止損的參數(shù),總參數(shù)不會(huì)超過15個(gè),在一組高達(dá)十幾萬個(gè)數(shù)據(jù)點(diǎn)的時(shí)間序列上進(jìn)行回測,過度擬合的可能性不大;
現(xiàn)在云服務(wù)器價(jià)格也不貴,租一個(gè)核多一點(diǎn)的,把算法和數(shù)據(jù)丟上去7×24小時(shí)的跑,Alpha值達(dá)到一定標(biāo)準(zhǔn)的策略存下來;
把上一步中保存下來的策略作為雛形,研究員再來進(jìn)行針對(duì)性的有效性驗(yàn)證和更精細(xì)化的策略改進(jìn),把策略開發(fā)變成有的放矢,而不是盲人摸象。
這種策略開發(fā)方式使用傳統(tǒng)的商業(yè)軟件(如TB、MC等)幾乎不可能實(shí)現(xiàn),而Python這類開源軟件就成為了最好的選擇,用戶可以自行決定幾乎所有的算法(指標(biāo)如何選擇、遺傳算法優(yōu)化參數(shù)時(shí)如何迭代等等)。
怎么安裝 盡管TA-Lib原生提供了基于SWIG封裝的Python API,但是由于性能和編譯不方便的原因,作者推薦Github上的一位開發(fā)者mrjbq7基于Cython封裝的版本。
安裝過程:
在這里下載TA_Lib-0.4.9-cp27-none-win32.whl放到桌面上,也就是vn.py建議的運(yùn)行環(huán)境Anaconda 2.7 32位
在桌面上按住Shift點(diǎn)擊鼠標(biāo)右鍵后,選擇在此處打開命令窗口打開cmd
安裝wheel包,在cmd中運(yùn)行:
pip install wheel 安裝TA-Lib,在cmd中運(yùn)行:
pip install TA_Lib-0.4.9-cp27-none-win32.whl 打開Python,運(yùn)行:
import talib 沒有報(bào)錯(cuò)則說明安裝成功
一定要嘗試自己編譯的用戶,可以根據(jù)該項(xiàng)目的網(wǎng)站上的教程來安裝。作者的兩臺(tái)電腦,一臺(tái)直接安裝成功,另一臺(tái)安裝了MinGW的電腦則報(bào)GCC編譯錯(cuò)誤(其實(shí)自己編譯沒有任何意義,感謝加州大學(xué)歐文分校打包的whl文件,省去了很多麻煩)。
linux下的安裝建議使用Anaconda的conda工具:
conda install -c quantopian ta-lib=0.4.9 具體可以參考這里:https://anaconda.org/Quantopian/ta-lib
DEMO 下面這個(gè)策略DEMO可以直接在vn.trader的CTA模塊中使用(回測、模擬交易),請(qǐng)不要用于實(shí)盤!
import talib as ta import numpy as np
from ctaBase import * from ctaTemplate import CtaTemplate
class TalibDoubleSmaDemo(CtaTemplate): “”“基于Talib模塊的雙指數(shù)均線策略Demo”“”
className = 'TalibDoubleSmaDemo'author = u'ideaplat'# 策略參數(shù)fastPeriod = 5 # 快速均線參數(shù)slowPeriod = 20 # 慢速均線參數(shù)initDays = 5 # 初始化數(shù)據(jù)所用的天數(shù)# 策略變量bar = NonebarMinute = EMPTY_STRINGcloseHistory = [] # 緩存K線收盤價(jià)的數(shù)組maxHistory = 50 # 最大緩存數(shù)量fastMa0 = EMPTY_FLOAT # 當(dāng)前最新的快速均線數(shù)值fastMa1 = EMPTY_FLOAT # 上一根的快速均線數(shù)值slowMa0 = EMPTY_FLOAT # 慢速均線數(shù)值slowMa1 = EMPTY_FLOAT# 參數(shù)列表,保存了參數(shù)的名稱paramList = ['name', 'className', 'author', 'vtSymbol', 'fastPeriod', 'slowPeriod']# 變量列表,保存了變量的名稱varList = ['inited', 'trading', 'pos', 'fastMa0', 'fastMa1', 'slowMa0', 'slowMa1']# ----------------------------------------------------------------------def __init__(self, ctaEngine, setting): """Constructor""" super(TalibDoubleSmaDemo, self).__init__(ctaEngine, setting)# ----------------------------------------------------------------------def onInit(self): """初始化策略(必須由用戶繼承實(shí)現(xiàn))""" self.writeCtaLog(u'雙SMA演示策略初始化') initData = self.loadBar(self.initDays) for bar in initData: self.onBar(bar) self.putEvent()# ----------------------------------------------------------------------def onStart(self): """啟動(dòng)策略(必須由用戶繼承實(shí)現(xiàn))""" self.writeCtaLog(u'雙SMA演示策略啟動(dòng)') self.putEvent()# ----------------------------------------------------------------------def onStop(self): """停止策略(必須由用戶繼承實(shí)現(xiàn))""" self.writeCtaLog(u'雙SMA演示策略停止') self.putEvent()# ----------------------------------------------------------------------def onTick(self, tick): """收到行情TICK推送(必須由用戶繼承實(shí)現(xiàn))""" # 計(jì)算K線 tickMinute = tick.datetime.minute if tickMinute != self.barMinute: if self.bar: self.onBar(self.bar) bar = CtaBarData() bar.vtSymbol = tick.vtSymbol bar.symbol = tick.symbol bar.exchange = tick.exchange bar.open = tick.lastPRice bar.high = tick.lastPrice bar.low = tick.lastPrice bar.close = tick.lastPrice bar.date = tick.date bar.time = tick.time bar.datetime = tick.datetime # K線的時(shí)間設(shè)為第一個(gè)Tick的時(shí)間 # 實(shí)盤中用不到的數(shù)據(jù)可以選擇不算,從而加快速度 # bar.volume = tick.volume # bar.openInterest = tick.openInterest self.bar = bar # 這種寫法為了減少一層訪問,加快速度 self.barMinute = tickMinute # 更新當(dāng)前的分鐘 else: # 否則繼續(xù)累加新的K線 bar = self.bar # 寫法同樣為了加快速度 bar.high = max(bar.high, tick.lastPrice) bar.low = min(bar.low, tick.lastPrice) bar.close = tick.lastPrice# ----------------------------------------------------------------------def onBar(self, bar): """收到Bar推送(必須由用戶繼承實(shí)現(xiàn))""" # 把最新的收盤價(jià)緩存到列表中 self.closeHistory.append(bar.close) # 檢查列表長度,如果超過緩存上限則移除最老的數(shù)據(jù) # 這樣是為了減少計(jì)算用的數(shù)據(jù)量,提高速度 if len(self.closeHistory) > self.maxHistory: self.closeHistory.pop(0) # 如果小于緩存上限,則說明初始化數(shù)據(jù)尚未足夠,不進(jìn)行后續(xù)計(jì)算 else: return # 將緩存的收盤價(jià)數(shù)轉(zhuǎn)化為numpy數(shù)組后,傳入talib的函數(shù)SMA中計(jì)算 closeArray = np.array(self.closeHistory) fastSMA = ta.SMA(closeArray, self.fastPeriod) slowSMA = ta.SMA(closeArray, self.slowPeriod) # 讀取當(dāng)前K線和上一根K線的數(shù)值,用于判斷均線交叉 self.fastMa0 = fastSMA[-1] self.fastMa1 = fastSMA[-2] self.slowMa0 = slowSMA[-1] self.slowMa1 = slowSMA[-2] # 判斷買賣 crossOver = self.fastMa0>self.slowMa0 and self.fastMa1<self.slowMa1 # 金叉上穿 crossBelow = self.fastMa0<self.slowMa0 and self.fastMa1>self.slowMa1 # 死叉下穿 # 金叉和死叉的條件是互斥 if crossOver: # 如果金叉時(shí)手頭沒有持倉,則直接做多 if self.pos == 0: self.buy(bar.close, 1) # 如果有空頭持倉,則先平空,再做多 elif self.pos < 0: self.cover(bar.close, 1) self.buy(bar.close, 1) # 死叉和金叉相反 elif crossBelow: if self.pos == 0: self.short(bar.close, 1) elif self.pos > 0: self.sell(bar.close, 1) self.short(bar.close, 1) # 發(fā)出狀態(tài)更新事件 self.putEvent()# ----------------------------------------------------------------------def onOrder(self, order): """收到委托變化推送(必須由用戶繼承實(shí)現(xiàn))""" # 對(duì)于無需做細(xì)粒度委托控制的策略,可以忽略onOrder pass# ----------------------------------------------------------------------def onTrade(self, trade): """收到成交推送(必須由用戶繼承實(shí)現(xiàn))""" # 對(duì)于無需做細(xì)粒度委托控制的策略,可以忽略onOrder pass(感謝社區(qū)ideaplat用戶貢獻(xiàn)的代碼!作者做了一些小修改。)
將上面的代碼保存到一個(gè)talibDemo.py文件中后,參考vn.trader下ctaAlgo文件夾內(nèi)的ctaBacktesting.py運(yùn)行回測,也可以通過ctaSetting.py進(jìn)行配置后,在vn.trader中進(jìn)行模擬交易。
從上面的DEMO中我們可以看到,talib中的技術(shù)指標(biāo)函數(shù)主要接受一個(gè)numpy數(shù)組作為原始數(shù)據(jù)及若干個(gè)指標(biāo)算法中的參數(shù)作為輸入,返回的數(shù)據(jù)也是一個(gè)numpy數(shù)組,使用起來非常方便。
注意緩存數(shù)據(jù)時(shí)使用的是Python列表而非numpy數(shù)組,主要原因是numpy數(shù)組的append方法本質(zhì)是結(jié)合原數(shù)組中的數(shù)據(jù)和新的數(shù)據(jù)生成一個(gè)新的數(shù)組對(duì)象,相對(duì)于列表的append開銷要高很多。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注