Explore o novo chatbot do Developer Center! O MongoDB AI chatbot pode ser acessado na parte superior da sua navegação para responder a todas as suas perguntas sobre o MongoDB .

Saiba por que o MongoDB foi selecionado como um líder no 2024 Gartner_Magic Quadrupnt()
Desenvolvedor do MongoDB
Centro de desenvolvedores do MongoDB
chevron-right
Produtos
chevron-right
MongoDB
chevron-right

Análise de moeda com coleções de séries temporais # 3 — Cálculo de MACD e RSI

Fuat Sungur8 min read • Published Aug 27, 2021 • Updated Sep 11, 2024
MongoDBTime SeriesJavaScript
Ícone do FacebookÍcone do Twitterícone do linkedin
Avalie esse Artigo
star-empty
star-empty
star-empty
star-empty
star-empty
No primeiro post desta série, aprenderam como agrupar dados de moeda com base em determinados intervalos de tempo para gerar gráficos Atlas de velas. No segundo artigo, aprenderam como calcular a média móvel simples e a média móvel exponencial nas moedas com base em uma determinada janela de tempo. Agora, nesta postagem, aprenderemos a calcular indicadores técnicos mais complexos.

Indicador MacD

O MACD (Moving Average Convergence Divergence) é outro indicador de negociação e fornece visibilidade da tendência e do momentum da moeda/ação. O cálculo do MACD utiliza fundamentalmente vários cálculos deEMA com parâmetros diferentes.
Conforme mostrado no diagrama abaixo, o indicador MACD tem três componentes principais: Linha MACD, Sinal MACD e Histograma. (A linha azul representa a linha MACD, a linha vermelha representa o sinal MACD e as barras verde e vermelha representam o histograma):
Macd histograma
  • A linha MACD é calculada subtraindo a média móvel exponencial do período de 26(principalmente, os dias são usados para o período) da média móvel exponencial do período de 12.
  • Depois de obter a Linha MacD, podemos calcular o Sinal macd. O sinal MACD é calculado obtendo a média móvel exponencial de nove períodos da Linha MACD.
  • O histograma MACD é calculado subtraindo-se o sinal MACD da linha MACD.
Podemos usar o MongoDB Aggregation Framework para calcular esse indicador complexo.
Nas postagens anteriores do blog, vimos como podemos agrupar os dados brutos de segundo nível em intervalos de cinco minutos por meio do estágio$group e do operador$dateTrunc :
1db.ticker.aggregate([
2 {
3 $match: {
4 symbol: "BTC-USD",
5 },
6 },
7 {
8 $group: {
9 _id: {
10 symbol: "$symbol",
11 time: {
12 $dateTrunc: {
13 date: "$time",
14 unit: "minute",
15 binSize: 5,
16 },
17 },
18 },
19 high: { $max: "$price" },
20 low: { $min: "$price" },
21 open: { $first: "$price" },
22 close: { $last: "$price" },
23 },
24 },
25 {
26 $sort: {
27 "_id.time": 1,
28 },
29 },
30 {
31 $project: {
32 _id: 1,
33 price: "$close",
34 },
35 }
36]);
Depois disso, precisamos calcular duas médias móveis exponenciais com parâmetros diferentes:
1{
2 $setWindowFields: {
3 partitionBy: "_id.symbol",
4 sortBy: { "_id.time": 1 },
5 output: {
6 ema_12: {
7 $expMovingAvg: { input: "$price", N: 12 },
8 },
9 ema_26: {
10 $expMovingAvg: { input: "$price", N: 26 },
11 },
12 },
13 },
14}
Depois de calcularmos as duas médias móveis exponenciais separadas, precisamos aplicar a operação$subtract na próxima etapa da aggregation pipeline:
1{ $addFields : {"macdLine" : {"$subtract" : ["$ema_12", "$ema_26"]}}}
Depois de obter o campomacdLine, podemos aplicar outra média móvel exponencial a esse campo recém-gerado (macdLine) para obter o valor do sinal MacD:
1{
2 $setWindowFields: {
3 partitionBy: "_id.symbol",
4 sortBy: { "_id.time": 1 },
5 output: {
6 macdSignal: {
7 $expMovingAvg: { input: "$macdLine", N: 9 },
8 },
9 },
10 },
11}
Portanto, teremos mais dois campos: macdLine e macdSignal. Podemos gerar outro campo como macdHistogram que é calculado subtraindo macdSignal do valormacdLine:
1{ $addFields : {"macdHistogram" : {"$subtract" : ["$macdLine", "$macdSignal"]}}}
Agora temos três campos derivados: macdLine, macdSignale macdHistogram. Abaixo, você pode ver como o MacD é visualizado junto com as velas:
Gráficos de velas
Este é o pipeline de agregação completo:
1db.ticker.aggregate([
2 {
3 $match: {
4 symbol: "BTC-USD",
5 },
6 },
7 {
8 $group: {
9 _id: {
10 symbol: "$symbol",
11 time: {
12 $dateTrunc: {
13 date: "$time",
14 unit: "minute",
15 binSize: 5,
16 },
17 },
18 },
19 high: { $max: "$price" },
20 low: { $min: "$price" },
21 open: { $first: "$price" },
22 close: { $last: "$price" },
23 },
24 },
25 {
26 $sort: {
27 "_id.time": 1,
28 },
29 },
30 {
31 $project: {
32 _id: 1,
33 price: "$close",
34 },
35 },
36 {
37 $setWindowFields: {
38 partitionBy: "_id.symbol",
39 sortBy: { "_id.time": 1 },
40 output: {
41 ema_12: {
42 $expMovingAvg: { input: "$price", N: 12 },
43 },
44 ema_26: {
45 $expMovingAvg: { input: "$price", N: 26 },
46 },
47 },
48 },
49 },
50 { $addFields: { macdLine: { $subtract: ["$ema_12", "$ema_26"] } } },
51 {
52 $setWindowFields: {
53 partitionBy: "_id.symbol",
54 sortBy: { "_id.time": 1 },
55 output: {
56 macdSignal: {
57 $expMovingAvg: { input: "$macdLine", N: 9 },
58 },
59 },
60 },
61 },
62 {
63 $addFields: { macdHistogram: { $subtract: ["$macdLine", "$macdSignal"] } },
64 },
65]);

Indicador RSI

O RSI (Relativity Strength Index) é outro indicador técnico financeiro que revela se o ativo foi sobrecomprado ou sobrevendido. Geralmente, ele usa uma janela de período 14, e o valor do RSI é medido em uma escala de 0 a 100. Se o valor estiver mais próximo de 100, isso indica que o ativo foi sobrecomprado nesse período. E se o valor estiver mais próximo de 0, isso indica que o ativo foi sobrevendido nesse período. Principalmente, 70 e 30 são usados para limites superiores e inferiores.
O cálculo do RSI é um pouco mais complicado do que o MacD:
  • Para cada ponto de dados, os valores de ganho e perda são definidos comparando um ponto de dados anterior.
  • Depois de definirmos os valores de ganho e perda para cada ponto de dados, podemos obter uma média móvel de ganho e perda para um período 14. (Você não precisa aplicar um período 14 . O que quer que funcione para você, você pode definir de acordo.)
  • Depois de obtermos o ganho médio e o valor médio da perda, podemos dividir o ganho médio pela perda média.
  • Depois disso, podemos suavizar o valor para normalizá-lo entre 0 e 100.

Calculando ganhos e perdas

Em primeiro lugar, precisamos definir o valor de ganho e de perda para cada intervalo.
O valor de ganho e perda é calculado subtraindo uma informação de preço anterior das informações de preço atual:
  • Se a diferença for positiva, significa que há um aumento de preço e o valor do ganho será a diferença entre o preço atual e o preço anterior. O valor da perda será 0.
  • Se a diferença for negativa, significa que há uma queda no preço e o valor da perda será a diferença entre o preço anterior e o preço atual. O valor do ganho será 0.
Considere o seguinte conjunto de dados de entrada:
1{"_id": {"time": ISODate("20210101T17:00:00"), "symbol" : "BTC-USD"}, "price": 35050}
2{"_id": {"time": ISODate("20210101T17:05:00"), "symbol" : "BTC-USD"}, "price": 35150}
3{"_id": {"time": ISODate("20210101T17:10:00"), "symbol" : "BTC-USD"}, "price": 35280}
4{"_id": {"time": ISODate("20210101T17:15:00"), "symbol" : "BTC-USD"}, "price": 34910}
Depois de calcular o ganho e a perda, teremos os seguintes dados:
1{"_id": {"time": ISODate("20210101T17:00:00"), "symbol" : "BTC-USD"}, "price": 35050, "previousPrice": null, "gain":0, "loss":0}
2{"_id": {"time": ISODate("20210101T17:05:00"), "symbol" : "BTC-USD"}, "price": 35150, "previousPrice": 35050, "gain":100, "loss":0}
3{"_id": {"time": ISODate("20210101T17:10:00"), "symbol" : "BTC-USD"}, "price": 35280, "previousPrice": 35150, "gain":130, "loss":0}
4{"_id": {"time": ISODate("20210101T17:15:00"), "symbol" : "BTC-USD"}, "price": 34910, "previousPrice": 35280, "gain":0, "loss":370}
Mas no MongoDB Aggregation Pipeline, como podemos nos referir ao documento anterior a partir do documento atual? Como podemos derivar o novo campo ($previousPrice) do documento anterior na janela classificada?
MongoDB 5.0 introduziu o operador$shift que inclui dados de outro documento na mesma partição no local fornecido, por exemplo, você pode se referir ao documento que é três documentos antes do documento atual ou dois documentos após o documento atual na classificação janela.
Definimos nossa janela com particionamento e introduzimos um novo campo como previousPrice:
1{
2 $setWindowFields: {
3 partitionBy: "$_id.symbol",
4 sortBy: { "_id.time": 1 },
5 output: {
6 previousPrice: { $shift: { by: -1, output: "$price" } },
7 },
8 },
9}
$shift usa dois parâmetros:
  • by especifica o local do documento que incluiremos. Como queremos incluir o documento anterior, o configuramos como -1. Se quisermos incluir um próximo documento, o definiremos como 1.
  • output Especifica o campo do documento que queremos incluir no documento atual.
Depois de definir as informações$previousPrice para o documento atual, precisamos subtrair o valor anterior do valor atual. Vamos ter outro campo derivado "diff " que representa o valor de diferença entre o valor atual e o valor anterior:
1{
2 $addFields: {
3 diff: {
4 $subtract: ["$price", { $ifNull: ["$previousPrice", "$price"] }],
5 },
6 },
7}
Definimos o valordiff e agora definiremos mais dois campos, gain e loss, para usar nos estágios seguintes. Apenas aplicamos a lógica de ganho/perda aqui:
1{
2 $addFields: {
3 gain: { $cond: { if: { $gte: ["$diff", 0] }, then: "$diff", else: 0 } },
4 loss: {
5 $cond: { if: { $lte: ["$diff", 0] }, then: { $abs: "$diff" }, else: 0 },
6 },
7 },
8}
Depois de enriquecermos os dados do símbolo com informações de ganho e perda para cada documento, podemos aplicar mais partições para obter a média móvel dos campos de ganho e perda considerando os pontos de dados 14 anteriores:
1{
2 $setWindowFields: {
3 partitionBy: "$_id.symbol",
4 sortBy: { "_id.time": 1 },
5 output: {
6 avgGain: {
7 $avg: "$gain",
8 window: { documents: [-14, 0] },
9 },
10 avgLoss: {
11 $avg: "$loss",
12 window: { documents: [-14, 0] },
13 },
14 documentNumber: { $documentNumber: {} },
15 },
16 },
17}
Aqui também usamos outro operador recém-introduzido, $documentNumber. Enquanto faço cálculos na janela, fornecemos um número sequencial para cada documento, porque filtraremos os documentos que têm o número do documento menor ou igual a 14. (O RSI é calculado após a chegada de pelo menos 14 pontos de dados.) Faremos a filtragem nos estágios posteriores. Aqui, definimos apenas o número do documento.
Depois de calcularmos o ganho médio e a perda média para cada símbolo, encontraremos o valor da força relativa. Isso é calculado dividindo-se o valor médio de ganho pelo valor médio de perda. Uma vez que aplicamos a operação de divisão, precisamos antecipar o problema da "divide by 0" também:
1{
2 $addFields: {
3 relativeStrength: {
4 $cond: {
5 if: {
6 $gt: ["$avgLoss", 0],
7 },
8 then: {
9 $divide: ["$avgGain", "$avgLoss"],
10 },
11 else: "$avgGain",
12 },
13 },
14 },
15}
O valor da força relativa foi calculado e agora é hora de suavizar o valor da Força relativa para normalizar os dados entre 0 e 100:
1{
2 $addFields: {
3 rsi: {
4 $cond: {
5 if: { $gt: ["$documentNumber", 14] },
6 then: {
7 $subtract: [
8 100,
9 { $divide: [100, { $add: [1, "$relativeStrength"] }] },
10 ],
11 },
12 else: null,
13 },
14 },
15 },
16}
Basicamente, definimos null para os primeiros 14 documentos. E para os outros, o valor RSI foi definido.
Abaixo, você pode ver um gráfico de velas de intervalo de um minuto e um gráfico RSI. Após 14 pontos de dados, o RSI começa a ser calculado. Para cada intervalo, calculamos o RSI por meio de queries de agregação processando os dados anteriores desse símbolo:
Gráficos de velas
Este é o pipeline de agregação completo:
1db.ticker.aggregate([
2 {
3 $match: {
4 symbol: "BTC-USD",
5 },
6 },
7 {
8 $group: {
9 _id: {
10 symbol: "$symbol",
11 time: {
12 $dateTrunc: {
13 date: "$time",
14 unit: "minute",
15 binSize: 5,
16 },
17 },
18 },
19 high: { $max: "$price" },
20 low: { $min: "$price" },
21 open: { $first: "$price" },
22 close: { $last: "$price" },
23 },
24 },
25 {
26 $sort: {
27 "_id.time": 1,
28 },
29 },
30 {
31 $project: {
32 _id: 1,
33 price: "$close",
34 },
35 },
36 {
37 $setWindowFields: {
38 partitionBy: "$_id.symbol",
39 sortBy: { "_id.time": 1 },
40 output: {
41 previousPrice: { $shift: { by: -1, output: "$price" } },
42 },
43 },
44 },
45 {
46 $addFields: {
47 diff: {
48 $subtract: ["$price", { $ifNull: ["$previousPrice", "$price"] }],
49 },
50 },
51 },
52 {
53 $addFields: {
54 gain: { $cond: { if: { $gte: ["$diff", 0] }, then: "$diff", else: 0 } },
55 loss: {
56 $cond: { if: { $lte: ["$diff", 0] }, then: { $abs: "$diff" }, else: 0 },
57 },
58 },
59 },
60 {
61 $setWindowFields: {
62 partitionBy: "$_id.symbol",
63 sortBy: { "_id.time": 1 },
64 output: {
65 avgGain: {
66 $avg: "$gain",
67 window: { documents: [-14, 0] },
68 },
69 avgLoss: {
70 $avg: "$loss",
71 window: { documents: [-14, 0] },
72 },
73 documentNumber: { $documentNumber: {} },
74 },
75 },
76 },
77 {
78 $addFields: {
79 relativeStrength: {
80 $cond: {
81 if: {
82 $gt: ["$avgLoss", 0],
83 },
84 then: {
85 $divide: ["$avgGain", "$avgLoss"],
86 },
87 else: "$avgGain",
88 },
89 },
90 },
91 },
92 {
93 $addFields: {
94 rsi: {
95 $cond: {
96 if: { $gt: ["$documentNumber", 14] },
97 then: {
98 $subtract: [
99 100,
100 { $divide: [100, { $add: [1, "$relativeStrength"] }] },
101 ],
102 },
103 else: null,
104 },
105 },
106 },
107 },
108]);

Conclusão

O MongoDB Aggregation Framework fornece um ótimo conjunto de ferramentas para transformar qualquer formato de dados no formato desejado. Como você pode ver nos exemplos, usamos uma grande variedade de estágios e operadores de pipeline de agregação. Conforme discutimos nas postagens anteriores do blog, coleções de séries temporais e funções de janela são ótimas ferramentas para processar dados baseados em tempo em uma janela.
Nesta postagem, examinamos os operadores $shift e $documentNumber que foram introduzidos com o MongoDB 5.0. O operador$shift inclui outro documento na mesma janela do documento atual para processar dados posicionais junto com dados atuais. Em um cálculo de indicador técnico RSI, ele é comumente usado para comparar o ponto de dados atual com os pontos de dados anteriores, e $shift facilita a referência a documentos posicionais em uma janela. Por exemplo, diferença de preço entre o ponto de dados atual e o ponto de dados anterior.
Outro operador recém-introduzido é $documentNumber. $documentNumber fornece um número sequencial para que os documentos classificados sejam processados posteriormente nos estágios subsequentes de agregação. Em um cálculo de RSI, precisamos pular o cálculo do valor de RSI para os primeiros períodos de dados 14 e $documentNumber nos ajuda a identificar e filtrar esses documentos em estágios posterioresdo pipeline de agregação.

Ícone do FacebookÍcone do Twitterícone do linkedin
Avalie esse Artigo
star-empty
star-empty
star-empty
star-empty
star-empty
Relacionado
Podcast

Na interseção de AI/ML e HCI com Dogulas Eck do Google (MongoDB Podcast)


Aug 14, 2024 | 30 min
Artigo

Mapeando Termos e Conceitos do SQL para o MongoDB


Oct 01, 2024 | 15 min read
Artigo

TableCheck: restaurantes habilitados com as melhores ferramentas de reserva da classe, desenvolvido pela MongoDB


Aug 28, 2024 | 4 min read
Artigo

Dados da primavera desbloqueados: técnicas de otimização de desempenho com o MongoDB


Dec 04, 2024 | 5 min read
Sumário
  • Indicador MacD