Всем привет! В этом уроке мы научимся создавать самый простой индикатор для МТ4 — Скользящую среднюю (Moving Averages), рассчитывать ее значения, сохранять данные в буфер и отображать на графике. Поехали.
Создаем индикатор Скользящая средняя
Зажимаем Ctrl+N и создаем новый пользовательский индикатор. Имя можно начать вводить по номеру урока — 2.5, остальную информацию об авторе по желанию. Далее. Дополнительные функции нам не нужны, жмем Далее. Отображение должно быть на графике, галочку на подвальный индикатор не ставим. Буферы мы лучше создадим вручную, поэтому отрисовку задавать не нужно. Но хочу заметить, что с помощью этой функции можно проделать ту же работу с буферами индикатора, которую я опишу ниже, просто в качестве объяснения удобнее и проще все сделать вручную в данном случае.
Данные свойств индикатора, отображаемые сверху программного кода, будут иметь вид:
0 1 2 3 4 5 6 7 8 9 |
//+------------------------------------------------------------------+ //| 2.5 Our Moving Averages.mq4 | //| Copyright (c) DaVinci FX Group | //| https://www.davinci-fx.com/ | //+------------------------------------------------------------------+ #property copyright "Copyright (c) DaVinci FX Group" #property link "https://www.davinci-fx.com/" #property version "1.00" #property strict #property indicator_chart_window |
Так как одна из целей этого урока познакомить вас с буферами индикатора, которые содержат все числовые данные индикатора, то создадим два буфера. Один будет выполнять отрисовку скользящей средней красного цвета, а второй зеленого для тех участков, где цена выше линии MA.
Поэтому добавляем ниже новое свойство индикатора indicator_buffers, которое определяет количество буферов.
0 |
#property indicator_buffers 2 |
Теперь введем внешние переменные для настройки индикатора: непосредственно период для нашей скользящей и сдвиг на указанное число баров вправо относительно графика.
0 1 |
extern int MA_Period = 50; extern int MA_Shift = 0; |
Буфер индикатора должен где-то хранить данные. Для этого нужно создать пользовательский массив, который будет передавать данные в буфер. В нашем случае нужно два массива для двух буферов. Так как мы не может знать количество свечей, которые будут проанализированы, то массив для буферов всегда задается динамическим.
0 1 |
double MABuffer1[]; //Буфер индикатора double MABuffer2[]; |
В итоге вне функций наш код будет иметь вид:
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
//+------------------------------------------------------------------+ //| 2.5 Our Moving Averages | //| Copyright (c) DaVinci FX Group | //| https://www.davinci-fx.com/ | //+------------------------------------------------------------------+ #property copyright "Copyright (c) DaVinci FX Group" #property link "https://www.davinci-fx.com/" #property version "1.00" #property strict #property indicator_chart_window #property indicator_buffers 2 extern int MA_Period = 50; extern int MA_Shift = 0; double MABuffer1[]; double MABuffer2[]; |
Переходим к функции обработки событий OnInit. При инициализации мы должны задать необходимые настройки для буферов индикатора, такие как тип линии, толщина, цвет, имя и т.д.
Сначала нам необходимо сделать проверку на корректное значение внешней настройки периода MA_Period. Так как для построения индикатора скользящая средняя нужно знать значения минимум трех свечей, то выполним проверку и если индикатор ее не пройдет — прерываем его работу.
0 1 2 3 |
if(MA_Period <= 2) { Print("Ошибка: Период индикатора должен быть больше 2"); return(INIT_FAILED); } |
Значения буферов индикатора отображаются в окне «Окно Данных» (Ctrl+D). Так как точность скользящей средней по количеству знаков после запятой должна быть аналогична точности цены на графике, то укажем формат Digits соответствующий текущему символу:
0 |
IndicatorDigits(Digits); |
Теперь нужно настроить стиль отображения линии скользящей средней для каждого буфера. Для этого воспользуемся функцией SetIndexStyle, которая имеет пять параметров:
- Порядковый номер буфера, для которого эти свойства задаются. Отсчет начинается с нуля.
- Тип отрисовки линии на графике: простая линия, отрезок, гистограмма, стрелки и т.д. В нашем случае это простая линия DRAW_LINE.
- Стиль отображения созданной линии: сплошная, штриховая, пунктирная, штрих-пунктирная и т.д. Наша линия будет сплошной STYLE_SOLID.
- Толщина линии. Действует только для сплошных линий. Вы не можете задать толщину для пунктирной или штриховой линии, она должна быть равна единице.
- Цвет линии. Линия не может быть радужной, ей задается один конкретный цвет на всю ее длину.
0 |
SetIndexStyle(0,DRAW_LINE,STYLE_SOLID,2,clrRed); |
0 |
SetIndexStyle(1,DRAW_LINE,STYLE_SOLID,2,clrGreen); |
0 |
string ind_name = "SMA("+IntegerToString(MA_Period)+")"; |
0 1 |
SetIndexLabel(0,ind_name); //назначение имени буферам SetIndexLabel(1,ind_name + "(Green)"); |
Как мы помним, во внешние настройки мы указали значение сдвига линии индикатора относительно графика. Значит нужно его назначить индикатору для каждого индекса отдельно.
0 1 |
SetIndexShift(0,MA_Shift); SetIndexShift(1,MA_Shift); |
Последнее, что осталось сделать в функции инициализации — назначить каждому буферу свой массив, в котором будут храниться данные индикатора. Массивы уже были созданы, осталось их связать с буфером.
0 1 |
SetIndexBuffer(0,MABuffer1); SetIndexBuffer(1,MABuffer2); |
В итоге вся функция OnInit будет выглядеть так:
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
int OnInit() { if(MA_Period <= 2) { Print("Ошибка: Период индикатора должен быть больше 2"); return(INIT_FAILED); } IndicatorDigits(Digits); SetIndexStyle(0,DRAW_LINE,STYLE_SOLID,2,clrRed); SetIndexStyle(1,DRAW_LINE,STYLE_SOLID,2,clrGreen); string ind_name = "SMA("+IntegerToString(MA_Period)+")"; SetIndexLabel(0,ind_name); SetIndexLabel(1,ind_name + "(Green)"); SetIndexShift(0,MA_Shift); SetIndexShift(1,MA_Shift); SetIndexBuffer(0,MABuffer1); SetIndexBuffer(1,MABuffer2); return(INIT_SUCCEEDED); } |
Переходим к функции обработки событий OnCalculate. В первую очень мы должны выполнить проверку на количество загруженных баров в истории, чтобы их было больше, чем заданный период скользящей средней.
0 |
if(rates_total <= MA_Period) return(0); |
Соответственно индикатор не будет работать, пока история не подгрузится. Далее нам необходимо рассчитать количество свеч, по которым нужно стоить индикатор. Суть идеи в том, что на первом тике после установки индикатора мы должны построить линию скользящей на протяжении всей истории котировок. На втором и последующих тиках уже не нужно делать перерасчет на истории, анализу должны подвергаться новые свечи, которые со временем будут появляться. Т.е. изначально индикатор проанализирует все доступные бары, а дальше будет обновляться только при появлении новых. Этот метод сократит количество итераций и нагрузку на сам индикатор.
Для этого воспользуемся двумя первыми параметрами функции OnCalculate: rates_total и prev_calculated. Подробнее их мы рассматривали на уроке 2.0 Пишем первый Индикатор.
0 1 |
int limit = rates_total-prev_calculated-1; if(prev_calculated > 0) limit++; |
С помощью этих строк было определено сколько баров нуждается в анализе и записано в переменную limit. На первом тике их количество равнялось всем барам на истории, а на последующих уже -1 или 0. Если тик не первый, то значение limit увеличивается на единицу.
Осталось запустить пользовательскую функцию для расчета и построения SMA. Отдельная функция нужна для удобства чтения кода. Плюс, в данном примере мы будем рассматривать построение только простой скользящей средней, потому что принцип работы встроенного индикатора не наша первоначальная цель. Для других типов скользящей нужны другие пользовательские функции.
В итоге функция OnCalculate будет иметь следующий вид:
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
int OnCalculate(const int rates_total, const int prev_calculated, const datetime &time[], const double &open[], const double &high[], const double &low[], const double &close[], const long &tick_volume[], const long &volume[], const int &spread[]) { if(rates_total <= MA_Period) return(0); int limit = rates_total-prev_calculated-1; if(prev_calculated > 0) limit++; if(limit > 0) sma(limit); return(rates_total); } |
Нам осталось разобрать последнюю функцию — sma(). В нее мы передаем значение переменной limit. Тип функции будет void, потому функция не должна ничего возвращать. Введем переменную значения суммы цен анализируемых свечей sum, а также сравнение, чтобы limits не был меньше заданного пользователем периода скользящей MA_Period.
0 1 2 3 |
void sma(int limits) { double sum = 0; if(limits < MA_Period) limits = MA_Period; } |
Как мы знаем, простая скользящая средняя рассчитывается по формуле:
То есть цены закрытия последних N свечей делятся на количество этих свечей (что и является периодом индикатора). Соответственно мы должны просуммировать количество заданных свечей, чтобы получить суммарное значение их цен закрытия. Для этого воспользуемся циклом for, который запишем строчкой ниже:
0 |
for(int i=1;i<MA_Period;i++,limits--) sum+=Close[limits]; |
Узнав сумму цен, можно запустить последний расчет, цель которого будет в том, чтобы полученное значение разделить на период индикатора и получить в итоге цену SMA для каждого бара. Для этого нам нужен цикл while.
0 1 2 3 4 5 6 7 8 9 10 |
while(limits>=0) //цикл для построения индикатора на графике { sum += Close[limits]; double SMA = sum/MA_Period; MABuffer1[limits]= SMA; if(Close[limits] > SMA) MABuffer2[limits]= SMA; sum -= Close[limits+MA_Period-1]; limits--; } |
После расчетов нам необходимо построить две линии индикатора на графике. Первая линия будет строиться на каждой свече, поэтому массиву MABuffer1 передаем значение SMA. Массив уже привязан к буферу, поэтому построение графика произойдет автоматически. Для второго массива мы добавили условие по которому он будет использоваться только, если цена закрытия анализируемого бара выше, чем скользящая средняя. Соответственно зеленая линия будет строиться поверх красной только при соблюдении данного условия. Осталось вычесть уже ненужную цену закрытия свечи из общей суммы и уменьшить количество свечей limits в цикле.
В итоге готовая пользовательская функция будет иметь вид:
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
void sma(int limits) { double sum = 0; if(limits < MA_Period) limits = MA_Period; for(int i=1;i<MA_Period;i++,limits--) sum+=Close[limits]; while(limits>=0) //цикл для построения индикатора на графике { sum += Close[limits]; double SMA = sum/MA_Period; MABuffer1[limits]= SMA; if(Close[limits] > SMA) MABuffer2[limits]= SMA; sum -= Close[limits+MA_Period-1]; limits--; } } |
Все, теперь можно установить индикатор на график и посмотреть, что получилось.
Проверьте «Окно Данных», там должно быть два значения данных буферов, которые будут изменяться в зависимости от выделенной мышью свечи. Там, где зеленая линия будет два одинаковых значения, потому что второй массив построен поверх первого. На красной линии значение будет только у первого массива.
Если вы захотите улучшить данный код и добавить в расчет другие типы скользящих средних, такие как: Exponential MA, Smoothed MA или Linear Weighted MA, то их открытый код можно скачать с официального сайта MQL5.
На этом все, весь код урока приложен во вложении с комментариями к каждой строчке. Подписывайтесь на рассылку, чтобы не пропустить новые статьи.
До встречи на нашем сайте DaVinci FX Group.
[download url=»http://www.davinci-fx.com/wp-content/uploads/2021/01/2.5-Our-Moving-Averages.rar» title=»Скачать код урока»]