pyecharts绘制K线

举报
钱塘小甲子 发表于 2019/01/23 20:05:47 2019/01/23
【摘要】 最近想扩展一下vnpy,优化一些功能和代码的性能。在看backtesting部分代码的时候,发现,vnpy其实回测功能挺弱的,可以自己扩展一下。随之而来的就是一个回测结果可视化的问题。vnpy原生的回测结果没有绘制k线,所以也就没有指标的可视化和开仓平仓的可视化,只有随后交易结果的可视化。笔者自己其实有点点不习惯,没有看到策略的可视化回测结果,有点点不开心,所以打算自己做一下。首先就是选择可...

最近想扩展一下vnpy,优化一些功能和代码的性能。在看backtesting部分代码的时候,发现,vnpy其实回测功能挺弱的,可以自己扩展一下。随之而来的就是一个回测结果可视化的问题。vnpy原生的回测结果没有绘制k线,所以也就没有指标的可视化和开仓平仓的可视化,只有随后交易结果的可视化。笔者自己其实有点点不习惯,没有看到策略的可视化回测结果,有点点不开心,所以打算自己做一下。首先就是选择可视化的工具,pyecharts应该是一个首选了,而且现在发展的越来越好了。


        那么,首先来尝试一下k线部分pyechats官方的代码吧。


        下面的代码来自官网哦

http://pyecharts.org

from pyecharts import Kline
 
v1 = [[2320.26, 2320.26, 2287.3, 2362.94], [2300, 2291.3, 2288.26, 2308.38],
      [2295.35, 2346.5, 2295.35, 2345.92], [2347.22, 2358.98, 2337.35, 2363.8],
      [2360.75, 2382.48, 2347.89, 2383.76], [2383.43, 2385.42, 2371.23, 2391.82],
      [2377.41, 2419.02, 2369.57, 2421.15], [2425.92, 2428.15, 2417.58, 2440.38],
      [2411, 2433.13, 2403.3, 2437.42], [2432.68, 2334.48, 2427.7, 2441.73],
      [2430.69, 2418.53, 2394.22, 2433.89], [2416.62, 2432.4, 2414.4, 2443.03],
      [2441.91, 2421.56, 2418.43, 2444.8], [2420.26, 2382.91, 2373.53, 2427.07],
      [2383.49, 2397.18, 2370.61, 2397.94], [2378.82, 2325.95, 2309.17, 2378.82],
      [2322.94, 2314.16, 2308.76, 2330.88], [2320.62, 2325.82, 2315.01, 2338.78],
      [2313.74, 2293.34, 2289.89, 2340.71], [2297.77, 2313.22, 2292.03, 2324.63],
      [2322.32, 2365.59, 2308.92, 2366.16], [2364.54, 2359.51, 2330.86, 2369.65],
      [2332.08, 2273.4, 2259.25, 2333.54], [2274.81, 2326.31, 2270.1, 2328.14],
      [2333.61, 2347.18, 2321.6, 2351.44], [2340.44, 2324.29, 2304.27, 2352.02],
      [2326.42, 2318.61, 2314.59, 2333.67], [2314.68, 2310.59, 2296.58, 2320.96],
      [2309.16, 2286.6, 2264.83, 2333.29], [2282.17, 2263.97, 2253.25, 2286.33],
      [2255.77, 2270.28, 2253.31, 2276.22]]
kline = Kline("K 线图示例")
kline.add("日K", ["2017/7/{}".format(i + 1) for i in range(31)], v1)
kline.render()

 首先,k线在pyechats里面定义为Kline类。这里先是写好了k线的数据,v1.我们注意到,数据的结构是一个列表的列表,里面的每一个列表是open close low high,也就是oclh格式。非常符合我们做量化的风格。


        有了数据之后是实例化k线对象,传个图像的名称就可以了。然后就是关键的add方法,其实就是往图像里面插入数据,我们可以看一下add的实现:


    def __add(self, name, x_axis, y_axis, **kwargs):
        """
        :param name:
            系列名称,用于 tooltip 的显示,legend 的图例筛选。
        :param x_axis:
            x 坐标轴数据。
        :param y_axis:
            y 坐标轴数据。数据中,每一行是一个『数据项』,每一列属于一个『维度』。
            数据项具体为 [open, close, lowest, highest] (即:[开盘值, 收盘值,
             最低值, 最高值])。
        :param kwargs:
        """

在Kline类里面,add其实调用了__add,其中,name参数其实就是一个图例,然后是x坐标的数据,y坐标的数据。

        然后就是render方法就可以获得一个html格式的结果,用浏览器打开就可以了。

大概是下面这样:

1.png

图片来自pyecharts官网。


        此外还可以增加一些别的设置,这些设置都是通过add方法中设置一些关键字来完成,比如:


kline.add(

    "日K",

    ["2017/7/{}".format(i + 1) for i in range(31)],

    v1,

    mark_point=["max"],

    is_datazoom_show=True,

)

        这里,我们发现多了mark_point和is_datazoom_show的一个设置,其中,mark_point是用来标记处最大值,而is_datazoom_show是用来标记出是否具有伸缩坐标轴功能的。


1.JPG


        同样的,坐标轴伸缩方向可以通过datazoom_orient来设置:


kline.add(

    "日K",

    ["2017/7/{}".format(i + 1) for i in range(31)],

    v1,

    mark_point=["max"],

    is_datazoom_show=True,

    datazoom_orient="vertical",

)

        此外,还可以在图上画一些别的线,比如close价的最大值。


kline.add(

    "日K",

    ["2017/7/{}".format(i + 1) for i in range(31)],

    v1,

    mark_line=["max"],

    mark_line_symbolsize=0,

    datazoom_orient="vertical",

    mark_line_valuedim="close",

)



        那么,接下来我们来看一下如何改进这个k先的绘制方法吧,我们从一个pandas开始。后面的代码就是笔者自己写的哦。


        我们现在有一个pandas, 里面的数据如下:


1.JPG


        还有一列ma10没有放上去,其实就是5日均线和10日均线。


def backtesting_plot(table_name, indicator_name_list):

    # data preparation

    da = pd.DataFrame(data=table_name)

    da['volume'] = da['volume'].apply(lambda vol: vol if vol > 0 else 0)

    date = da["datetime"].apply(lambda x: str(x)).tolist()

    k_plot_value = da.apply(lambda record: [record['open'], record['close'], record['low'], record['high']], axis=1).tolist()

    

    # K chart

    kline = Kline()

    kline.add("Backtesting Result", date, k_plot_value)

    

    indicator_lines = Line()

    for indicator_name in indicator_name_list:

        indicator_lines.add(indicator_name, date, da[indicator_name].tolist())

    # trading volume bar chart

    bar = Bar()

    bar.add("volume", date, da["volume"],

            tooltip_tragger="axis", is_legend_show=False, is_yaxis_show=False, yaxis_max=5*max(da["volume"]))

    # buy and sell

    v1 = date[10]

    v2 = da['high'].iloc[10]

    es = EffectScatter("buy")

    es.add("buy", [v1], [v2])

    v1 = date[18]

    v2 = da['high'].iloc[18]

    es.add( "sell",  [v1],  [v2], symbol="pin",)

    

    overlap = Overlap()

    overlap.add(kline)

    overlap.add(indicator_lines,)

    overlap.add(bar,yaxis_index=1, is_add_yaxis=True)

    overlap.add(es)

    overlap.render(path='tt.html')

        我们看一下上面这个函数,首先我们从pandas中拿出数据,转换成pyecharts能接受的list格式。要提醒大家的是,这里的datetime也要转化成字符串格式。


        然后就是实例化Kline和技术指标的Line


# K chart

    kline = Kline()

    kline.add("Backtesting Result", date, k_plot_value)

    

    indicator_lines = Line()

    for indicator_name in indicator_name_list:

        indicator_lines.add(indicator_name, date, da[indicator_name].tolist())

        然后用bar来制作成交量。这样的话基本就形成了。


        但是我们进一步希望能够在k线图上绘制出买卖信号发生的信息,也就是交易发出的时间点,那么我们用es来添加,这里随便使用了10和18天作为一个买卖时间点。


        我们做了这么多的图怎么让他们一起显示出来呢?这里就要用到overlap了,也就是叠加的类。


    overlap = Overlap()

    overlap.add(kline)

    overlap.add(indicator_lines,)

    overlap.add(bar,yaxis_index=1, is_add_yaxis=True)

    overlap.add(es)

    overlap.render(path='tt.html')

        我们最后来看一下结果怎么样:


1.JPG


 


        可以说相当漂亮了。笔者圈起来的就是我们绘制的时候设置的buy和sell两个点。


【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。