百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

zigzag源代码——包括python和mql语言

itomcoil 2025-01-04 20:24 34 浏览


"""
reference:
https://github.com/jbn/ZigZag.git
"""
import numpy as np

PEAK = 1
VALLEY = -1


def identify_initial_pivot(X, up_thresh, down_thresh):
    x_0 = X[0]
    x_t = x_0

    max_x = x_0
    min_x = x_0

    max_t = 0
    min_t = 0

    up_thresh += 1
    down_thresh += 1

    for t in range(1, len(X)):
        x_t = X[t]

        if x_t / min_x >= up_thresh:
            return VALLEY if min_t == 0 else PEAK

        if x_t / max_x <= down_thresh:
            return PEAK if max_t == 0 else VALLEY

        if x_t > max_x:
            max_x = x_t
            max_t = t

        if x_t < min_x:
            min_x = x_t
            min_t = t

    t_n = len(X)-1
    return VALLEY if x_0 < X[t_n] else PEAK


def peak_valley_pivots(X, up_thresh, down_thresh):
    """
    Find the peaks and valleys of a series.

    :param X: the series to analyze
    :param up_thresh: minimum relative change necessary to define a peak
    :param down_thesh: minimum relative change necessary to define a valley
    :return: an array with 0 indicating no pivot and -1 and 1 indicating
        valley and peak


    The First and Last Elements
    ---------------------------
    The first and last elements are guaranteed to be annotated as peak or
    valley even if the segments formed do not have the necessary relative
    changes. This is a tradeoff between technical correctness and the
    propensity to make mistakes in data analysis. The possible mistake is
    ignoring data outside the fully realized segments, which may bias
    analysis.
    """
    if down_thresh > 0:
        raise ValueError('The down_thresh must be negative.')

    initial_pivot = identify_initial_pivot(X, up_thresh, down_thresh)
    t_n = len(X)
    pivots = np.zeros(t_n, dtype=np.int_)
    trend = -initial_pivot
    last_pivot_t = 0
    last_pivot_x = X[0]

    pivots[0] = initial_pivot

    # Adding one to the relative change thresholds saves operations. Instead
    # of computing relative change at each point as x_j / x_i - 1, it is
    # computed as x_j / x_1. Then, this value is compared to the threshold + 1.
    # This saves (t_n - 1) subtractions.
    up_thresh += 1
    down_thresh += 1

    for t in range(1, t_n):
        x = X[t]
        r = x / last_pivot_x

        if trend == -1:
            if r >= up_thresh:
                pivots[last_pivot_t] = trend
                trend = PEAK
                last_pivot_x = x
                last_pivot_t = t
            elif x < last_pivot_x:
                last_pivot_x = x
                last_pivot_t = t
        else:
            if r <= down_thresh:
                pivots[last_pivot_t] = trend
                trend = VALLEY
                last_pivot_x = x
                last_pivot_t = t
            elif x > last_pivot_x:
                last_pivot_x = x
                last_pivot_t = t

    if last_pivot_t == t_n-1:
        pivots[last_pivot_t] = trend
    elif pivots[t_n-1] == 0:
        pivots[t_n-1] = -trend

    return pivots


def max_drawdown(X):
    """
    Compute the maximum drawdown of some sequence.

    :return: 0 if the sequence is strictly increasing.
        otherwise the abs value of the maximum drawdown
        of sequence X
    """
    mdd = 0
    peak = X[0]

    for x in X:
        if x > peak:
            peak = x

        dd = (peak - x) / peak

        if dd > mdd:
            mdd = dd

    return mdd if mdd != 0.0 else 0.0


def pivots_to_modes(pivots):
    """
    Translate pivots into trend modes.

    :param pivots: the result of calling ``peak_valley_pivots``
    :return: numpy array of trend modes. That is, between (VALLEY, PEAK] it
    is 1 and between (PEAK, VALLEY] it is -1.
    """

    modes = np.zeros(len(pivots), dtype=np.int_)
    mode = -pivots[0]

    modes[0] = pivots[0]

    for t in range(1, len(pivots)):
        x = pivots[t]
        if x != 0:
            modes[t] = mode
            mode = -x
        else:
            modes[t] = mode

    return modes


def compute_segment_returns(X, pivots):
    """
    :return: numpy array of the pivot-to-pivot returns for each segment."""
    pivot_points = X[pivots != 0]
    return pivot_points[1:] / pivot_points[:-1] - 1.0

使用示例:

import matplotlib
matplotlib.use("TkAgg")
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import sys
import pathlib
sys.path.append("%s/zigzag" % pathlib.Path().absolute())
from zigzag import zigzag


def plot_pivots(X, pivots):
    plt.xlim(0, len(X))
    plt.ylim(X.min()*0.99, X.max()*1.01)
    plt.plot(np.arange(len(X)), X, 'k:', alpha=0.5)
    plt.plot(np.arange(len(X))[pivots != 0], X[pivots != 0], 'k-')
    plt.scatter(np.arange(len(X))[pivots == 1], X[pivots == 1], color='g')
    plt.scatter(np.arange(len(X))[pivots == -1], X[pivots == -1], color='r')


np.random.seed(1997)
X = np.cumprod(1 + np.random.randn(100) * 0.01)
pivots = zigzag.peak_valley_pivots(X, 0.03, -0.03)

plot_pivots(X, pivots)
plt.show()

modes = zigzag.pivots_to_modes(pivots)
print(pd.Series(X).pct_change().groupby(modes).describe().unstack())
print(zigzag.compute_segment_returns(X, pivots))

pandas 的数据输入示例:

from pandas_datareader import get_data_yahoo

X = get_data_yahoo('GOOG')['Adj Close']
pivots = peak_valley_pivots(X.values, 0.2, -0.2)
ts_pivots = pd.Series(X, index=X.index)
ts_pivots = ts_pivots[pivots != 0]
X.plot()
ts_pivots.plot(style='g-o');

1.Zigzag3个参数https://github.com/twopirllc/pandas-ta

Zigzag在识别?低点的过程中, 主要设置了以下三个参数: ExtDepth, DextDeviation

以及ExtBackstep。 程序中的表?:

extern int ExtDepth=12;

extern int ExtDeviation=5;

extern int ExtBackstep=3;

说明:

ExtDepth ?于设置?低点是相对与过去多少个Bars(价格图形中的?个柱?)??。 Mt4中默认是12 ExtDeviation ?于设 置重新计算?低点时, 与前??低点的相对点差。 默认值是5 也就是说如果

A)当前?点>上个?点5 ,或者

B)当前低点<上个低点

5的情况下, 则会对之前计算过的ExtBackstepsBars值的?低点进?重新计算。

ExtBackstep ?于设置回退计算的Bars的个数。

2.Zigzag算法

1对计算位置进?初期化

1.1判断是否是第?次进??低点计算, 如果是, 则设定计算位置为除去ExtDepth个图形最初的部分。 1.2如果之前已经计算过, 找到最近已知的三个拐点 (?点或低点) , 将计算位置设置为倒数第三个拐点之后, 重新计 2.从步骤1已经设置好的计算位置开始, 将对?于存储?低点的变量进?初始化, 准备计算?低点 2.1计算ExtDepth区间内的低点, 如果该低点是当前低点, 则进?2.1.1的计算, 并将其记录成?个低点。

2.1.1如果当前低点?上?个低点值?于相对点差(ExtDeviation) 并且之前ExtBackstepBars的记录的中, ?于当前低点的 值清空。

2.2?点的计算如同2.1以及分?处理2.1.1

3.从步骤1已经设置好的计算位置开始, 定义指标?点和低点

3.1如果开始位置为?点, 则接下来寻找低点, 在找到低点之后, 将下?个寻找?标定义为?点

3.2如果开始位置为低点, 则与3.1反之。

以上可能?较难以理解, 我们这边举个例?说明:

假设上次计算的结果如下: 倒数第14Bar出现了?个?点(3.1) 倒数第4个是低点(1.5)

倒数第1个是新的?点(2.1)——因为距离倒数第14已经?于ExtDepth(14-1>12)

Bar-14Bar-4Bar-1 Bar-Current

?(3.1)(1.5)?(2.1) X

对于Bar-Current 即当前的价格X


CaseI.

如果X >=2.1

ExtDeviation 则根据Zigzag的定义, 这将是?个新的?点。 假设这?X=2.3 那么我们绘制指标的时候应该成为:

Bar-14 Bar-4Bar-Current

?(3.1)

(1.5)?(2.3)

CaseII.

如果1.5 - ExtDeviation<

X<2.1 ExtDeviation 则我们继续等待价格的变化, 所绘制的指标也不会变化。

CaseIII.

如果1.5 - ExtDeviation>=

X 则这是?个新的低点。 假设这?X=1.3 则我们绘制指标的时候应该成为:

Bar-14Bar-Current

?(3.1)(1.3)

这个时候, 之前的Bar-4因为在我们定义的ExtBackstep之内(1-4) 所以他的最低值会被清空,

根据算法第三步的定义, 我们会?直寻找低点直到发现Bar-Current 这时候已经遍历过Bar-1 所以Bar-1定义的? 点也不再成为拐点。

这也就是所谓的重绘部分, 也因此诟病为―未来函数‖——因为所看见的当前最后的?低点可能在下个时间段??被抹去。 3Zigzag源码及解释:

Mt4Zigzag源码??的注释特别稀罕, 估计是感觉实现?较简单, 所以?概略去——恩, 极坏的编程习惯。 下?简要说明?下, 中?部分都是追加的解释:

// ——————————————————————

//|

Zigzag.mq4 |

//|

Copyright ?2005-2007, MetaQuotes Software Corp. |

//|

http://www.doczj.com/doc/855cc57301f69e3143329458.html / |

// ——————————————————————

#property copyrightCopyright ?2007, MetaQuotes Software

Corp.

#property

link

http://www.doczj.com/doc/855cc57301f69e3143329458.html /

indicator_chart_window


//主窗?进?指标显?

#property indicator_buffers

1 //指标运?到数值的个数

#property indicator_color1

Red

//指标显?颜?

//—- indicator parameters

//Zigzag的三个参数

extern int ExtDepth=12;

extern int ExtDeviation=5;

extern int ExtBackstep=3;

//—- indicator buffers

//指标的数值存储变量

double

ZigzagBuffer[];

//拐点

double

HighMapBuffer[];

//?点的临时变量数组

double

LowMapBuffer[];

//低点的临时变量数组

int level=3; // recounting’s depth

//最近已知的三个拐点

bool downloadhistory=false; //是否第?次计算

// ——————————————————————//| Custom indicator initialization

function

|

// ——————————————————————

IndicatorBuffers(3);

//对于缓冲储存器分配记忆应??定义指标计算, ?F1可以看到该函数的帮助和解释//—- drawing settings SetIndexStyle(0,DRAW_SECTION);

//划线的风格

//—- indicator buffers mapping

SetIndexBuffer(0,ZigzagBuffer);


SetIndexBuffer(1,HighMapBuffer);

SetIndexBuffer(2,LowMapBuffer);

SetIndexEmptyValue(0,0.0);

//—- indicator short name

IndicatorShortName(ZigZag(

ExtDepth ,ExtDeviation,ExtBackstep

));

//设置指标的简称。

//—- initialization done

return(0);

}

// ——————————————————————

//|

|

// ——————————————————————

//start函数是Mt4的主函数, 当每次价格变动之后都会触发该函数的执?

int start()

{

//变量定义

//i 临时变量;

//limit 算法中所谓的开始计算位置;

//counterZ 临时变量

//whatlookfor ?于标识当前计算的是?点或者低点

int

limit,counterZ,whatlookfor;

//以下都是临时变量, 具体设值时解释

int

shift,back,lasthighpos,lastlowpos;

double val ,res;

double

curlow ,curhigh,lasthigh,lastlow;

if (counted_bars==0

&& downloadhistory) // history was

downloaded

{


//指标载?时counted_bars0 ?downloadhistoryfalse 将在下?次价格变化时进?ArrayInitialize(ZigzagBuffer,0.0); ArrayInitialize(HighMapBuffer,0.0);

ArrayInitialize(LowMapBuffer,0.0);

}

if (counted_bars==0)

{ //初期化, 第?次运?时limit为除去ExtDepth个图形最初的部分。 (算法1.1

limit=Bars-ExtDepth;

downloadhistory=true;

(counted_bars>0)

{//如果之前已经计算过, 找到最近已知的三个拐点 (?点或低点) , 将计算位置设置为倒数第三个拐点。 (算法1.2

while (counterZ

&& i<100)

{

res=ZigzagBuffer[i];

if (res!=0) counterZ ;

i ;

}

i– ; //在上?while中最后?次找到的时候进?

1 所以要-1才能得到真正第三个拐点处。

limit=i; //计算位置赋值

if (LowMapBuffer[i]!=0)

{//如果倒数第三个拐点是低点

curlow=LowMapBuffer[i];

//?标在于寻找?点

whatlookfor=1;

}

else

{

curhigh=HighMapBuffer[i];

}

for (i=limit-1;i>=0;i–)

{//清空第三个拐点后的数值, 准备重新计算最后的拐点

ZigzagBuffer[i]=0.0;

LowMapBuffer[i]=0.0;

HighMapBuffer[i]=0.0;


}

}

//算法Step2部分: 计算?低点

for(shift=limit;

shift>=0; shift–)

{

//2.1计算ExtDepth区间内的低点

val=Low[iLowest(NULL,0,MODE_LOW,ExtDepth,shift)];

if(val==lastlow) val=0.0;

else

{//如果该低点是当前低点,

lastlow=val;

if((Low[shift]-val)>(ExtDeviation*Point))

val=0.0; //是否?上个低点还低ExtDeviation 不是的话则不进?回归处理

for(back=1; back<=ExtBackstep; back )

{//回退ExtBackstepBar 把?当前低点?的纪录值给清空res=LowMapBuffer[shift back];

if((res!=0)&&(res>val))

LowMapBuffer[shift back]=0.0;

}

}

}

//将新的低点进?记录

if (Low[shift]==val) LowMapBuffer[shift]=val; else LowMapBuffer[shift]=0.0;

//— high

val=High[iHighest(NULL,0,MODE_HIGH ,ExtDepth,shift)];

if(val==lasthigh) val=0.0;

else

{

lasthigh=val;

if((val-High[shift])>(ExtDeviation*Point))

val=0.0;

else

for(back=1; back<=ExtBackstep; back )

{

res=HighMapBuffer[shift back];


if((res!=0)&&(res

HighMapBuffer[shift back]=0.0;

}

}

}

if (High[shift]==val) HighMapBuffer[shift]=val; else HighMapBuffer[shift]=0.0;

}

// final cutting

if (whatlookfor==0)

{

lastlow=0;

lasthigh=0;

}

else

{

lastlow=curlow;

lasthigh=curhigh;

//算法step3.定义指标的?低点

for

(shift=limit;shift>=0;shift–)

{

res=0.0;

switch(whatlookfor)

{

//初期化的情况下, 尝试找第?个?点或者是地点

case 0: // look for peak or lawn

if (lastlow==0 &&

lasthigh==0)

{//lastlow lasthigh之前已经初始化, 再次判断以保证正确性? if (HighMapBuffer[shift]!=0)

{//发现?点

lasthigh=High[shift];

lasthighpos=shift;

whatlookfor=-1; //下个寻找?标是低点

ZigzagBuffer[shift]=lasthigh;

res=1;


}

if (LowMapBuffer[shift]!=0)

lastlowpos=shift;

whatlookfor=1;

//下个寻找?标是?点

ZigzagBuffer[shift]=lastlow;

res=1;

}

}

break;

case 1: // look for

peak

//寻找?点

if (LowMapBuffer[shift]!=0.0 &&

LowMapBuffer[shift]

&& HighMapBuffer[shift]==0.0)

{//如果在上个低点和下个?点间发现新的低点, 则把上个低点抹去, 将新发现的低点作为最后?个低点

ZigzagBuffer[lastlowpos]=0.0;

lastlowpos=shift;

lastlow=LowMapBuffer[shift];

ZigzagBuffer[shift]=lastlow;

res=1;

}

if (HighMapBuffer[shift]!=0.0 &&

lasthigh=HighMapBuffer[shift];

lasthighpos=shift; ZigzagBuffer[shift]=lasthigh;

whatlookfor=-1;

//下?个?标将是寻找低点

res=1;

}

break;

case -1: // look for

lawn

//寻找低点

if (HighMapBuffer[shift]!=0.0 && HighMapBuffer[shift]>lasthigh && LowMapBuffer[shift]==0.0) {


ZigzagBuffer[lasthighpos]=0.0; lasthighpos=shift;

lasthigh=HighMapBuffer[shift]; ZigzagBuffer[shift]=lasthigh;

}

if (LowMapBuffer[shift]!=0.0 && HighMapBuffer[shift]==0.0)

lastlow=LowMapBuffer[shift];

lastlowpos=shift;

ZigzagBuffer[shift]=lastlow;

whatlookfor=1;

}

break;

default: return;

}

}

return(0);

}

// ——————————————————————

4.总结

以上就是对Zigzag算法和实现的分析。 希望能够对?家编写指标和EA有所帮助。

相关推荐

selenium(WEB自动化工具)

定义解释Selenium是一个用于Web应用程序测试的工具。Selenium测试直接运行在浏览器中,就像真正的用户在操作一样。支持的浏览器包括IE(7,8,9,10,11),MozillaF...

开发利器丨如何使用ELK设计微服务中的日志收集方案?

【摘要】微服务各个组件的相关实践会涉及到工具,本文将会介绍微服务日常开发的一些利器,这些工具帮助我们构建更加健壮的微服务系统,并帮助排查解决微服务系统中的问题与性能瓶颈等。我们将重点介绍微服务架构中...

高并发系统设计:应对每秒数万QPS的架构策略

当面试官问及"如何应对每秒几万QPS(QueriesPerSecond)"时,大概率是想知道你对高并发系统设计的理解有多少。本文将深入探讨从基础设施到应用层面的解决方案。01、理解...

2025 年每个 JavaScript 开发者都应该了解的功能

大家好,很高兴又见面了,我是"高级前端进阶",由我带着大家一起关注前端前沿、深入前端底层技术,大家一起进步,也欢迎大家关注、点赞、收藏、转发。1.Iteratorhelpers开发者...

JavaScript Array 对象

Array对象Array对象用于在变量中存储多个值:varcars=["Saab","Volvo","BMW"];第一个数组元素的索引值为0,第二个索引值为1,以此类推。更多有...

Gemini 2.5编程全球霸榜,谷歌重回AI王座,神秘模型曝光,奥特曼迎战

刚刚,Gemini2.5Pro编程登顶,6美元性价比碾压Claude3.7Sonnet。不仅如此,谷歌还暗藏着更强的编程模型Dragontail,这次是要彻底翻盘了。谷歌,彻底打了一场漂亮的翻...

动力节点最新JavaScript教程(高级篇),深入学习JavaScript

JavaScript是一种运行在浏览器中的解释型编程语言,它的解释器被称为JavaScript引擎,是浏览器的一部分,JavaScript广泛用于浏览器客户端编程,通常JavaScript脚本是通过嵌...

一文看懂Kiro,其 Spec工作流秒杀Cursor,可移植至Claude Code

当Cursor的“即兴编程”开始拖累项目质量,AWS新晋IDEKiro以Spec工作流打出“先规范后编码”的系统工程思维:需求-设计-任务三件套一次生成,文档与代码同步落地,复杂项目不...

「晚安·好梦」努力只能及格,拼命才能优秀

欢迎光临,浏览之前点击上面的音乐放松一下心情吧!喜欢的话给小编一个关注呀!Effortscanonlypass,anddesperatelycanbeexcellent.努力只能及格...

JavaScript 中 some 与 every 方法的区别是什么?

大家好,很高兴又见面了,我是姜茶的编程笔记,我们一起学习前端相关领域技术,共同进步,也欢迎大家关注、点赞、收藏、转发,您的支持是我不断创作的动力在JavaScript中,Array.protot...

10个高效的Python爬虫框架,你用过几个?

小型爬虫需求,requests库+bs4库就能解决;大型爬虫数据,尤其涉及异步抓取、内容管理及后续扩展等功能时,就需要用到爬虫框架了。下面介绍了10个爬虫框架,大家可以学习使用!1.Scrapysc...

12个高效的Python爬虫框架,你用过几个?

实现爬虫技术的编程环境有很多种,Java、Python、C++等都可以用来爬虫。但很多人选择Python来写爬虫,为什么呢?因为Python确实很适合做爬虫,丰富的第三方库十分强大,简单几行代码便可实...

pip3 install pyspider报错问题解决

运行如下命令报错:>>>pip3installpyspider观察上面的报错问题,需要安装pycurl。是到这个网址:http://www.lfd.uci.edu/~gohlke...

PySpider框架的使用

PysiderPysider是一个国人用Python编写的、带有强大的WebUI的网络爬虫系统,它支持多种数据库、任务监控、项目管理、结果查看、URL去重等强大的功能。安装pip3inst...

「机器学习」神经网络的激活函数、并通过python实现激活函数

神经网络的激活函数、并通过python实现whatis激活函数感知机的网络结构如下:左图中,偏置b没有被画出来,如果要表示出b,可以像右图那样做。用数学式来表示感知机:上面这个数学式子可以被改写:...