Dan*_*ieu 13 c# technical-indicator
我正在编写一个小型技术分析库,其中包含TA-lib中不可用的项目。我从在cTrader上找到的示例开始,并将其与TradingView版本中的代码进行了匹配。
这是TradingView 的Pine Script代码:
len = input(9, minval=1, title="Length")
high_ = highest(hl2, len)
low_ = lowest(hl2, len)
round_(val) => val > .99 ? .999 : val < -.99 ? -.999 : val
value = 0.0
value := round_(.66 * ((hl2 - low_) / max(high_ - low_, .001) - .5) + .67 * nz(value[1]))
fish1 = 0.0
fish1 := .5 * log((1 + value) / max(1 - value, .001)) + .5 * nz(fish1[1])
fish2 = fish1[1]
Run Code Online (Sandbox Code Playgroud)
这是我尝试实现指标的尝试:
public class FisherTransform : IndicatorBase
{
public int Length = 9;
public decimal[] Fish { get; set; }
public decimal[] Trigger { get; set; }
decimal _maxHigh;
decimal _minLow;
private decimal _value1;
private decimal _lastValue1;
public FisherTransform(IEnumerable<Candle> candles, int length)
: base(candles)
{
Length = length;
RequiredCount = Length;
_lastValue1 = 1;
}
protected override void Initialize()
{
Fish = new decimal[Series.Length];
Trigger = new decimal[Series.Length];
}
public override void Compute(int startIndex = 0, int? endIndex = null)
{
if (endIndex == null)
endIndex = Series.Length;
for (int index = 0; index < endIndex; index++)
{
if (index == 1)
{
Fish[index - 1] = 1;
}
_minLow = Series.Average.Lowest(Length, index);
_maxHigh = Series.Average.Highest(Length, index);
_value1 = Maths.Normalize(0.66m * ((Maths.Divide(Series.Average[index] - _minLow, Math.Max(_maxHigh - _minLow, 0.001m)) - 0.5m) + 0.67m * _lastValue1));
_lastValue1 = _value1;
Fish[index] = 0.5m * Maths.Log(Maths.Divide(1 + _value1, Math.Max(1 - _value1, .001m))) + 0.5m * Fish[index - 1];
Trigger[index] = Fish[index - 1];
}
}
}
Run Code Online (Sandbox Code Playgroud)
问题
输出值似乎在预期范围内,但是我的Fisher变换交叉点与我在TradingView版本的指标上看到的不匹配。
题
如何在C#中正确实现Fisher变换指示器?我希望这与TradingView的Fisher转换输出匹配。
我知道的
我已根据我亲自编写的其他指标和TA-Lib的指标检查了我的数据,这些指标通过了单元测试。我还按照蜡烛对TradingView数据进行了蜡烛检查,发现我的数据符合预期。所以我不怀疑我的数据是问题。
下图是上面显示的应用于TradingView图表的Fisher转换代码。我的目标是尽可能匹配此输出。
费舍尔 青色 触发器 洋红色
预期产出:
跨界比赛于美国东部时间15:30完成
大约Fisher值是2.86
近似触发值为1.79
跨界比赛于美国东部时间10:45完成
大约Fisher值是-3.67
大约触发值为-3.10
我的实际输出:
跨界比赛于美国东部时间15:30完成
我的费舍尔价值是1.64
我的触发值为1.99
跨界比赛于美国东部时间10:45完成
我的费舍尔价值是-1.63
我的触发值为-2.00
为了使您的生活更轻松,我提供了一个小型控制台应用程序,该应用程序包含通过和失败的单元测试。所有单元测试均针对相同的数据集进行。通过的单元测试来自经过测试的工作 简单移动平均线指标。失败的单元测试与有问题的Fisher变换指示器相对。
项目文件 (5/14更新)
帮助我的FisherTransform测试通过,我将颁发赏金。
如果您需要任何其他资源或信息,请发表评论。
我会考虑的替代答案
用C#提交您自己的工作FisherTransform
说明为什么我的FisherTransform实际上按预期工作
该代码有两个错误。
1)错误的额外括号。正确的行是:
_value1 = Maths.Normalize(0.66m * (Maths.Divide(Series.Average[index] - _minLow, Math.Max(_maxHigh - _minLow, 0.001m)) - 0.5m) + 0.67m * _lastValue1);
Run Code Online (Sandbox Code Playgroud)
2)最小和最大函数必须为:
public static decimal Highest(this decimal[] series, int length, int index)
{
var maxVal = series[index]; // <----- HERE WAS AN ERROR!
var lookback = Math.Max(index - length, 0);
for (int i = index; i-- > lookback;)
maxVal = Math.Max(series[i], maxVal);
return maxVal;
}
public static decimal Lowest(this decimal[] series, int length, int index)
{
var minVal = series[index]; // <----- HERE WAS AN ERROR!
var lookback = Math.Max(index - length, 0);
for (int i = index; i-- > lookback;)
{
//if (series[i] != 0) // <----- HERE WAS AN ERROR!
minVal = Math.Min(series[i], minVal);
}
return minVal;
}
Run Code Online (Sandbox Code Playgroud)
3)混淆测试参数。请重新检查您的单元测试值。更新测试之后仍未修复。例如,第一个FisherTransforms_ValuesAreReasonablyClose_First()具有混合值
var fish = result.Fish.Last(); //is equal to -3.1113144510775780365063063706
var trig = result.Trigger.Last(); //is equal to -3.6057793808025449204415435710
// TradingView Values for NFLX 5m chart at 10:45 ET
var fisherValue = -3.67m;
var triggerValue = -3.10m;
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
412 次 |
| 最近记录: |