Введение
При использовании новой версии компилятора языка MQL4 некоторые старые программы могут выдавать ошибки.
В старой версии компилятора во избежание критического завершения программ многие ошибки обрабатывались средой исполнения и не приводили к остановке работы. Например, деление на ноль или выход за пределы массива являются критическими ошибками и обычно приводят к аварийному завершению. Проявляются такие ошибки лишь в некоторых состояниях при определенных значениях переменных, однако о таких ситуациях следует знать и корректно их обрабатывать.
Новый компилятор позволяет обнаружить реальные или потенциальные источники ошибок и повысить качество кода.
В этой статье мы рассмотрим возможные ошибки, возникающие при компиляции старых программ, и методы их устранения.
- Ошибки компиляции
- 1.1. Идентификатор совпадает с зарезервированным словом
- 1.2. Специальные символы в наименованиях переменных и функций
- 1.3. Ошибки использования оператора switch
- 1.4. Возвращаемые значения у функций
- 1.5. Массивы в аргументах функций
- Ошибки времени выполнения
- 2.1. Выход за пределы массива (Array out of range)
- 2.2. Деление на ноль (Zero divide)
- 2.3. Использование 0 вместо NULL для текущего символа
- 2.4. Строки в формате Unicodе и их использование в DLL
- 2.5. Совместное использование файлов
- 2.6. Особенность преобразования datetime
- Предупреждения компилятора
- 3.1. Пересечения имен глобальных и локальных переменных
- 3.2. Несоответствие типов
- 3.3. Неиспользуемые переменные
1. Ошибки компиляции
При наличии ошибок в коде программа не может быть скомпилирована.
Для полного контроля всех ошибок рекомендуется использовать строгий режим компиляции, который устанавливается директивой:
#property strict
Этот режим значительно упрощает поиск ошибок.
1.1. Идентификатор совпадает с зарезервированным словом
Если наименование переменной или функции совпадает с одним из зарезервированных слов:
int char[]; int char1[]; int char() { return(0); }
то компилятор выводит сообщения об ошибках:
Рис.1. Ошибки “unexpected token” и “name expected”
Для исправления данной ошибки нужно исправить имя переменной или функции.
1.2. Специальные символы в наименованиях переменных и функций
Если наименования переменных или функций содержат специальные символы ($, @, точка):
int $var1; int @var2; int var.3; void f@() { return; }
то компилятор выводит сообщения об ошибках:
Рис.2. Ошибки “unknown symbol” и “semicolon expected”
Для исправления данной ошибки нужно скорректировать имена переменных или функций.
1.3. Ошибки использования оператора switch
Старая версия компилятора позволяла использовать любые значения в выражениях и константах оператора switch:
void start() { double n=3.14; switch(n) { case 3.14: Print("Pi");break; case 2.7: Print("E");break; } }
В новом компиляторе выражения и константы оператора switch должны быть целыми числами, поэтому при использовании подобных конструкций возникают ошибки:
Рис.3. Ошибки “illegal switch expression type” и “constant expression is not integral”
В таких случаях можно использовать явные сравнения численных значений, например:
void start() { double n=3.14; if(n==3.14) Print("Pi"); else if(n==2.7) Print("E"); }
1.4. Возвращаемые значений функций
Все функции, кроме void, должны возвращать значение объявленного типа. Например:
int function()
{
}
При строгом режиме компиляции (strict) возникает ошибка:
Рис.4. Ошибка “not all control paths return a value”
В режиме компиляции по умолчанию компилятор выводит предупреждение:
Рис.5. Предупреждение “not all control paths return a value”
Если возвращаемое значение функции не соответствует объявлению:
int init() { return; }
то при строгом режиме компиляции возникает ошибка:
Рис.6. Ошибка “function must return a value”
В режиме компиляции по умолчанию компилятор выводит предупреждение:
Рис.7. Предупреждение ‘return – function must return a value”
Для исправления таких ошибок в код функции нужно добавить оператор возврата return c возвращаемым значением соответствующего типа.
1.5. Массивы в аргументах функций
Массивы в аргументах функций теперь передаются только по ссылке.
double ArrayAverage(double a[]) { return(0); }
Данный код при строгом режиме компиляции (strict) приведет к ошибке:
Рис.8. Ошибка компилятора “arrays passed by reference only”
В режиме компиляции по умолчанию компилятор выводит предупреждение:
Рис.9. Предупреждение компилятора “arrays passed by reference only”
Для исправления таких ошибок нужно явно указать передачу массива по ссылке, добавив префикс & перед именем массива:
double ArrayAverage(double &a[]) { return(0); }
Следует отметить, что теперь константные массивы (Time[], Open[], High[], Low[], Close[], Volume[]) не могут быть переданы по ссылке. Например, вызов:
ArrayAverage(Open);
вне зависимости от режима компиляции приводит к ошибке:
Рис.10. Ошибка ‘Open’ – constant variable cannot be passed as reference
Для устранения подобных ошибок нужно скопировать необходимые данные из константного массива:
double OpenPrices[]; ArrayCopy(OpenPrices,Open,0,0,WHOLE_ARRAY); ArrayAverage(OpenPrices);
2. Ошибки времени выполнения
Ошибки, возникающие в процессе исполнения кода программы принято называть ошибками времени выполнения (runtime errors). Такие ошибки обычно зависят от состояния программы и связаны с некорректными значениями переменных.
Например, если переменная используется в качестве индекса элементов массива, то ее отрицательные значения неизбежно приведут к выходу за пределы массива.
2.1. Выход за пределы массива (Array out of range)
Эта ошибка часто возникает в индикаторах при обращении к индикаторным буферам. Функция IndicatorCounted() возвращает количество баров, неизменившихся после последнего вызова индикатора. Значения индикаторов на уже рассчитанных ранее
барах не нуждаются в пересчете, поэтому для ускорения расчетов
достаточно обрабатывать только несколько последних баров.
Большинство индикаторов, в которых используется данный способ оптимизации вычислений, имеют вид:
int start() { if (Bars<100) return(-1); int counted_bars=IndicatorCounted(); if(counted_bars<0) return(-1); int limit=Bars-counted_bars; if(counted_bars==0) { limit--; limit-=10; } else { limit++; } for (int i=limit; i>0; i--) { Buff1[i]=0.5*(Open[i+5]+Close[i+10]) } }
Часто встречается некорректная обработка случая counted_bars==0 (начальную позицию limit нужно уменьшить на значение, равное 1 + максимальный индекс относительно переменной цикла).
Также следует помнить о том, что в момент исполнения функции start() мы можем обращаться к элементам массивов индикаторных буферов от 0 до Bars()-1. Если есть необходимость работы с массивами, которые не являются индикаторными буферами, то их размер следует увеличить при помощи функции ArrayResize() в соответствии с текущим размером индикаторных буферов. Максимальный индекс элемента для адресации также можно получить вызовом ArraySize() с одним из индикаторных буферов в качестве аргумента.
2.2. Деление на ноль (Zero divide)
Ошибка “Zero divide” возникает в случае, если при выполнении операции деления делитель оказывается равен нулю:
void OnStart() { int a=0, b=0,c; c=a/b; Print("c=",c); }
При выполнении данного скрипта во вкладке “Эксперты” возникает сообщение об ошибке и завершении работы программы:
Рис.11. Сообщение об ошибке “zero divide”
Обычно такая ошибка возникает в случаях, когда значение делителя определяется значениями каких-либо внешних данных. Например, если анализируются параметры торговли, то величина задействованной маржи оказывается равна 0 если нет открытых ордеров. Другой пример: если анализируемые данные считываются из файла, то в случае его отсутствия нельзя гарантировать корректную работу. По этой причине желательно стараться учитывать подобные случаи и корректно их обрабатывать.
Самый простой способ – проверять делитель перед операцией деления и выводить сообщение об некорректном значении параметра:
void OnStart() { int a=0, b=0,c; if(b!=0) {c=a/b; Print(c);} else {Print("Error: b=0"); return; }; }
В результате критической ошибки не возникает, но выводится сообщение о некорректном значении параметра и работа завершается:
Рис. 12. Сообщение о некорректном значении делителя
2.3. Использование 0 вместо NULL для текущего символа
В старой версии компилятора допускалось использование 0 (нуля) в качестве аргумента в функциях, требующих указания финансового инструмента.
Например, значение технического индикатора Moving Average для текущего символа можно было запрашивать следующим образом:
AlligatorJawsBuffer[i]=iMA(0,0,13,8,MODE_SMMA,PRICE_MEDIAN,i);
В новом компиляторе для указания текущего символа нужно явно указывать NULL:
AlligatorJawsBuffer[i]=iMA(NULL,0,13,8,MODE_SMMA,PRICE_MEDIAN,i);
Кроме того, текущий символ и период графика можно указать при помощи функций Symbol() и Period().
AlligatorJawsBuffer[i]=iMA(Symbol(),Period(),13,8,MODE_SMMA,PRICE_MEDIAN,i);
2.4. Строки в формате Unicodе и их использование в DLL
Строки теперь представляют собой последовательность символов Unicode.
Следует учитывать этот факт и использовать соответствующие функции Windows. Например, при использовании функций библиотеки wininet.dll вместо InternetOpenA() и InternetOpenUrlA() следует вызывать InternetOpenW() и InternetOpenUrlW().
В MQL4 изменилась внутренняя структура строк (теперь она занимает 12 байт), поэтому при передаче строк в DLL следует использовать структуру MqlString:
#pragma pack(push,1) struct MqlString { int size; LPWSTR buffer; int reserved; }; #pragma pack(pop,1)
2.5. Совместное использование файлов
В новом MQL4 при открытии файлов необходимо явно указывать флаги FILE_SHARE_WRITE и FILE_SHARE_READ для совместного использования.
В случае их отсутствия файл будет открыт в монопольном режиме, что не позволит больше никому его открывать, пока он не будет закрыт монополистом.
Например, при работе с оффлайновыми графиками требуется явно указывать флаги совместного доступа:
ExtHandle=FileOpenHistory(c_symbol+i_period+".hst",FILE_BIN|FILE_WRITE|FILE_SHARE_WRITE|FILE_SHARE_READ);
Подробности можно найти в статье в статье “Оффлайновые графики и новый MQL4“.
2.6. Особенность преобразования datetime
Следует иметь ввиду, что преобразование типа datetime в строку теперь зависит от режима компиляции:
datetime date=D'2014.03.05 15:46:58'; string str="mydate="+date;
Например, попытка работы с файлами, имя которых содержит двоеточие, приведет к ошибке.
3. Предупреждения компилятора
Предупреждения компилятора носят информационный характер и не являются сообщениями об ошибках, однако они указывают на возможные источники ошибок и лучше их скорректировать.
Чистый код не должен содержать предупреждений.
3.1. Пересечения имен глобальных и локальных переменных
Если на глобальном и локальном уровнях присутствуют переменные с одинаковыми именами:
int i; void OnStart() { int i=0,j=0; for (i=0; i<5; i++) {j+=i;} PrintFormat("i=%d, j=%d",i,j); }
то компилятор выводит предупреждение и укажет номер строки, на которой объявлена глобальная переменная:
Рис.13. Предупреждение “declaration of ‘%’ hides global declaration at line %”
Для исправления таких предупреждений нужно скорректировать имена глобальных переменных.
3.2. Несоответствие типов
В новой версии компилятора введена операция приведения типов.
#property strict void OnStart() { double a=7; float b=a; int c=b; string str=c; Print(c); }
В строгом режиме компиляции при несоответствии типов компилятор выводит предупреждения:
Рис.14. Предупреждения “possible loss of data due to type conversion” и “implicit conversion from ‘number’ to ‘string’
В данном примере компилятор предупреждает о возможной потере точности при присвоении различных типов данных и неявном преобразовании типа int в string.
Для исправления нужно использовать явное приведение типов:
#property strict void OnStart() { double a=7; float b=(float)a; int c=(int)b; string str=(string)c; Print(c); }
3.3. Неиспользуемые переменные
Наличие переменных, которые не используются в коде программы (лишние сущности) не является хорошим тоном.
void OnStart() { int i,j=10,k,l,m,n2=1; for(i=0; i<5; i++) {j+=i;} }
Сообщения о таких переменных выводятся вне зависимости от режима компиляции:
Рис.15. Предупреждения “variable ‘%’ not used’
Для исправления нужно убрать неиспользуемые переменные из кода программы.
Выводы
В статье рассмотрены типичные проблемы, с которыми могут столкнуться программисты при компиляции старых программ, содержащих ошибки.
Во всех случаях при отладке программ рекомендуется использовать строгий режим компиляции.
Предупреждение: все права на данные материалы принадлежат MetaQuotes Ltd. Полная или частичная перепечатка запрещена.
0
ERR_NO_ERROR
Нет ошибки
1
ERR_NO_RESULT
Нет ошибки, но результат неизвестен
2
ERR_COMMON_ERROR
Общая ошибка
3
ERR_INVALID_TRADE_PARAMETERS
Неправильные параметры
4
ERR_SERVER_BUSY
Торговый сервер занят
5
ERR_OLD_VERSION
Старая версия клиентского терминала
6
ERR_NO_CONNECTION
Нет связи с торговым сервером
7
ERR_NOT_ENOUGH_RIGHTS
Недостаточно прав
8
ERR_TOO_FREQUENT_REQUESTS
Слишком частые запросы
9
ERR_MALFUNCTIONAL_TRADE
Недопустимая операция, нарушающая функционирование сервера
64
ERR_ACCOUNT_DISABLED
Счет заблокирован
65
ERR_INVALID_ACCOUNT
Неправильный номер счета
128
ERR_TRADE_TIMEOUT
Истек срок ожидания совершения сделки
129
ERR_INVALID_PRICE
Неправильная цена
130
ERR_INVALID_STOPS
Неправильные стопы
131
ERR_INVALID_TRADE_VOLUME
Неправильный объем
132
ERR_MARKET_CLOSED
Рынок закрыт
133
ERR_TRADE_DISABLED
Торговля запрещена
134
ERR_NOT_ENOUGH_MONEY
Недостаточно денег для совершения операции
135
ERR_PRICE_CHANGED
Цена изменилась
136
ERR_OFF_QUOTES
Нет цен
137
ERR_BROKER_BUSY
Брокер занят
138
ERR_REQUOTE
Новые цены
139
ERR_ORDER_LOCKED
Ордер заблокирован и уже обрабатывается
140
ERR_LONG_POSITIONS_ONLY_ALLOWED
Разрешена только покупка
141
ERR_TOO_MANY_REQUESTS
Слишком много запросов
145
ERR_TRADE_MODIFY_DENIED
Модификация запрещена, так как ордер слишком близок к рынку
146
ERR_TRADE_CONTEXT_BUSY
Подсистема торговли занята
147
ERR_TRADE_EXPIRATION_DENIED
Использование даты истечения ордера запрещено брокером
148
ERR_TRADE_TOO_MANY_ORDERS
Количество открытых и отложенных ордеров достигло предела, установленного брокером
149
ERR_TRADE_HEDGE_PROHIBITED
Попытка открыть противоположный ордер в случае, если хеджирование запрещено
150
ERR_TRADE_PROHIBITED_BY_FIFO
Попытка закрыть позицию по инструменту в противоречии с правилом FIFO
4000
ERR_NO_MQLERROR
Нет ошибки
4001
ERR_WRONG_FUNCTION_POINTER
Неправильный указатель функции
4002
ERR_ARRAY_INDEX_OUT_OF_RANGE
Индекс массива — вне диапазона
4003
ERR_NO_MEMORY_FOR_CALL_STACK
Нет памяти для стека функций
4004
ERR_RECURSIVE_STACK_OVERFLOW
Переполнение стека после рекурсивного вызова
4005
ERR_NOT_ENOUGH_STACK_FOR_PARAM
На стеке нет памяти для передачи параметров
4006
ERR_NO_MEMORY_FOR_PARAM_STRING
Нет памяти для строкового параметра
4007
ERR_NO_MEMORY_FOR_TEMP_STRING
Нет памяти для временной строки
4008
ERR_NOT_INITIALIZED_STRING
Неинициализированная строка
4009
ERR_NOT_INITIALIZED_ARRAYSTRING
Неинициализированная строка в массиве
4010
ERR_NO_MEMORY_FOR_ARRAYSTRING
Нет памяти для строкового массива
4011
ERR_TOO_LONG_STRING
Слишком длинная строка
4012
ERR_REMAINDER_FROM_ZERO_DIVIDE
Остаток от деления на ноль
4013
ERR_ZERO_DIVIDE
Деление на ноль
4014
ERR_UNKNOWN_COMMAND
Неизвестная команда
4015
ERR_WRONG_JUMP
Неправильный переход
4016
ERR_NOT_INITIALIZED_ARRAY
Неинициализированный массив
4017
ERR_DLL_CALLS_NOT_ALLOWED
Вызовы DLL не разрешены
4018
ERR_CANNOT_LOAD_LIBRARY
Невозможно загрузить библиотеку
4019
ERR_CANNOT_CALL_FUNCTION
Невозможно вызвать функцию
4020
ERR_EXTERNAL_CALLS_NOT_ALLOWED
Вызовы внешних библиотечных функций не разрешены
4021
ERR_NO_MEMORY_FOR_RETURNED_STR
Недостаточно памяти для строки, возвращаемой из функции
4022
ERR_SYSTEM_BUSY
Система занята
4023
ERR_DLLFUNC_CRITICALERROR
Критическая ошибка вызова DLL-функции
4024
ERR_INTERNAL_ERROR
Внутренняя ошибка
4025
ERR_OUT_OF_MEMORY
Нет памяти
4026
ERR_INVALID_POINTER
Неверный указатель
4027
ERR_FORMAT_TOO_MANY_FORMATTERS
Слишком много параметров форматирования строки
4028
ERR_FORMAT_TOO_MANY_PARAMETERS
Число параметров превышает число параметров форматирования строки
4029
ERR_ARRAY_INVALID
Неверный массив
4030
ERR_CHART_NOREPLY
График не отвечает
4050
ERR_INVALID_FUNCTION_PARAMSCNT
Неправильное количество параметров функции
4051
ERR_INVALID_FUNCTION_PARAMVALUE
Недопустимое значение параметра функции
4052
ERR_STRING_FUNCTION_INTERNAL
Внутренняя ошибка строковой функции
4053
ERR_SOME_ARRAY_ERROR
Ошибка массива
4054
ERR_INCORRECT_SERIESARRAY_USING
Неправильное использование массива-таймсерии
4055
ERR_CUSTOM_INDICATOR_ERROR
Ошибка пользовательского индикатора
4056
ERR_INCOMPATIBLE_ARRAYS
Массивы несовместимы
4057
ERR_GLOBAL_VARIABLES_PROCESSING
Ошибка обработки глобальных переменных
4058
ERR_GLOBAL_VARIABLE_NOT_FOUND
Глобальная переменная не обнаружена
4059
ERR_FUNC_NOT_ALLOWED_IN_TESTING
Функция не разрешена в тестовом режиме
4060
ERR_FUNCTION_NOT_CONFIRMED
Функция не разрешена
4061
ERR_SEND_MAIL_ERROR
Ошибка отправки почты
4062
ERR_STRING_PARAMETER_EXPECTED
Ожидается параметр типа string
4063
ERR_INTEGER_PARAMETER_EXPECTED
Ожидается параметр типа integer
4064
ERR_DOUBLE_PARAMETER_EXPECTED
Ожидается параметр типа double
4065
ERR_ARRAY_AS_PARAMETER_EXPECTED
В качестве параметра ожидается массив
4066
ERR_HISTORY_WILL_UPDATED
Запрошенные исторические данные в состоянии обновления
4067
ERR_TRADE_ERROR
Ошибка при выполнении торговой операции
4068
ERR_RESOURCE_NOT_FOUND
Ресурс не найден
4069
ERR_RESOURCE_NOT_SUPPORTED
Ресурс не поддерживается
4070
ERR_RESOURCE_DUPLICATED
Дубликат ресурса
4071
ERR_INDICATOR_CANNOT_INIT
Ошибка инициализации пользовательского индикатора
4072
ERR_INDICATOR_CANNOT_LOAD
Ошибка загрузки пользовательского индикатора
4073
ERR_NO_HISTORY_DATA
Нет исторических данных
4074
ERR_NO_MEMORY_FOR_HISTORY
Не хватает памяти для исторических данных
4075
ERR_NO_MEMORY_FOR_INDICATOR
Не хватает памяти для расчёта индикатора
4099
ERR_END_OF_FILE
Конец файла
4100
ERR_SOME_FILE_ERROR
Ошибка при работе с файлом
4101
ERR_WRONG_FILE_NAME
Неправильное имя файла
4102
ERR_TOO_MANY_OPENED_FILES
Слишком много открытых файлов
4103
ERR_CANNOT_OPEN_FILE
Невозможно открыть файл
4104
ERR_INCOMPATIBLE_FILEACCESS
Несовместимый режим доступа к файлу
4105
ERR_NO_ORDER_SELECTED
Ни один ордер не выбран
4106
ERR_UNKNOWN_SYMBOL
Неизвестный символ
4107
ERR_INVALID_PRICE_PARAM
Неправильный параметр цены для торговой функции
4108
ERR_INVALID_TICKET
Неверный номер тикета
4109
ERR_TRADE_NOT_ALLOWED
Торговля не разрешена. Необходимо включить опцию «Разрешить советнику торговать» в свойствах эксперта
4110
ERR_LONGS_NOT_ALLOWED
Ордера на покупку не разрешены. Необходимо проверить свойства эксперта
4111
ERR_SHORTS_NOT_ALLOWED
Ордера на продажу не разрешены. Необходимо проверить свойства эксперта
4112
ERR_TRADE_EXPERT_DISABLED_BY_SERVER
Автоматическая торговля с помощью экспертов/скриптов запрещена на стороне сервера
4200
ERR_OBJECT_ALREADY_EXISTS
Объект уже существует
4201
ERR_UNKNOWN_OBJECT_PROPERTY
Запрошено неизвестное свойство объекта
4202
ERR_OBJECT_DOES_NOT_EXIST
Объект не существует
4203
ERR_UNKNOWN_OBJECT_TYPE
Неизвестный тип объекта
4204
ERR_NO_OBJECT_NAME
Нет имени объекта
4205
ERR_OBJECT_COORDINATES_ERROR
Ошибка координат объекта
4206
ERR_NO_SPECIFIED_SUBWINDOW
Не найдено указанное подокно
4207
ERR_SOME_OBJECT_ERROR
Ошибка при работе с объектом
4210
ERR_CHART_PROP_INVALID
Неизвестное свойство графика
4211
ERR_CHART_NOT_FOUND
График не найден
4212
ERR_CHARTWINDOW_NOT_FOUND
Не найдено подокно графика
4213
ERR_CHARTINDICATOR_NOT_FOUND
Индикатор не найден
4220
ERR_SYMBOL_SELECT
Ошибка выбора инструмента
4250
ERR_NOTIFICATION_ERROR
Ошибка отправки push-уведомления
4251
ERR_NOTIFICATION_PARAMETER
Ошибка параметров push-уведомления
4252
ERR_NOTIFICATION_SETTINGS
Уведомления запрещены
4253
ERR_NOTIFICATION_TOO_FREQUENT
Слишком частые запросы отсылки push-уведомлений
4260
ERR_FTP_NOSERVER
Не указан FTP сервер
4261
ERR_FTP_NOLOGIN
Не указан FTP логин
4262
ERR_FTP_CONNECT_FAILED
Ошибка при подключении к FTP серверу
4263
ERR_FTP_CLOSED
Подключение к FTP серверу закрыто
4264
ERR_FTP_CHANGEDIR
На FTP сервере не найдена директория для выгрузки файла
4265
ERR_FTP_FILE_ERROR
Не найден файл в директории MQL4Files для отправки на FTP сервер
4266
ERR_FTP_ERROR
Ошибка при передаче файла на FTP сервер
5001
ERR_FILE_TOO_MANY_OPENED
Слишком много открытых файлов
5002
ERR_FILE_WRONG_FILENAME
Неверное имя файла
5003
ERR_FILE_TOO_LONG_FILENAME
Слишком длинное имя файла
5004
ERR_FILE_CANNOT_OPEN
Ошибка открытия файла
5005
ERR_FILE_BUFFER_ALLOCATION_ERROR
Ошибка размещения буфера текстового файла
5006
ERR_FILE_CANNOT_DELETE
Ошибка удаления файла
5007
ERR_FILE_INVALID_HANDLE
Неверный хендл файла (файл закрыт или не был открыт)
5008
ERR_FILE_WRONG_HANDLE
Неверный хендл файла (индекс хендла отсутствует в таблице)
5009
ERR_FILE_NOT_TOWRITE
Файл должен быть открыт с флагом FILE_WRITE
5010
ERR_FILE_NOT_TOREAD
Файл должен быть открыт с флагом FILE_READ
5011
ERR_FILE_NOT_BIN
Файл должен быть открыт с флагом FILE_BIN
5012
ERR_FILE_NOT_TXT
Файл должен быть открыт с флагом FILE_TXT
5013
ERR_FILE_NOT_TXTORCSV
Файл должен быть открыт с флагом FILE_TXT или FILE_CSV
5014
ERR_FILE_NOT_CSV
Файл должен быть открыт с флагом FILE_CSV
5015
ERR_FILE_READ_ERROR
Ошибка чтения файла
5016
ERR_FILE_WRITE_ERROR
Ошибка записи файла
5017
ERR_FILE_BIN_STRINGSIZE
Размер строки должен быть указан для двоичных файлов
5018
ERR_FILE_INCOMPATIBLE
Неверный тип файла (для строковых массивов-TXT, для всех других-BIN)
5019
ERR_FILE_IS_DIRECTORY
Файл является директорией
5020
ERR_FILE_NOT_EXIST
Файл не существует
5021
ERR_FILE_CANNOT_REWRITE
Файл не может быть перезаписан
5022
ERR_FILE_WRONG_DIRECTORYNAME
Неверное имя директории
5023
ERR_FILE_DIRECTORY_NOT_EXIST
Директория не существует
5024
ERR_FILE_NOT_DIRECTORY
Указанный файл не является директорией
5025
ERR_FILE_CANNOT_DELETE_DIRECTORY
Ошибка удаления директории
5026
ERR_FILE_CANNOT_CLEAN_DIRECTORY
Ошибка очистки директории
5027
ERR_FILE_ARRAYRESIZE_ERROR
Ошибка изменения размера массива
5028
ERR_FILE_STRINGRESIZE_ERROR
Ошибка изменения размера строки
5029
ERR_FILE_STRUCT_WITH_OBJECTS
Структура содержит строки или динамические массивы
5200
ERR_WEBREQUEST_INVALID_ADDRESS
URL не прошел проверку
5201
ERR_WEBREQUEST_CONNECT_FAILED
Не удалось подключиться к указанному URL
5202
ERR_WEBREQUEST_TIMEOUT
Превышен таймаут получения данных
5203
ERR_WEBREQUEST_REQUEST_FAILED
Ошибка в результате выполнения HTTP запроса
Пользовательские ошибки
65536
ERR_USER_ERROR_FIRST
С этого кода начинаются ошибки, задаваемые пользователем
GetLastError() – функция, возвращающая код последней ошибки, которая хранится в предопределенной переменной _LastError. Значение этой переменной можно сбросить в ноль функцией ResetLastError().
0 |
ERR_NO_ERROR |
Нет ошибки |
1 |
ERR_NO_RESULT |
Нет ошибки, но результат неизвестен |
2 |
ERR_COMMON_ERROR |
Общая ошибка |
3 |
ERR_INVALID_TRADE_PARAMETERS |
Неправильные параметры |
4 |
ERR_SERVER_BUSY |
Торговый сервер занят |
5 |
ERR_OLD_VERSION |
Старая версия клиентского терминала |
6 |
ERR_NO_CONNECTION |
Нет связи с торговым сервером |
7 |
ERR_NOT_ENOUGH_RIGHTS |
Недостаточно прав |
8 |
ERR_TOO_FREQUENT_REQUESTS |
Слишком частые запросы |
9 |
ERR_MALFUNCTIONAL_TRADE |
Недопустимая операция, нарушающая функционирование сервера |
64 |
ERR_ACCOUNT_DISABLED |
Счет заблокирован |
65 |
ERR_INVALID_ACCOUNT |
Неправильный номер счета |
128 |
ERR_TRADE_TIMEOUT |
Истек срок ожидания совершения сделки |
129 |
ERR_INVALID_PRICE |
Неправильная цена |
130 |
ERR_INVALID_STOPS |
Неправильные стопы |
131 |
ERR_INVALID_TRADE_VOLUME |
Неправильный объем |
132 |
ERR_MARKET_CLOSED |
Рынок закрыт |
133 |
ERR_TRADE_DISABLED |
Торговля запрещена |
134 |
ERR_NOT_ENOUGH_MONEY |
Недостаточно денег для совершения операции |
135 |
ERR_PRICE_CHANGED |
Цена изменилась |
136 |
ERR_OFF_QUOTES |
Нет цен |
137 |
ERR_BROKER_BUSY |
Брокер занят |
138 |
ERR_REQUOTE |
Новые цены |
139 |
ERR_ORDER_LOCKED |
Ордер заблокирован и уже обрабатывается |
140 |
ERR_LONG_POSITIONS_ONLY_ALLOWED |
Разрешена только покупка |
141 |
ERR_TOO_MANY_REQUESTS |
Слишком много запросов |
145 |
ERR_TRADE_MODIFY_DENIED |
Модификация запрещена, так как ордер слишком близок к рынку |
146 |
ERR_TRADE_CONTEXT_BUSY |
Подсистема торговли занята |
147 |
ERR_TRADE_EXPIRATION_DENIED |
Использование даты истечения ордера запрещено брокером |
148 |
ERR_TRADE_TOO_MANY_ORDERS |
Количество открытых и отложенных ордеров достигло предела, установленного брокером |
149 |
ERR_TRADE_HEDGE_PROHIBITED |
Попытка открыть противоположный ордер в случае, если хеджирование запрещено |
150 |
ERR_TRADE_PROHIBITED_BY_FIFO |
Попытка закрыть позицию по инструменту в противоречии с правилом FIFO |
4000 |
ERR_NO_MQLERROR |
Нет ошибки |
4001 |
ERR_WRONG_FUNCTION_POINTER |
Неправильный указатель функции |
4002 |
ERR_ARRAY_INDEX_OUT_OF_RANGE |
Индекс массива — вне диапазона |
4003 |
ERR_NO_MEMORY_FOR_CALL_STACK |
Нет памяти для стека функций |
4004 |
ERR_RECURSIVE_STACK_OVERFLOW |
Переполнение стека после рекурсивного вызова |
4005 |
ERR_NOT_ENOUGH_STACK_FOR_PARAM |
На стеке нет памяти для передачи параметров |
4006 |
ERR_NO_MEMORY_FOR_PARAM_STRING |
Нет памяти для строкового параметра |
4007 |
ERR_NO_MEMORY_FOR_TEMP_STRING |
Нет памяти для временной строки |
4008 |
ERR_NOT_INITIALIZED_STRING |
Неинициализированная строка |
4009 |
ERR_NOT_INITIALIZED_ARRAYSTRING |
Неинициализированная строка в массиве |
4010 |
ERR_NO_MEMORY_FOR_ARRAYSTRING |
Нет памяти для строкового массива |
4011 |
ERR_TOO_LONG_STRING |
Слишком длинная строка |
4012 |
ERR_REMAINDER_FROM_ZERO_DIVIDE |
Остаток от деления на ноль |
4013 |
ERR_ZERO_DIVIDE |
Деление на ноль |
4014 |
ERR_UNKNOWN_COMMAND |
Неизвестная команда |
4015 |
ERR_WRONG_JUMP |
Неправильный переход |
4016 |
ERR_NOT_INITIALIZED_ARRAY |
Неинициализированный массив |
4017 |
ERR_DLL_CALLS_NOT_ALLOWED |
Вызовы DLL не разрешены |
4018 |
ERR_CANNOT_LOAD_LIBRARY |
Невозможно загрузить библиотеку |
4019 |
ERR_CANNOT_CALL_FUNCTION |
Невозможно вызвать функцию |
4020 |
ERR_EXTERNAL_CALLS_NOT_ALLOWED |
Вызовы внешних библиотечных функций не разрешены |
4021 |
ERR_NO_MEMORY_FOR_RETURNED_STR |
Недостаточно памяти для строки, возвращаемой из функции |
4022 |
ERR_SYSTEM_BUSY |
Система занята |
4023 |
ERR_DLLFUNC_CRITICALERROR |
Критическая ошибка вызова DLL-функции |
4024 |
ERR_INTERNAL_ERROR |
Внутренняя ошибка |
4025 |
ERR_OUT_OF_MEMORY |
Нет памяти |
4026 |
ERR_INVALID_POINTER |
Неверный указатель |
4027 |
ERR_FORMAT_TOO_MANY_FORMATTERS |
Слишком много параметров форматирования строки |
4028 |
ERR_FORMAT_TOO_MANY_PARAMETERS |
Число параметров превышает число параметров форматирования строки |
4029 |
ERR_ARRAY_INVALID |
Неверный массив |
4030 |
ERR_CHART_NOREPLY |
График не отвечает |
4050 |
ERR_INVALID_FUNCTION_PARAMSCNT |
Неправильное количество параметров функции |
4051 |
ERR_INVALID_FUNCTION_PARAMVALUE |
Недопустимое значение параметра функции |
4052 |
ERR_STRING_FUNCTION_INTERNAL |
Внутренняя ошибка строковой функции |
4053 |
ERR_SOME_ARRAY_ERROR |
Ошибка массива |
4054 |
ERR_INCORRECT_SERIESARRAY_USING |
Неправильное использование массива-таймсерии |
4055 |
ERR_CUSTOM_INDICATOR_ERROR |
Ошибка пользовательского индикатора |
4056 |
ERR_INCOMPATIBLE_ARRAYS |
Массивы несовместимы |
4057 |
ERR_GLOBAL_VARIABLES_PROCESSING |
Ошибка обработки глобальных переменных |
4058 |
ERR_GLOBAL_VARIABLE_NOT_FOUND |
Глобальная переменная не обнаружена |
4059 |
ERR_FUNC_NOT_ALLOWED_IN_TESTING |
Функция не разрешена в тестовом режиме |
4060 |
ERR_FUNCTION_NOT_CONFIRMED |
Функция не разрешена |
4061 |
ERR_SEND_MAIL_ERROR |
Ошибка отправки почты |
4062 |
ERR_STRING_PARAMETER_EXPECTED |
Ожидается параметр типа string |
4063 |
ERR_INTEGER_PARAMETER_EXPECTED |
Ожидается параметр типа integer |
4064 |
ERR_DOUBLE_PARAMETER_EXPECTED |
Ожидается параметр типа double |
4065 |
ERR_ARRAY_AS_PARAMETER_EXPECTED |
В качестве параметра ожидается массив |
4066 |
ERR_HISTORY_WILL_UPDATED |
Запрошенные исторические данные в состоянии обновления |
4067 |
ERR_TRADE_ERROR |
Ошибка при выполнении торговой операции |
4068 |
ERR_RESOURCE_NOT_FOUND |
Ресурс не найден |
4069 |
ERR_RESOURCE_NOT_SUPPORTED |
Ресурс не поддерживается |
4070 |
ERR_RESOURCE_DUPLICATED |
Дубликат ресурса |
4071 |
ERR_INDICATOR_CANNOT_INIT |
Ошибка инициализации пользовательского индикатора |
4072 |
ERR_INDICATOR_CANNOT_LOAD |
Ошибка загрузки пользовательского индикатора |
4073 |
ERR_NO_HISTORY_DATA |
Нет исторических данных |
4074 |
ERR_NO_MEMORY_FOR_HISTORY |
Не хватает памяти для исторических данных |
4075 |
ERR_NO_MEMORY_FOR_INDICATOR |
Не хватает памяти для расчёта индикатора |
4099 |
ERR_END_OF_FILE |
Конец файла |
4100 |
ERR_SOME_FILE_ERROR |
Ошибка при работе с файлом |
4101 |
ERR_WRONG_FILE_NAME |
Неправильное имя файла |
4102 |
ERR_TOO_MANY_OPENED_FILES |
Слишком много открытых файлов |
4103 |
ERR_CANNOT_OPEN_FILE |
Невозможно открыть файл |
4104 |
ERR_INCOMPATIBLE_FILEACCESS |
Несовместимый режим доступа к файлу |
4105 |
ERR_NO_ORDER_SELECTED |
Ни один ордер не выбран |
4106 |
ERR_UNKNOWN_SYMBOL |
Неизвестный символ |
4107 |
ERR_INVALID_PRICE_PARAM |
Неправильный параметр цены для торговой функции |
4108 |
ERR_INVALID_TICKET |
Неверный номер тикета |
4109 |
ERR_TRADE_NOT_ALLOWED |
Торговля не разрешена. Необходимо включить опцию «Разрешить советнику торговать» в свойствах эксперта |
4110 |
ERR_LONGS_NOT_ALLOWED |
Ордера на покупку не разрешены. Необходимо проверить свойства эксперта |
4111 |
ERR_SHORTS_NOT_ALLOWED |
Ордера на продажу не разрешены. Необходимо проверить свойства эксперта |
4112 |
ERR_TRADE_EXPERT_DISABLED_BY_SERVER |
Автоматическая торговля с помощью экспертов/скриптов запрещена на стороне сервера |
4200 |
ERR_OBJECT_ALREADY_EXISTS |
Объект уже существует |
4201 |
ERR_UNKNOWN_OBJECT_PROPERTY |
Запрошено неизвестное свойство объекта |
4202 |
ERR_OBJECT_DOES_NOT_EXIST |
Объект не существует |
4203 |
ERR_UNKNOWN_OBJECT_TYPE |
Неизвестный тип объекта |
4204 |
ERR_NO_OBJECT_NAME |
Нет имени объекта |
4205 |
ERR_OBJECT_COORDINATES_ERROR |
Ошибка координат объекта |
4206 |
ERR_NO_SPECIFIED_SUBWINDOW |
Не найдено указанное подокно |
4207 |
ERR_SOME_OBJECT_ERROR |
Ошибка при работе с объектом |
4210 |
ERR_CHART_PROP_INVALID |
Неизвестное свойство графика |
4211 |
ERR_CHART_NOT_FOUND |
График не найден |
4212 |
ERR_CHARTWINDOW_NOT_FOUND |
Не найдено подокно графика |
4213 |
ERR_CHARTINDICATOR_NOT_FOUND |
Индикатор не найден |
4220 |
ERR_SYMBOL_SELECT |
Ошибка выбора инструмента |
4250 |
ERR_NOTIFICATION_ERROR |
Ошибка отправки push-уведомления |
4251 |
ERR_NOTIFICATION_PARAMETER |
Ошибка параметров push-уведомления |
4252 |
ERR_NOTIFICATION_SETTINGS |
Уведомления запрещены |
4253 |
ERR_NOTIFICATION_TOO_FREQUENT |
Слишком частые запросы отсылки push-уведомлений |
4260 |
ERR_FTP_NOSERVER |
Не указан FTP сервер |
4261 |
ERR_FTP_NOLOGIN |
Не указан FTP логин |
4262 |
ERR_FTP_CONNECT_FAILED |
Ошибка при подключении к FTP серверу |
4263 |
ERR_FTP_CLOSED |
Подключение к FTP серверу закрыто |
4264 |
ERR_FTP_CHANGEDIR |
На FTP сервере не найдена директория для выгрузки файла |
4265 |
ERR_FTP_FILE_ERROR |
Не найден файл в директории MQL4Files для отправки на FTP сервер |
4266 |
ERR_FTP_ERROR |
Ошибка при передаче файла на FTP сервер |
5001 |
ERR_FILE_TOO_MANY_OPENED |
Слишком много открытых файлов |
5002 |
ERR_FILE_WRONG_FILENAME |
Неверное имя файла |
5003 |
ERR_FILE_TOO_LONG_FILENAME |
Слишком длинное имя файла |
5004 |
ERR_FILE_CANNOT_OPEN |
Ошибка открытия файла |
5005 |
ERR_FILE_BUFFER_ALLOCATION_ERROR |
Ошибка размещения буфера текстового файла |
5006 |
ERR_FILE_CANNOT_DELETE |
Ошибка удаления файла |
5007 |
ERR_FILE_INVALID_HANDLE |
Неверный хендл файла (файл закрыт или не был открыт) |
5008 |
ERR_FILE_WRONG_HANDLE |
Неверный хендл файла (индекс хендла отсутствует в таблице) |
5009 |
ERR_FILE_NOT_TOWRITE |
Файл должен быть открыт с флагом FILE_WRITE |
5010 |
ERR_FILE_NOT_TOREAD |
Файл должен быть открыт с флагом FILE_READ |
5011 |
ERR_FILE_NOT_BIN |
Файл должен быть открыт с флагом FILE_BIN |
5012 |
ERR_FILE_NOT_TXT |
Файл должен быть открыт с флагом FILE_TXT |
5013 |
ERR_FILE_NOT_TXTORCSV |
Файл должен быть открыт с флагом FILE_TXT или FILE_CSV |
5014 |
ERR_FILE_NOT_CSV |
Файл должен быть открыт с флагом FILE_CSV |
5015 |
ERR_FILE_READ_ERROR |
Ошибка чтения файла |
5016 |
ERR_FILE_WRITE_ERROR |
Ошибка записи файла |
5017 |
ERR_FILE_BIN_STRINGSIZE |
Размер строки должен быть указан для двоичных файлов |
5018 |
ERR_FILE_INCOMPATIBLE |
Неверный тип файла (для строковых массивов-TXT, для всех других-BIN) |
5019 |
ERR_FILE_IS_DIRECTORY |
Файл является директорией |
5020 |
ERR_FILE_NOT_EXIST |
Файл не существует |
5021 |
ERR_FILE_CANNOT_REWRITE |
Файл не может быть перезаписан |
5022 |
ERR_FILE_WRONG_DIRECTORYNAME |
Неверное имя директории |
5023 |
ERR_FILE_DIRECTORY_NOT_EXIST |
Директория не существует |
5024 |
ERR_FILE_NOT_DIRECTORY |
Указанный файл не является директорией |
5025 |
ERR_FILE_CANNOT_DELETE_DIRECTORY |
Ошибка удаления директории |
5026 |
ERR_FILE_CANNOT_CLEAN_DIRECTORY |
Ошибка очистки директории |
5027 |
ERR_FILE_ARRAYRESIZE_ERROR |
Ошибка изменения размера массива |
5028 |
ERR_FILE_STRINGRESIZE_ERROR |
Ошибка изменения размера строки |
5029 |
ERR_FILE_STRUCT_WITH_OBJECTS |
Структура содержит строки или динамические массивы |
5200 |
ERR_WEBREQUEST_INVALID_ADDRESS |
URL не прошел проверку |
5201 |
ERR_WEBREQUEST_CONNECT_FAILED |
Не удалось подключиться к указанному URL |
5202 |
ERR_WEBREQUEST_TIMEOUT |
Превышен таймаут получения данных |
5203 |
ERR_WEBREQUEST_REQUEST_FAILED |
Ошибка в результате выполнения HTTP запроса |
Пользовательские ошибки |
||
65536 |
ERR_USER_ERROR_FIRST |
С этого кода начинаются ошибки, задаваемые пользователем |
Introduction
Some older programs can return errors in the new version of the MQL4 of compiler.
To avoid critical completion of programs, the previous version compiler handled many errors in the runtime environment. For example, division by zero or array out of range are critical errors and usually lead to application crash. Such errors occur only in some states for certain values of variables. Read this article to know how to handle such cases.
The new compiler can detect actual or potential sources of errors and improve code quality.
In this article, we discuss possible errors that can be detected during compilation of old programs and ways to fix them.
- Compilation Errors
- 1.1. Identifier coincides with a reserved word
- 1.2. Special characters in the names of variables and functions
- 1.3. Errors using the switch operator
- 1.4. Function return values
- 1.5. Arrays in function arguments
- Runtime Errors
- 2.1. Array out of range
- 2.2. Zero divide
- 2.3. Use of 0 instead of NULL for the current character
- 2.4. Unicode format strings and their use in a DLL
- 2.5. File sharing
- 2.6. Datetime conversion
- Compiler Warnings
- 3.1. Names of global and local variables coincide
- 3.2. Mismatch of types
- 3.3. Unused variables
1 Compilation Errors
If a program code contains errors, it cannot be compiled.
To fully control all errors, it is recommended to use strict compilation mode, which is set by the following directive:
#property strict
This mode greatly simplifies troubleshooting.
1.1. Identifier coincides with a reserved word
If the name of a variable or function coincides with one of the reserved words
int char[]; int char1[]; int char() { return(0); }
the compiler returns an error messages:
Figure 1. Errors «unexpected token» and «name expected»
To fix this error, you need to use the correct name of the variable or function.
1.2. Special characters in the names of variables and functions
If names of variables or functions contain special characters ($, @, point):
int $var1; int @var2; int var.3; void f@() { return; }
the compiler returns an error messages:
Figure 2. Errors «unknown symbol» and «semicolon expected»
To fix this error, you need to use correct function or variable names.
1.3. Errors using the switch operator
In the old version of the compiler you could use any values in expressions and constants of the switch operator:
void start() { double n=3.14; switch(n) { case 3.14: Print("Pi");break; case 2.7: Print("E");break; } }
In the new compiler, constants and expressions of the switch operator must be integers, so errors occur when you try to use such constructions:
Figure 3. Errors «illegal switch expression type» and «constant expression is not integral»
In such cases, you can use explicit comparison of numerical values, for example:
void start() { double n=3.14; if(n==3.14) Print("Pi"); else if(n==2.7) Print("E"); }
1.4. Return values of functions
All functions except void should return the declared type value. For example:
int function()
{
}
In strict compilation mode an error occurs:
Figure 4. Error «not all control paths return a value»
In default compilation mode, the compiler returns a warning:
Figure 5. Warning «not all control paths return a value»
If the return value of the function does not match the declared one:
int init() { return; }
an error is detected during strict compilation:
Figure 6. Error «function must return a value»
In default compilation mode, the compiler returns a warning:
Figure 7. Warning ‘return — function must return a value»
To fix such errors, add the return operator with the return value of the corresponding type to the function code.
1.5. Arrays in function arguments
In function arguments, arrays are now passed by reference only.
double ArrayAverage(double a[]) { return(0); }
In strict compilation mode, this code will cause an error:
Figure 8. Compiler error «arrays passed by reference only»
In default compilation mode, the compiler returns a warning:
Figure 9. Compiler warning «arrays passed by reference only»
To fix this error, you must explicitly specify that the array is passed by reference by adding the prefix & before the name of the array:
double ArrayAverage(double &a[]) { return(0); }
It should be noted that now constant arrays (Time[], Open[], High[], Low[], Close[], Volume[]) cannot be passed by reference. For example, the following call:
ArrayAverage(Open);
regardless of the compilation mode leads to an error:
Figure 10. Error ‘Open’ — constant variable cannot be passed as reference
To avoid these errors, copy the required data from the constant array:
double OpenPrices[]; ArrayCopy(OpenPrices,Open,0,0,WHOLE_ARRAY); ArrayAverage(OpenPrices);
2. Runtime Errors
Errors that occur during the execution of the program code are called runtime errors. Such errors usually depend on the state of a program and are associated with incorrect values of the variables.
For example, if a variable is used as an index of array elements, its negative values will inevitably lead to the Array out of Range error.
2.1. Array out of Range
This error often occurs in indicators when accessing indicator buffers. The IndicatorCounted() function returns the number of bars unchanged since the last indicator call. Indicator values on previously calculated bars do not need recalculation, so for faster calculations you only need to process the last few bars.
Most indicators that use this method of calculation optimization look like this:
int start() { if (Bars<100) return(-1); int counted_bars=IndicatorCounted(); if(counted_bars<0) return(-1); int limit=Bars-counted_bars; if(counted_bars==0) { limit--; limit-=10; } else { limit++; } for (int i=limit; i>0; i--) { Buff1[i]=0.5*(Open[i+5]+Close[i+10]) } }
Often the case of counted_bars==0 is handled incorrectly (the initial limit position should be reduced by the value equal to 1 + maximum index relative to the loop variable).
Also, remember that at the time of the start() function execution we can access elements of arrays of indicator buffers from 0 to Bars ()-1. If you need to work with the arrays that are not indicator buffers, increase their size using the ArrayResize() function in accordance with the current size of indicator buffers. The maximum index of the element to address can also be obtained by calling ArraySize() with one of the indicator buffers used as an argument.
2.2. Zero Divide
The Zero Divide error occurs when a divisor in a division operation is equal to zero:
void OnStart() { int a=0, b=0,c; c=a/b; Print("c=",c); }
When you run this script, an error message appears in the Experts tab and the program shuts down:
Figure 11. Error message «zero divide»
Usually this error occurs when the value of the divisor is determined by the values of any external data. For example, if trade parameters are analyzed, the value of the used margin is equal to 0 if there are no open orders. Another example: if the analyzed data are read from a file, we can not guarantee correct operation if the file is not available. So you should take into account such cases and process them correctly.
The easiest way is to check the divisor before the division operation and report an incorrect parameter value:
void OnStart() { int a=0, b=0,c; if(b!=0) {c=a/b; Print(c);} else {Print("Error: b=0"); return; }; }
This does not cause a critical error, but a message about an incorrect parameter value appears and program shuts down:
Figure 12. Incorrect divider message
2.3. Use of 0 instead of NULL for the current character
In the old version of the compiler 0 (zero) could be used as an argument in functions that require specification of a financial instrument.
For example, the value of the Moving Average technical indicator for the current symbol could be requested as follows:
AlligatorJawsBuffer[i]=iMA(0,0,13,8,MODE_SMMA,PRICE_MEDIAN,i);
In the new compiler you should explicitly specify NULL to specify the current symbol:
AlligatorJawsBuffer[i]=iMA(NULL,0,13,8,MODE_SMMA,PRICE_MEDIAN,i);
In addition, the current symbol and chart period can be specified using the Symbol() and Period() functions.
AlligatorJawsBuffer[i]=iMA(Symbol(),Period(),13,8,MODE_SMMA,PRICE_MEDIAN,i);
2.4. Unicode strings and their use in a DLL
Strings are now represented as a sequence of Unicode characters.
Remember this and use appropriate Windows functions. For example, when using the wininet.dll library instead InternetOpenA() and InternetOpenUrlA(), you should call InternetOpenW() and InternetOpenUrlW().
The internal structure of strings has changed in MQL4 (now it takes 12 bytes), and the MqlString structure should be used when passing strings to DLL:
#pragma pack(push,1) struct MqlString { int size; LPWSTR buffer; int reserved; }; #pragma pack(pop,1)
2.5. File sharing
In the new MQL4, FILE_SHARE_WRITE and FILE_SHARE_READ flags should explicitly be specified for shared use when opening files.
If the flags are absent, the file is opened in exclusive mode and cannot be opened by anyone else till it is closed by the user who opened it.
For example, when working with offline charts share flags should be explicitly specified:
ExtHandle=FileOpenHistory(c_symbol+i_period+".hst",FILE_BIN|FILE_WRITE|FILE_SHARE_WRITE|FILE_SHARE_READ);
For more details please read Offline Charts in the New MQL4.
2.6. Datetime Conversion
Conversion of datetime to a string now depends on the compilation mode:
datetime date=D'2014.03.05 15:46:58'; string str="mydate="+date;
For example, trying to work with files whose name contains a colon causes an error.
3. Compiler Warnings
Compiler Warnings are informative and are not error messages, but they indicate possible error sources.
A clear code should not contain warnings.
3.1. Names of global and local variables coincide
If variables on the global and local levels have similar names:
int i; void OnStart() { int i=0,j=0; for (i=0; i<5; i++) {j+=i;} PrintFormat("i=%d, j=%d",i,j); }
the compiler displays a warning showing the line number on which the global variable is declared:
Figure 13. Warning «declaration of ‘%’ hides global declaration at line %»
To fix such warnings correct names of global variables.
3.2. Mismatch of Types
The new compiler has a new typecasting operation.
#property strict void OnStart() { double a=7; float b=a; int c=b; string str=c; Print(c); }
In strict compilation mode the compiler shows warnings if types mismatch:
Figure 14. Warnings «possible loss of data due to type conversion» and «implicit conversion from ‘number’ to ‘string’
In this example, the compiler warns about the possible loss of accuracy for different data types assigned and implicit conversion from int to string.
To fix the warning use explicit typecasting:
#property strict void OnStart() { double a=7; float b=(float)a; int c=(int)b; string str=(string)c; Print(c); }
3.3. Unused Variables
The presence of variables that are not used in the program code (superfluous entities) is not a good habit.
void OnStart() { int i,j=10,k,l,m,n2=1; for(i=0; i<5; i++) {j+=i;} }
Reports of such variables are displayed regardless of the compilation mode:
Figure 15. Warning «variable ‘%’ not used’
To fix it, remove unused variables from your code.
Conclusions
The article describes common problems that may occur during compilation of old programs that contain errors.
In all cases it is recommended to use strict compilation mode for program debugging.
21
Incomplete record of a date in the datetime string
22
Wrong number in the datetime string for the date. Requirements:
Year 1970 <= X <= 3000
Month 0 <X <= 12
Day 0 <X <= 31/30/28 (29 )….
23
Wrong number of datetime string for time. Requirements:
Hour 0 <= X <24
Minute 0 <= X <60
24
Invalid color in RGB format: one of RGB components is less than 0 or greater than 255
25
Unknown character of the escape sequences.
Known: n r t \ ” ‘ X x
26
Too large volume of local variables (> 512Kb) of the function, reduce the number
29
Enumeration already defined (duplication) – members will be added to the first definition
30
Overriding macro
31
The variable is declared but is not used anywhere
32
Constructor must be of void type
33
Destructor must be of void type
34
Constant does not fit in the range of integers (X> _UI64_MAX | | X <_I64_MIN) and will be converted to the double type
35
Too long HEX – more than 16 significant characters (senior nibbles are cut)
36
No nibbles in HEX string “0x”
37
No function – nothing to be performed
38
A non-initialized variable is used
41
Function has no body, and is not called
43
Possible loss of data at typecasting. Example: int x = (double) z;
44
Loss of accuracy (of data) when converting a constant. Example: int x = M_PI
45
Difference between the signs of operands in the operations of comparison. Example: (char) c1> (uchar) c2
46
Problems with function importing – declaration of #import is required or import of functions is closed
47
Too large description – extra characters will not be included in the executable file
48
The number of indicator buffers declared is less than required
49
No color to plot a graphical series in the indicator
50
No graphical series to draw the indicator
51
‘OnStart’ handler function not found in the script
52
‘OnStart’ handler function is defined with wrong parameters
53
‘OnStart’ function can be defined only in a script
54
‘OnInit’ function is defined with wrong parameters
55
‘OnInit’ function is not used in scripts
56
‘OnDeinit’ function is defined with wrong parameters
57
‘OnDeinit’ function is not used in scripts
58
Two ‘OnCalculate’ functions are defined. OnCalculate () at one price array will be used
59
Overfilling detected when calculating a complex integer constant
60
Probably, the variable is not initialized.
61
This declaration makes it impossible to refer to the local variable declared on the specified line
62
This declaration makes it impossible to refer to the global variable declared on the specified line
63
Cannot be used for static allocated array
64
This variable declaration hides predefined variable
65
The value of the expression is always true/false
66
Using a variable or bool type expression in mathematical operations is unsafe
67
The result of applying the unary minus operator to an unsigned ulong type is undefined
68
The version specified in the #property version property is unacceptable for the Market section; the correct format of #property version id “XXX.YYY”
69
Empty controlled statement found
70
Invalid function return type or incorrect parameters during declaration of the event handler function
71
An implicit cast of structures to one type is required
72
This declaration makes direct access to the member of a class declared in the specified string impossible. Access will be possible only with the scope resolution operation ::
73
Binary constant is too big, high-order digits will be truncated
74
Parameter in the method of the inherited class has a different const modifier, the derived function has overloaded the parent function
75
Negative or too large shift value in shift bitwise operation, execution result is undefined
76
Function must return a value
77
void function returns a value
78
Not all control paths return a value
79
Expressions are not allowed on a global scope
80
Check operator precedence for possible error; use parentheses to clarify precedence
81
Two OnCalCulate() are defined. OHLC version will be used
82
Struct has no members, size assigned to 1 byte
83
Return value of the function should be checked
84
Resource indicator is compiled for debugging. That slows down the performance. Please recompile the indicator to increase performance
85
Too great character code in the string, must be in the range 0 to 65535
86
Unrecognized character in the string
87
No indicator window property (setting the display in the main window or a subwindow) is defined. Property #property indicator_chart_window is applied
88
Property ignored, it must be declared on the global scope. The warning is not generated for the following properties: copyright, link, version and strict.
-
#22
Объяснение как исправить некоторые ошибки и предупреждения – пост.
Рассмотрены следующие ошибки:
1) ‘return’ – expressions are not allowed on a global scope
2) ‘char’ – unexpected token
3) ‘.’ – semicolon expected
Предупреждения:
1) variable ‘var’ not used;
2) ‘void’ function returns a value
3) arrays passed by reference only
4) not all control paths return a value
5) declaration of ‘n’ hides global declaration at line
-
#23
Я специально выкладывал не исправленные версии индикаторов, чтобы люди сами исправляли, а не скачивали готовое решение =)
Привет Gravity*, посмотри пожалуйста код этого индикатора что я скинул, там чего то не хватает и я не могу это поправить… Поправь пожалуйста чтобы не было ошибок…. Сейчас их там 3 штуки))
-
Breaк_Bon.mq4
5,9 КБ · Просмотры: 27
-
#24
Привет Gravity*, посмотри пожалуйста код этого индикатора что я скинул, там чего то не хватает и я не могу это поправить… Поправь пожалуйста чтобы не было ошибок…. Сейчас их там 3 штуки))
Привет. Посмотрел, не знаю чего там не хватает и это вроде как советник. Моих знаний тут не хватает.
-
#25
Ребят я не шарю в mql4 написал тут мелкую модификацию МА но нефига не показывает на графике, ощибок в консоли нету, помогите плиз)
-
bearMa.mq4
1,9 КБ · Просмотры: 15
-
#26
Ребят я не шарю в mql4 написал тут мелкую модификацию МА но нефига не показывает на графике, ощибок в консоли нету, помогите плиз)
-
bearMa.mq4
2,4 КБ · Просмотры: 17
-
#27
Кинул на график, нечего не рисует на графике..
Been doing this code in “Include” file. But I am encountering the error “not all control paths return a value. What should I do?
double CalculateTakeProfit (double entryPrice, int takeProfitPips, double GetPipValue)
{
if (bIsBuyPosition == True)
{
double result = 0;
entryPrice = Ask;
result = (entryPrice + takeProfitPips * GetPipValue());
return result;
}
else if (bIsBuyPosition == False)
{
double result = 0;
entryPrice = Bid;
result = (entryPrice - takeProfitPips * GetPipValue());
return result;
}
}
1 Answers
Your if... else
is wrong and you are also not using the variables passed to the function. You are instead referencing another function or overwriting them. Mixing variable types in a calculation can also lead to undesirable results (takeProfitPips should be of type double
). You can also cut a few lines of your code down as follows
double CalculateTakeProfit(double entryPrice, double takeProfitPips, double GetPipValue)
{
if(bIsBuyPosition) return(entryPrice+takeProfitPips*GetPipValue);
else return(entryPrice-takeProfitPips*GetPipValue);
}