干货 | 数据管理:用python从零实现基于事件驱动的量化回测(Backtest)系统(三)

前面两篇文章,分别讲述了基于事件驱动(Event-driven)的量化回测系统的层次结构,以及事件类型。本文重点讲述市场数据是如何在回测系统以及实盘中使用的。

我们很重要的目标是要实现最大化在回测和实盘间复用代码,避免开发两套系统,也有效避免回测过程与实盘不一样。这样DataHandler提供数据,Strategy产生信号以及Porfolio类处理订单都应该使用一致的数据接口。

DataHandler是一个数据管理基类,所以子类比如基于CSV文件提供数据是CSVDataHandler或者MongodbDataHandler等,替换这个数据处理器,并不会影响Strategy或Porfolio,这样就相对容易实现功能扩展。

下面我们开始实现DataHandler。

首先我们要导入相应的头文件,包括datetime,pandas等,基类有两个抽象接口,是子类必须实现的。

update_bars:在循环中每调用一次,时间会往前走一个周期,如果是日线则就是一天,相当于新的一天收市。

get_latest_bars:是在策略计算中,可以取最近的N个bars。用于各种指标计算,比如MA(5)等

# data.py

importdatetime

importos, os.path

importpandasaspd

importcopy

fromabcimportABCMeta, abstractmethod

fromaiquant.engine.eventimportBarEvent

'''

DataHandler是一个抽象数据处理类,所以实际数据处理类都继承于此(包含历史回测、实盘)

'''

classDataHandler(object):

__metaclass__= ABCMeta

@abstractmethod

defget_latest_bars(self, symbol, N=1):

"""

返回最近的N个Bar,如果当前Bar的数量不足N,则有多少就返回多少

"""

raiseNotImplementedError("没有实现get_latest_bars()")

@abstractmethod

defupdate_bars(self):

"""

把symbol列表里所有symbol最近的Bar导入

"""

raiseNotImplementedError("没有实现update_bars()")

为简单起见,本文先实现一个数据保存的csv里的数据管理器。

classCSVDataHandler(DataHandler):

def__init__(self,events,csv_dir,symbol_list):

self.b_continue_backtest =True

self.events = events

#symbol_list:传入要处理的symbol列表集合,list类型

self.symbol_list = symbol_list

self.symbol_list_with_benchmark = copy.deepcopy(self.symbol_list)

self.symbol_list_with_benchmark.append('000300')

self.csv_dir = csv_dir

self.symbol_data = {}#symbol_data,{symbol:DataFrame}

self.latest_symbol_data = {}#最新的bar:{symbol:[bar1,bar2,barNew]}

self.b_continue_backtest =True

self._open_convert_csv_files()

def_open_convert_csv_files(self):

comb_index =None

forsinself.symbol_list_with_benchmark:

#加载csv文件,date,OHLC,Volume

self.symbol_data[s] = pd.read_csv(

os.path.join(self.csv_dir,'%s.csv'% s),

header=0,index_col=0,parse_dates=False,

names=['date','open','high','low','close','volume']

).sort_index()

# Combine the index to pad forward values

ifcomb_indexis None:

comb_index =self.symbol_data[s].index

else:

#这里要赋值,否则comb_index还是原来的index

comb_index = comb_index.union(self.symbol_data[s].index)

#设置latest symbol_data为None

self.latest_symbol_data[s] = []

# Reindex the dataframes

forsinself.symbol_list_with_benchmark:

#这是一个发生器iterrows[index,series],用next(self.symbol_data[s])

#pad方式,就是用前一天的数据再填充这一天的丢失,对于资本市场这是合理的,比如这段时间停牌。那就是按停牌前一天的价格数据来计算。

self.symbol_data[s] =self.symbol_data[s].reindex(index=comb_index,method='pad').iterrows()

def_get_new_bar(self, symbol):

"""

row = (index,series),row[0]=index,row[1]=[OHLCV]

"""

row =next(self.symbol_data[symbol])

#return tuple(symbol,row[0],row[1][0],row[1][1],row[1][2],row[1][3],row[1][4]])

row_dict = {'symbol':symbol,'date':row[0],'open':row[1][0],'high':row[1][1],'low':row[1][2],'close':row[1][3]}

returnrow_dict

defupdate_bars(self):

"""

Pushes the latest bar to the latest_symbol_data structure

for all symbols in the symbol list.

"""

forsinself.symbol_list_with_benchmark:

try:

bar =self._get_new_bar(s)

print(bar)

exceptStopIteration:

self.b_continue_backtest =False

else:

ifbaris not None:

self.latest_symbol_data[s].append(bar)

self.events.put(BarEvent())

defget_latest_bars(self, symbol, N=1):

"""

Returns the last N bars from the latest_symbol list,

or N-k if less available.

"""

try:

bars_list =self.latest_symbol_data[symbol]

exceptKeyError:

print("That symbol is not available in the historical data set.")

else:

returnbars_list[-N:]

后续还需要扩展到真实环境,比如直接从线上的mongodb里访问数据。

关于作者:魏佳斌,互联网产品/技术总监,北京大学光华管理学院(MBA),特许金融分析师(CFA)。深度关注互联网发展趋势,AI金融量化。致力于使用最新的人工智能技术去理解经济、金融,实现信息增值。

扫描下方二维码,关注:AI量化实验室(ailabx),了解AI量化最前沿技术、资讯。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容

  • **2014真题Directions:Read the following text. Choose the be...
    又是夜半惊坐起阅读 9,459评论 0 23
  • 2017.4.17星期一,晴 上午工作忙碌之余,又往工商局打了电话询问调解之事。问我可不可以主动联系调解员,对方说...
    annaqueen阅读 190评论 0 0
  • 沉香几墨满室闻, 风翻随阅葬花文。 宝黛情嗔多留问, 旧园如今莫寻门。
    刘陌Stanger阅读 136评论 0 0