Уменьшаю время локальной загрузки QUIK до нуля.- Это просто.
Пользователь
Сообщений: Регистрация: 30.01.2015
21.02.2026 11:49:00
Преамбула: Раньше неоднократно пытался перейти на новые версии терминала, но получение тормозов при старте, и не только, терминала возвращало меня на версию 8.7 ----------------- Неделю назад Сбербанк без объявления перевел своих клиентов на единый брокерский счет, что привело к необходимости установить версию 12 без вариантов. --------------- В конечном итоге пришлось установить версию 12.8.3.4 из установочного пакета разработчиков, так как Сбербанк пока предлагает более глючную версию 12.6 ------------------ Сегодня на хабре написал статью, в которой привел высказывание автора Claude Code: «Мы начинаем видеть мир, в котором клиент жалуется на что-то, а проблема решается за считанные минуты, — сказал он. ------------------- Надеюсь, что когда-то и разработчики QUIK достигнут это. ------------------- Но пока актуальным является высказывание Остапа Бендера:"Спасение утопающих - дело рук самих утопающих" ----------------------- Поэтому, познав все прелести 6-8 минутной загрузки терминала был вынужден решать эту проблему самостоятельно. -------------- Решение: Свои результаты и рекомендации размещаю на этом форуме. Я никого не заставляю делать так, как делаю сам. Я лишь рассказываю что и как делаю и что из этого получилось. ================ В результате исследований мне удалось уменьшить время загрузки с 240...320 сек до 110 секунд. Из этих 110 секунд 70 секунд тратится на локальную загрузку терминала. Для тех, кто не читал мой пост по этой проблеме, поясняю: ---------------- Процесс старта терминала QUIK условно разделяю на три этапа 1) загрузка терминала на основе сохраненной на ПК информации до подключения к серверу брокера 2) ввод логина и пароля и подключение к серверу 3) ввод кода со смартфона для двухфакторной аутентификации и получения текущей информации с сервера. ====================== В этой теме рассказываю как сделал время первого этапа равным нулю. -------------------------------- Это можно сделать, если есть свободное место на жестком диске в размере объема памяти на ПК. Для этого включаю на ПК либо гибридный режим сна, либо режим Гибернации. Предпочитаю Гибернацию. Режим гибридного сна отличается от режима сна тем, что образ памяти сохраняется на диске и это позволяет восстановить состояние ПК при выходе из сна после отключения сетевого питания. ----------------- Режим Гибернации отличается от режима сна тем, что ПК полностью выключается, как обычно. Но перед выключением ОС запишет на диск образ памяти и загрузит его при включении. -------------------- Это не только исключает локальную загрузку QUIK, но и сокращает загрузку Windows. ============= При этом последовательность действий такая. Надо не закрывать QUIK, а лишь разорвать соединение с сервером. Перевести ПК в режим Гибернации. --------------------- После включения получаю состояние ПК в момент выключения и перехожу ко 2-ому этапу загрузки терминала QUIK.
Уменьшить объем памяти и время старта QUIK-это просто
Пользователь
Сообщений: Регистрация: 30.01.2015
19.02.2026 11:00:54
Всем, добрый день, -------------------- Ранее я уже писал, что в старте терминала QUIK условно можно выделить 3 этапа. ------------ Первый - автономная загрузка терминала до момента запроса пароля. Второй - ожидание кода авторизации со смартфона Третий - загрузка с сервера брокера текущего состояния торгов ------------------------ В этой теме я расскажу как сократить первый этап. Для этого написал bat файл (nk.bat)
Код
@echo off
del /f /q "*.log"
start "" "info.exe"
создайте его текстовым редактором и поместите в папку QUIK, где находится info.exe ---------------- чтобы запустить QUIK надо запустить этот файл (щелкнуть вместо info.exe по файлу nk.bat) ==================== Зачем нужен этот файл? Проблема в том, что терминал QUIK зачем-то при старте загружает в память info.log (этот файл особо влияет на объем занимаемой памяти) Данный bat файл сначала сотрет все log файлы, а потом запустит QUIK. ---------------------- Если не хотите удалять все лог файлы, то вместо *.log поставьте info.log . Этого достаточно, чтобы получить сокращение. ----------------------------- У меня при использовании этого bat файла объем памяти занятой терминалом сокращается с 900 Мбайт до 600 Мбайт. ------------------------- Для сбора статистики просьба сообщить как влияет bat файл в Вашем случае на старт терминала.
Причины тормозов при старте. Вскрытие покажет.
Пользователь
Сообщений: Регистрация: 30.01.2015
18.02.2026 06:54:12
Всем,добрый день, Предлагаю обсудить проблему тормозов при старте терминала. ------------------- До последней прошедшей недели пользовался версией 8.7 Неоднократно пытался установить более свежую версию, но всегда получал тормоза и возвращался к 8.7 ------------------- На прошедшей неделе Сбербанк перешел на единый счет и мне пришлось устанавливать версии 12 так как исчезли параметры по фьючерсам. ----------------------- Сначала установил версию 12.6 которая выложена у Сбербанка. Получил увеличение объема занятой памяти терминалом в 2 раза с 600 Мбайт до 1200Мбайт и жуткие тормоза -------------------------- По совету разработчиков установил новую версию 12.8.4 Объем занимаемой памяти при старте сократился с 1200 до 900. Почему он изменяется ? ----------------------------- Самое странное, что во время торгов объем занимаемой памяти постепенно уменьшается. Если никаких действий не совершать , то объем занимаемой памяти сокращается до 50Мбайт. Но если начать что-то делать, то он снова увеличивается, но не более 700, а иногда и не более 350Mбайт. -------------------- Трудно что-либо придумать почему так происходит. -------------------------- Но так как максимальный объем всегда получается при старте и это приводит к очень длительной загрузки, то давайте обсудим возможные причины этого. ---------------------------- Время старта можно разделить на три этапа. Первый - запуск терминал. Второй -ввод пароля Третий- ввод кода со смартфона. ------------------- самый короткий интервал - второй. Задержки при нем пока не будем рассматривать. ----------------- Сначала я предположил, как и многие на форуме, что проблема медленной загрузки в длительном получении начальной информации с сервера брокера. Но теперь у меня другая рабочая гипотеза. ================= Основа ее в том, что время первого и третьего этапов одинаковое. Характерно , первый и третий этапы имеют одинаковое значение времени исполнения -в моем случае это по 4 минуты каждый. Но в первом этапе нет никакой связи с сервером. Получается, что соединение с сервером не влияет ни на объем занимаемой памяти ни на время старта. ================= Поэтому я считаю, что указанные проблемы (время старта и объем занимаемой памяти ) связаны исключительно с вычислениями, которые совершает терминал при старте. ===================== Более того, эта гипотеза подтверждается тем, что в третьем этапе терминал сначала уменьшает объем занятой памяти до 350 Мбайт. А потом медленно начинает увеличивать его до тех же 900 Мбайт что и в первом этапе. ===================== Как правило время и занятая память у всех пользователей разные. Это связано с конкретной настройкой интерфейса. Т е числом окон с графиками, числом таблиц и стаканов и числом скриптов на Lua. ===================== Полагаю, что максимальный объем памяти в момент старта возникает по причине, возможно не единственной, но одной из основных, РАСЧЕТА ИНДИКАТОРОВ НА LUA. ======================== Известно, что индикаторы на луа вычисляются несколько раз при открытии графика. ---------------------------------------- В результате такого расчета возникает куча мусора от этих расчетов, так как сборщик мусора при старте не работает. ------------------------ В начале третьего этапе терминал удаляет все расчеты первого этапа и память сокращается. ------------------------- Так как у меня установлена подписка вручную, то ничего нового при соединении с сервером не приходит. --------------------------------- Терминал снова начинает свой расчет индикаторов и графиков и это увеличивает объем памяти до максимума для всей сессии. ------------------------- Но это еще не все. Длительный расчет индикаторов и графиков при старте связан с тем, что хотя все свечи от первой до последней имеются в начале расчета, Расчет индикаторов, а возможно и построение графиков выполняется для каждой свечи отдельно в цикле Calculate. Так как большинство индикаторов - это интегральная обработка данных в окне, то при 3000 свечах (это то что сохраняет сервер) для каждого индикатора в цикле при старте вычисления в скользящем окне выполняются 3000 и более раз. Если история сохраняется на ПК, то со временем тормоза и объем занимаемой памяти растет. -------------------------------- Предположу, что это тупиковый момент в развитии QUIK и он требует кардинальных изменений. --------------------------- Еще заметил, что число подписанных инструментов не влияет на объем и время при старте. ------------------------ Резюме: На объем и время при старте влияет повторный расчет индикаторов на Lua, циклический расчет индикаторов (возможно и графиков) для каждой свечи при старте и размер сохраненной истории, по которым считаются эти индикаторы. ---------------------------- Тот факт, что объем памяти при старте версии 12.6 на 300 Мбайт больше, чем при старте версии 12.8 говорит о том, что, как говорил сатирик "что-то не так в консерватории" ------------------------- Возможно, решением было бы исключение повторных расчетов индикаторов и принудительный запуск сборщика мусора при завершении третьего этапа. ------------------------- Исследования продолжаются.... ========================= P.S. В настоящее время кувыркаюсь с единим счетом. Постоянно исчезают позиции по фьючерсам. Кроме того, как уже писал на форуме,но без ответа, сваливание в кучу информации об акциях и фьючерсах имеет не только систематические ошибки, но и противоречит ФЗ "О рынке ЦБ" Но это уже другая история
Окно сообщений закрывает окно ввода
Пользователь
Сообщений: Регистрация: 30.01.2015
17.02.2026 07:28:02
Версия 12.8.3.4 при старте терминала:
Варианты построения робота
Пользователь
Сообщений: Регистрация: 30.01.2015
16.02.2026 12:03:54
Тема предназначена для начинающих писателей роботов ------------------ Вариантов написания роботов в терминале QUIK на луа множество. Кратко расскажу о них на основе своего опыта. ------------- По функциональному назначению можно разделить роботов на 3 класса. 1) Роботы , которые прогнозируют моменты изменения рынка для изменения позиции.Роботы-советники 2) Роботы, которые совершают транзакции на основе простейших алгоритмов и управляют стоп-заявками. 3) Роботы , которые объединяют функции 1 и 2. --------------- Любого робота можно написать либо в виде скрипта-индикатора,далее робот-индикатор, либо в виде скрипта не индикатора, далее робот-скрипт. Разница в том, что робот на основе индикатора запускается на графике инструмента, которым торгуем, а робот-скрипт, не индикатор, запускается без автоматической привязки к инструменту. ------------------ Робот индикатор самый простой в написании. в нем не надо подписываться на инструмент. В него автоматически поступают данные лишь конкретного инструмента. -------------------- Недостаток его в том, что его надо запускать на графике инструмента и следовательно надо открывать окно с графиком. Но обычно так все и торгуют. --------------- Робот-скрипт является универсальным и позволяет построить рой роботов, которые могут обмениваться данными и проводить расчеты параллельно. ---------------------------------- На этом пока все. Если есть вопросы, то отвечу --------------------- Продолжение следует...
Скрипт для обнаружение задержек данных
Пользователь
Сообщений: Регистрация: 30.01.2015
16.02.2026 08:30:40
Для обнаружения ситуаций, когда соединение с сервером нормальное, но с сервера не поступают данные ВООБЩЕ написал скрипт. Скрипт пишет данные о времени последней связи с сервером и фиксирует задержу в лог файле. скрипт в стадии тестирования. Желающие присоединяйтесь.
Код
function HMS(T)
local hms=0; for s in T:gmatch("[^:]+") do hms=60*hms+tonumber(s) end
return hms;
end
MaxTime=30
function main()
while run do
TS=getInfoParam("SERVERTIME")-- Время сервера
TW=getInfoParam("LASTRECORDTIME")-- Время последней записи
if TS and TW then
TL=getInfoParam("LASTPINGTIME") --Время последней проверки связи
if TL_OLD and TL~=TL_OLD then
local str=os.date()
if D>MaxTime then str=str..",задержка(сек)="..D; end
Log:seek("set",begin);
Log:write(str.."\n"); Log:flush();
if D>MaxTime then begin=Log:seek(); end
end
TS_HMS=HMS(TS); D=math.abs(HMS(TW)-TS_HMS)
end
TL_OLD=TL;
sleep(200);
end
end
function OnInit(ph)
ph=ph:sub(1,ph:len()-3).."log";
Log = io.open(ph, "w");Log:close(); Log = io.open(ph, "r+");
begin=Log:seek();
D=0;
run=true
end
На едином счете клиентский портфель считается с грубыми системными ошибками
Пользователь
Сообщений: Регистрация: 30.01.2015
12.02.2026 19:53:46
Ув.Разработчики. Сообщаю Вам о следующих грубейших ошибках расчета клиентского портфеля. ------------------- При торговле фьючерсами не может быть никаких маржинальных сделок. Т е такие показатели как НПР1, НПР2, Требование, УДС, и все показатели со словом маржа не существуют для фьючерсов. Так как никаких займов брокер клиентам на фьючерсном рынке не дает и не может давать ни по регламенту ни по закону. Указание Банка России от 12.02.2024 N 6681-У "О требованиях к осуществлению брокерской деятельности при совершении брокером отдельных сделок за счет клиента" (Зарегистрировано в Минюсте России 02.07.2024 N 78736) У Вас же все эти показатели считаются для фьючерсов как для акций. -------------------- Если Вы полагаете, что Вы считаете все правильно, то дайте ссылку на документ, где указана методика расчета этих показатель для фьючерсов и объясните о какой марже идет речь, что такое нач и кон маржа для фьючерсов. ---------------------- Брокер Сбербанк это объяснить не смог. ==================== Есть ошибки и в других таблицах, но давайте решим сначала по клиентскому портфелю для единого счета.
Причина очень медленной загрузки QUIK
Пользователь
Сообщений: Регистрация: 30.01.2015
11.02.2026 16:01:09
В данной теме раскрою вам секрет, почему при старте терминала QUIK приходится ждать не просто минуты, а десятки минут. ------------------------ Все происходит очень просто. Любой желающий может повторить мой эксперимент. ------------------- Берем версию QUIK 12.8.3 Открываем диспетчер задач и наблюдаем объем занимаемой памяти Начальный объем примерно 170Мбайт, но он быстро увеличивается до 400 и потом медленно, минут за 5 до 800 Мбайт. Это происходит загрузка сохраненных данных торгов и истории. ---------------------------- После этого QUIK запрашивает логи и пароль и через некоторое время код со смартфона. --------------------- После этого QUIK сбрасывает объем занимаемой памяти до 400 МБайт и начинает через интернет получать данные для настройки графиков и таблиц . ------------------------- В конечном итоге он займет 1200 Мбайт (если удалите всю накопленную историю, то получите от 850 до 1100 Мбайт) --------------------- Т е ему надо получить с сервера данные и чего-то там наплодить размером от 400 до 800 Мбайт ======================= При этом процессе Вам надо наблюдать работу сети ================== Прикольно, но факт. Обмен данными с сервером происходит малыми посылками. Например , у меня для полной загрузки терминал QUIK обменялся с сервером аж 1 млн. пакетов. При этом канал Ethernet и Интернет был загружен аж на 0.1% !!! Т е при возможности обмена на скорости 100Мбит/s, сервер и терминал обмениваются мелкими пакетами на скорости в среднем 50 Кбит/s Т е в 2000 раз медленнее, чем возможно. -------------------------- Причем возникает естественный вопрос. Нафига надо обмениваться 1 000 000 сообщений при старте? Почему невозможно все передать на скорости в 1000 раз большей и пакетами в 1000 раз длиннее? ------------------------- Но это риторические вопросы. ================= Учитывая, что версии 12 занимают памяти в 2 раза больше, чем версии 8 (ранее писал 1200 -12 версия 600- .8 версия) следует ожидать еще большего зависания терминалов в следующих версиях. ------------------------------ Еще замечу, что у меня очень малая подписка на инструменты и индикаторы, открыто обычно не более 5 графиков и 2 стакана и таблицы, включая ТТП, с сотней инструментов. На бирже инструментов тысячи. ------------------------------------ Представляю сколько ждут начинающие, которые открывают сотню графиков и стаканов. ============================ Поэтому начинайте загружать терминал вечером. Утром все будет загружено. (Шутка)
Приколы терминала QUIK
Пользователь
Сообщений: Регистрация: 30.01.2015
11.02.2026 09:14:51
Так как после установки версии 12.6 начались чудеса, то решил провести подробные исследования процесса загрузки терминала. ------------- Для начала я восстановил архив котировок, который у меня составлял 100 Мбайт и запустил QUIK ---------------------- Начальный размер памяти при старте составил 800Мбайт:
терминал грузился 5 минут и конечный размер составил 1100 Мбайт
после того как ввел пароль терминал снова стал загружать данные с сервера и далал это следующие 5 минут начал с 800Мбайт
в результате получилось 1200Мбайт
============================== После этого я решил перезаказать данные для построения графиков . У меня лишь графики сбербанка и индекса Загрузка началась с 400Мбайт
В итоге загрузка данных продолжалась следующие 10 минут и результат 1200 Мбайт занятой памяти
================================ А теперь внимание, будет самое интересное. ==================== У меня есть скрипт советника, который без проблем работал уже 2 года на версии 8.7 ---------------------------------- Я его загружаю на версию 12.6 и о ЧУДО получаю 22 Мбайт !!! занятой памяти терминалом. Но при этом получаю большую загрузку процессора (на 8.7 такого не было) ----------------------- Тогда я удаляю свой скрипт Загрузка процессора возвращается в норму А что же с занятой памятью? 68 Мбайт вместо 1200 Мбайт
Зависание версии 12.6. при откреплении окна.
Пользователь
Сообщений: Регистрация: 30.01.2015
11.02.2026 07:52:03
Спешу поздравить всех с ляпом в версии 12.6. ---------------------- Если открепить окно с графиком для перемещения его за пределы основного окна QUIK, например на второй монитор, то получаем зависание терминала. ----------------------- При этом загрузка процессора возрастает с 3% до 30%. -------------------------- Если это окно свернуть, то все возвращается обратно.
Вопрос по единому счету
Пользователь
Сообщений: Регистрация: 30.01.2015
10.02.2026 17:29:05
Сегодня сбербанк перешел на единый счет Торгую лишь фьчерсами В результате нигде не могу найти сумму свободных средств по фьючерсам Кто знает где искать? ---------------- В сбербанке не знают.
Тест быстродействия многопоточного бота
Пользователь
Сообщений: Регистрация: 30.01.2015
06.02.2026 11:54:44
Тема для начинающих писателей роботов ----------------------- Алгоритм теста: --------------------- Скрипт на Lua в терминале QUIK обрабатывает колбеки и выставляет заявки по требованию приложения. --------------------------------- Полученные события, кроме событий OnOrder, OnStopOrder, OnTransReply, которые обрабатывает сам, скрипт рассылает приложениям. ------------------------------------- Для каждого инструмента или класса работает отдельное приложение, которое получает данные от скрипта , прогнозирует изменение позиции и отсылает требования на выставление или снятие заявок скрипту. ----------------------------------- Приложения можно реализовать в виде отдельного скрипта для терминала QUIK , либо отдельного приложения Windows. -------------------------------------- Механизм обмена сообщениями реализован через файлы, поэтому он один и тот же как для скриптов так и для приложений. ------------------------------------- Весь процесс обмена сообщений пишется в лог файлы. Результаты теста такие: это логи событий, от скрипта к приложениям для инструментов. Первое число - задержка получения события приложением. В данном тесте задержка равна кванту windows, который установлен 2ms. В конце каждого события через символ "> " указано сколько всего событий в очереди.
это сообщения от приложений. Сообщение отсылается на каждое событие по данному инструменту: В сообщении первым указано время задержки приема его скриптом. Если сообщения от приложения скрипт получил одновременно, т е они были все записаны до того момента как скрипт их прочитал, то в таблице у них одно время и они отделены пустой строкой от следующих сообщений.
Отключать сборщик мусора при работе с колбеками. Вы прикалываетесь?
Пользователь
Сообщений: Регистрация: 30.01.2015
04.02.2026 07:26:30
Тема для начинающих писателей роботов. -------------------------- Рано и поздно у Вас возникнет желание, либо необходимость, ускорить работу вашего робота. Да, это скорее всего, реально Но существует много ненужных телодвижений там, где что-либо делать не имеет смысла. ----------------------- Например, отключать сборщик мусора в колбеках , а потом его включать. ------------------------- Поясняю, почему в этом нет смысла. ====================== Для этого напишем следующий тест:
Код
local t1=nkQt.count(); local x1=collectgarbage("count");
print("занято память КБ= "..x1//1)
local e={}; for j=1,100000 do e[j]=j end --создаем таблицу
local t1=nkQt.count(); local x=collectgarbage("count")-x1;
e=nil; --удаляем таблицу в мусор
print("объем мусора КБ="..x//1)
local t1=nkQt.count(); local x=collectgarbage("collect")
print("сборка мусора(ms)="..(0.1*(nkQt.count()-t1))/1000)
local x=collectgarbage("count") print("занято память КБ= "..x//1)
В тесте создаем таблицу и заполняем ее. Потом удаляем таблицу. В результате у нас создается объем мусора в 2 Мбайта. Сборщик мусора собирает его за 0.038 ms(0.000038 сек). --------------------------- Как уже писал на форуме, Скрипт на луа из-за Sleep(1) в Main имеет задержу исполнения не менее 16 ms, равную кванту времени Windows. -------------------------------------- Многие, по рекомендации разработчиков QUIK, ставят Sleep(1000) т.е. задержку в 1000 ms ---------------------------------------- Таким образом, если до очередного срабатывания колбека включится сборщик, то он задержит колбек на 0.04 ms. и сборщик мусора создаст дополнительную задержку не более, чем в 0.25% от задержки от Sleep(1) ------------------------ Можно перепрограммировать системный таймер и получить задержку не менее 1 ms. ------------------------------ Тогда задержка от сборщика составит не более 4% от задержки Sleep(1), но это всего 0.00004 секунды. -------------------------------------------------- Полагаете, что с этим надо бороться. Вы серьезно? ======================================================== Для справки: ------------------ За квант времени в 1-2 мс поток получает возможность выполнить многие миллионы инструкций процессора. ------------------ Не забывайте и тот факт, что информация с биржи, поступающая к Вам через интернет, приходит с задержкой десять и более ms.
Тест ускорения скриптов
Пользователь
Сообщений: Регистрация: 30.01.2015
01.02.2026 09:24:56
Ранее привел результаты теста обмена данными через файлы.
Основной вывод : ------------------------------ SLEEP в MAIN приводит к задержке работы скриптов минимум на квант Windows. Его величина по умолчанию равна 16 ms. ------------------------------- Прикольно то, что до Windows10 2004 года этот квант можно было сравнительно просто уменьшить до 0.5ms. Теперь это сделать просто не получится. -------------------------------- Для решения проблемы сделал другую функцию sleep, которая уменьшает этот интервал до 2 ms. ---------------------- Результаты теста: Тест с функцией sleep(1) из библиотеки QLUA Задержка 15985 мкс округляем 16 ms
Таким образом удалось уменьшить задержку в скриптах в 8 раз.
Передача данных из скрипта в скрипт QUIK или приложение
Пользователь
Сообщений: Регистрация: 30.01.2015
30.01.2026 13:50:35
Алгоритм теста: --------------- два скрипта. ------------------ Первый скрипт: Содержит колбеки. Полученная в колбеке таблица преобразуется в строку и записывается в файл. ------------------------- Второй скрипт: В функции main читает новые строки из файла. Если строки есть, то выводит их в лог файл. -------------------------- В результате получаем в лог файле вторго скрипта таблицы из колбеков первого. Ниже приведено содержимое лог файла при купокупки и продажи акции на демо сервере ---------------------------- В строке записана следующая информация 136, млсек --задержка приема данных вторым скриптом в ms относительно колбека первого скрипта. 307620878521: --значение счетчика в мкс во втором скрипте 307620741599: --значение счетчика в мкс в первом скрипте 6, мксек-- время на передачу таблицы из колбека в функцию вывода в файл в первом скрипте, Param={sec_code=RUAL,class_code=QJSIM}, -- таблица параметров колбека 14 мксек -- время на преобразование таблицы в строку.
Гонял тест на версии 12.8.3.4 и обнаружил такое : --------------------- Если принудительно не удалять таблицы параметров в колбеках, то сборщик мусора их не собирает. ----------------------------- В итоге растет объем используемой памяти. --------------------- Предлагаю заинтересованным проверить это. Подтвердить или опровергнуть.
Вопрос к Разработчикам. Можете объяснить этот прикол?
Пользователь
Сообщений: Регистрация: 30.01.2015
29.01.2026 11:27:32
Прикол от Сбербанка. Торги сегодня акции сбербанка Соединение есть, но данные не приходят уже 5 минут. Потом данные поступают ---------------------- Такой прикол наблюдается эпизодически уже не первый день. ------------------- Можете объяснить:
Универсальный обработчик событий - это просто
Пользователь
Сообщений: Регистрация: 30.01.2015
29.01.2026 06:21:45
Эта тема для начинающих писателей роботов на Lua в QUIK. Cкрипт, на основе которого можно строить различных роботов. В нем реализована очередь , что обеспечивает обработку всех событий
Код
TQ={}; jQr,jQw=0,0; --очередь
----------------------------
function main()
while run==1 do wait_connect()
while run==2 do
while jQw>jQr do local n=jQr+1;local t=TQ[n];TQ[n]=nil;jQr=n;t.fun(t);end
sleep(10)
end
end
end
-----------------------------
local function wQ(t,fun) --функция записи обработчика в очередь
if fun then
t.fun=fun; local n=jQw;if jQr==jQw then jQw=0;jQr=0;end n=jQw+1;TQ[n]=t; jQw=n;
end
end
-----функции обратного вызова QLua------------------------
function OnDepoLimit(t) wQ(t,setPos); end
----------------------------
function OnFuturesClientHolding(t) wQ(t,setPos); end
-----------------------------
function OnOrder(t) wQ(t,Order); end
----------------------------
function OnStopOrder(t) wQ(t,StopOrder); end
-----------------------
function OnTransReply(t) wQ(t,TransReply); end
----------------------------
function OnParam(c, s) wQ({sec_code=s,class_code=c},Param); end
----------------------------
function OnQuote(c, s) wQ({sec_code=s,class_code=c},Quote); end
------------------------
function OnDisconnected() run=1 end
-------------------------
function OnConnected(flag) Trans,Tsec,Tclas,Tclient,pos={},{},{},{},{} run=1 end
---------------------------
function CloseTH() if td_id then DestroyTable(td_id); end run =0 if f then f:close() end end
------------------------------
function OnStop(s) CloseTH();return 1000 end
---------------------------
function OnClose() CloseTH() end
---------------------------
function OnCleanUp() run=1; end
---------------
function OnInit(path) OnConnected(true); end
Функции обратного вызова записываются одинаково. В них мы помещаем в очередь TQ полученные параметры и имя функции, которая обрабатывает данное событие. Функция обработки событий реализуется отдельно. Если функция обработки события отсутствует, то событие не будет записываться во очередь.
Изменение параметров встроенных индикаторов из скрипта
Пользователь
Сообщений: Регистрация: 30.01.2015
27.01.2026 07:24:22
Добрый день, --------------------- Сейчас в QLUA есть возможность перезапустить скрипт индикатора при изменении параметров в окне настройки индикатора. ----------------------- Предлагаю реализовать возможность изменения параметров встроенного индикатора из скрипта. ================== Поясняю: -------------------- Да, можно написать копию встроенного индикатора на луа и таким образом решить задачу изменения параметров индикатора. ----------------- Но такое решение работает раз в 100 медленнее, чем встроенное. -------------------- Кроме того, в терминале QUIK много встроенных индикаторов и нет смысла делать их в виде копии на луа, либо использовать Вашу библиотеку скриптов на луа и изучать алгоритмы индикаторов.
алгоритм бота - стопа
Пользователь
Сообщений: Регистрация: 30.01.2015
09.01.2026 07:55:35
Предлагаю обсудить один из моих вариантов бота, который управляет стопом. ------------------------- Конструктивная критика приветствуется. =================== Открываем график с инструментом , помещаем на него скрипт-индикатор бота и совершаем сделки. ---------------------- Бот имеет два режима. ---------------------------------- В первом режиме бот автоматически установит стоп, если есть позиция по инструменту. -------------------------- Во втором режиме бот начнет управлять стопом, если руками установить стоп в любую позицию. Когда бот обнаружит установленный стоп, он будет его переставлять в соответствии с заданным алгоритмом. ---------------------- Если позиция закроется, то стоп будет снят. При появлении новой позиции бот будет управлять стопом. ---------------------------------- Чтобы бот перестал управлять, надо снять стоп руками. --------------------------------- При старте QUIK бот начинает работать в установленном режиме.
Торговый советник - это просто.
Пользователь
Сообщений: Регистрация: 30.01.2015
04.01.2026 19:07:20
Тема для начинающих писателей роботов. Продолжение темы:
------------------------- Чтобы сразу не сливать депозит, лучше написать робот-советник, который на истории покажет , что он сможет. --------------------------- Напишем такой советник. Для этого в приведенный по ссылке выше скрипт добавим расчет прибыли/убытков при совершении сделок по приведенному ранее(см ссылку) алгоритму ------------------------ Примем что брокер и биржа забирают com=0.006 (0.6%) Расчет будет проводить в %, количество лотов примем равным 1. Сделки совершаются в long и short. В результате получился вот такой скрипт:
Код
function OnCalculate(i)
local buy,sel;
if i==1 then OnChangeSettings()
i1=0; t={}; Prof=0; Los=0; Q=Settings.Q; com=Settings.com; ProfAll=0; BS=0;
elseif i1~=i then
ind(i1,Settings.tag,t);
if #t>0 then
local Oi,Hi,Li=O(i1),H(i1),L(i1) local prib=0; local z=T(i1); x=t[#t];
params.DATE=100*(100*z.year+z.month)+z.day;
params.TIME=100*(100*z.hour+z.min)+z.sec;
if Li>x and 0>=BS then buy=Li-0.05; if BS==0 then prib=-com; else prib=(1-com)*(price/Oi-1); end BS=1; price=Oi; params.YVALUE=buy-0.1; end
if x>Hi and BS>=0 then sel=Hi+0.05 if BS==0 then prib=-com; else prib=(1-com)*(Oi/price-1); end price=Oi; BS=-1; params.YVALUE=sel+0.1; end
if buy or sel then
ProfAll=ProfAll+prib; params.TEXT=(0.1*(1000*ProfAll//1)).."%"
if ProfAll>=0 then params.B=255 params.R=0 else params.B=0 params.R=255 end
Label=AddLabel(Settings.tag,params);
end
end
end
i1=i;
return buy,sel;
end
function Init()
local t={} Settings.line =t;
t[#t+1]={Name = "Bu",Color = RGB(0,255,0),Type = 10,Width = 4};
t[#t+1]={Name = "Se",Color = RGB(255,0,0),Type = 11,Width = 4};
return #t; end
function OnDestroy () DelAllLabels(Settings.tag) end
function OnChangeSettings()
DelAllLabels(Settings.tag);
end
а это результат его тестирования на демо-сервере:
Данные с индикатора. Просто и быстро.
Пользователь
Сообщений: Регистрация: 30.01.2015
04.01.2026 09:18:57
Тема для начинающих писателей роботов. ---------------------------- Ранее на форуме я рассказывал как построить робота-эксперта на основе двух любых индикаторов отображаемых на графике.
--------------------------- В этой теме поясню как читать данные с индикатора быстро и просто. ------------------------ Напишем пример скрипта индикатора, который считывает значения индикатора moving в массив t и выводит на график значения индикатора смещенные на 0.1 в виде зеленых треугольников . --------------------------- При этом сделаем так, чтобы индикатор не тратил время на считывание тиков, а читал значение индикатора один раз в момент появления новой свечи цены.
В моих роботах именно в этот момент и принимается решение о выставлении заявки.
Т е решение принимается в момент открытия новой свечи . ------------------------------ Если робот будет торговать на интервала 30 минут, то чтение индикатора будет происходить всего 1 раз в 30 минут именно на открытии новой свечи.
Код
Settings={ tag="moving", Name = '*nk_test'}
function OnCalculate(i)
if i==1 then i1=0; t={};
elseif i1~=i then local M=getNumCandles (Settings.tag);
if M>=i then x=getCandlesByIndex(Settings.tag,0,i1,1); if x then t[#t+1]=x[0].close end i1=i; end
end
i1=i; if #t>0 and t[#t]~=0 then return t[#t]+0.1; end
end
function Init()
local t={} Settings.line =t;
t[#t+1]={Name = "Bu",Color = RGB(0,255,0),Type = 10,Width = 3};
return #t;
end
это результат:
Торговые роботы без программирования
Пользователь
Сообщений: Регистрация: 30.01.2015
03.01.2026 10:33:09
Тема создана с целью обсудить мой подход к созданию торговых роботов без написания сложных скриптов на Lua. -------------------------- Приведу примеры, потом объясню что и как. ---------------------- Пример 1: Скрипт скользящего стопа на основе индикатора fractals записывается таким образом:
Пример 2: Скрипт торгового робота на основе индикаторов moving и RSI
Робот строится как скрипт индикатора и состоит из обязательной таблицы Settings, встроенных в QUIK индикаторов и библиотеки nklibi, начальным объемом 11 КБ. ------------------- Все необходимые параметры робота определяются в Settings. Изначально указывается, если надо, создание лог файла, набор индикаторов, на основе которых будет работать робот. Для каждого индикатора указывается линия, параметр свечи и смещение относительно текущего значения. ------------------- Для применения робота необходимо открыть график инструмента, поместить на него указанные в Settings индикаторы и записать им в качестве идентификаторов указанные имена. ------------------- Алгоритмы новых роботов можно добавить либо в библиотеку, либо в конец скрипта. =================== Библиотека nklibi реализует конечный автомат управления заявками и стоп-заявками, а также определяет по графику инструмента все необходимые для торговли параметры. ==================== Если график с индикатором робота связать якорем с таблицей текущих параметров(ТТП) торгов, то, перебирая инструменты в ТТП, можно переключать робота на различные торгуемые инструменты. ---------------------------------- Конструктивные предложения по расширению возможностей данного подхода приветствуются. ---------------------------- В настоящее время библиотека nklibi находится в стадии тестирования.
Индикатор с метками
Пользователь
Сообщений: Регистрация: 30.01.2015
23.12.2025 08:36:00
По просьбе трудящихся, написал Пример ПРАВИЛЬНОГО индикатора, который через секунду удаляет ранее выставленную метку и выводит новую. -------------------------------- Пользуйтесь на здоровье. ----------------------------
Код
Settings={ Name = "*nk_test", tag ="Metka" }
params={R = 255, G = 255, B = 255, TRANSPARENCY = 0,FONT_HEIGHT = 10,TRANSPARENT_BACKGROUND = 1, FONT_FACE_NAME = 'Verdana' }
function OnCalculate(i) if i==1 then OnChangeSettings() elseif i== Size() then -- последняя свеча if os.time() > LastSecond then -- раз в секунду (или больше) LastSecond = os.time(); -- потиковое обновление (с задержкой) if Label then DelLabel(Settings.tag,Label) end params.TEXT =tostring(C(i)) params.HINT = tostring(i) local h=0 local m=i; while 14>i-m do if H(m) > h then h=H(m) end;m=m-1; end local Ti=T(i-3) params.DATE=100*(100*Ti.year+Ti.month)+Ti.day; params.TIME=100*(100*Ti.hour+Ti.min)+Ti.sec; params.YVALUE =h; Label= AddLabel(Settings.tag, params) end -- every second end end
function OnChangeSettings() sec = getDataSourceInfo().sec_code; DelAllLabels(Settings.tag); LastSecond=0 end function OnDestroy() OnChangeSettings() end function Init() return 1 end
Ускорение загрузки QUIK
Пользователь
Сообщений: Регистрация: 30.01.2015
20.12.2025 08:43:47
Добрый день, Предлагаю следующие решения : ------------------------------ 1) Сделать возможным указывать начальную дату загрузки сохраненной на ПК истории при старте QUIK. -------------------------- 2) Сделать возможным сохранять слепок памяти для последующей загрузки.
Время загрузки тарминала QUIK
Пользователь
Сообщений: Регистрация: 30.01.2015
02.12.2025 21:56:07
Я утверждаю, что время загрузки(старта) существенно зависит от наличия скриптов индикаторов на луа и архива истории. ------------------ Провел тест времени загрузки (начала работы )терминала QUIK (брокер Сбербанк) ---------------------
Тест 1: Сделал копию рабочей папки QUIK Удалил папку архивы и индикаторы и подключился к брокеру: ------------- Время старта вычислял по интервалу от момента двухфакторной до начала отображения стакана и графиков
Тест2: загрузил рабочий вариант QUIK
В результате Тест 1 время загрузки без скриптов на луа составляет 12 секунд, а время загрузки со скриптами индикаторов 69 секунд ---------------------------------- Выводы делайте сами.
Такого не может быть, но опять.
Пользователь
Сообщений: Регистрация: 30.01.2015
27.11.2025 08:20:32
Вопрос к разработчикам. -------------------- можете объяснить причину такого явления. ------------------ Терминал работает нормально. На рынке все спокойно. вдруг приходит сообщение: "Сервер принудительно разорвал соединение" ----------------------- после этого терминал пытается автоматически восстановить соединение. ---------------------- приходит сообщение: "Вы уже работаете в системе"
потом сообщение "соединение восстановить не удалось" --------------------- И так пока не надоест. ===================== Приходится закрывать принудительно терминал и запускать снова. ==================== Что за хрень происходит?
Структура моего робота
Пользователь
Сообщений: Регистрация: 30.01.2015
24.11.2025 11:18:21
В данной теме рассказываю, как построен мой торговый робот.
Полагаю, что тема будет интересна не только начинающим писателям роботов,
так как излагаю свои идеи, которые обладают определенной новизной .
----------------------------
Структура робота организована в виде сети роботов типа “звезда”.
Каждый робот – это отдельный поток или процесс ОС Windows.
------------------------------
В центре звезды находится главный робот nkbot, который обрабатывает события колбеков и таймера и раздает задания другим роботам.
События организованы в виде очереди. Если событий нет в течении заданного интервала, то робот выполняет задание по таймеру.
---------------------------
Все остальные роботы работают с заданной для каждого робота группой инструментов
и выполняют свои задачи по команде nkbot, либо по истечению заданного интервала времени.
---------------------------
Каждый робот работает в своем потоке OS Windows, либо в процессе терминала QUIK и имеет доступ к библиотеке QLua, либо в изолированном потоке(процессе) OS
Роботы в терминале запускаются как скрипты терминала QUIK.
----------------------------
Связь робота nkbot с каждым роботом сети выполняется через очередь команд в общей памяти.
Список работающих роботов создается в файле инициализации nkbot_ini.lua.
Задачи для роботов создаются в отдельных файлах в виде функций и загружаются в робота во время запуска скрипта по заданному списку задач для каждого робота.
--------------------------
Несколько примеров реализации отдельных элементов данной структуры.
Все роботы сети, кроме nkbot, имеют одинаковую структуру и реализуются одним скриптом.
Имя робота состоит из “nkbot_” и далее идентификатор робота.
Например, nkbot_B.lua, nkbot_C.lua два робота B и C
Файл nk_bot_B.lua
Код
Ts={SBER={int={1,2,5,10,20},q=true},
GAZP={int={2,5},q=true}, LKOH={int={5},q=true}}
--инструменты, которыми торгует робот , интервалы свечей и флаг
получения стакана
list_task="task1,task2" -- задачи, которые
быдет исполнять робот
nkLog=true; -- -включить вывод в лог файл
----------------не трогать---------------
minfo =
debug.getinfo(1, "S").source:sub(2); path =
minfo:match("(.*[/\\])") or "."
dofile(path.."stream.lua")
Файл nk_bot_С.lua
Код
Ts={SiZ5={int={5}}, RIZ5={int={5,10},},SRZ5={int={30}}}
--инструменты и интервалы свечей
list_task="task3,task4" -- список задач
nkLog=true;
----------------не трогать---------------
minfo =
debug.getinfo(1, "S").source:sub(2); path =
minfo:match("(.*[/\\])") or "."
dofile(path.."stream.lua")
--Пример ассинхронного запроса свечей--
--результат в массиве ds для каждого инструмента
Tclas={}
Tclas.QJSIM={SBER={int={1,2,5,10,20},ds={}}, GAZP={int={2,5},ds={}}, LKOH={int={5},ds={}}}
Tclas.SPBFUT={SiZ5={int={5},ds={}}, RIZ5={int={5,10},ds={}},SRZ5={int={30},ds={}}}
local tim=os.clock()
function OnParam(c,s)
local tc=Tclas[c]
if tc then local ts=tc[s];
if ts then local ts=tc[s]; local int=ts.int; local ds=ts.ds;local N=#int; local M=#ds;
if N~=M then local j=0;
while N>j do j=j+1;
if ds[j]==nil then local x=CreateDataSource(c,s, int[j]);
if x:Size()>0 then local ti=math.tointeger(1000*(os.clock()-tim)//1);
ds[j]=x; nkLog:write("time(ms)="..tostring(ti)..","..c..","..s..",j="..j..",int="..int[j]..",size="..ds[j]:Size().."\n"); nkLog:flush();
end
end
end
M=#ds; if N==M then nkLog:write(s..",подписка завершена\n"); nkLog:flush(); end
end
end
end
end
RUN = truefunction main()
while RUN do
-- Выполнение заданий скрипта --
sleep (10)
end
end
function OnStop(signal) RUN = false return 5000 end
function OnInit(p)
local p=string.reverse(p); local n,m=string.find(p,"\\");
local pS=string.reverse(string.sub(p,n)); name=string.reverse(string.sub(p,5,n-1)) ;
package.cpath =package.cpath..";"..pS.."?.dll;"
package.path =package.path..";"..pS.."?.lua;"..pS.."?.luac;"
fnlog=pS..name.."_nk.log"; nkLog=io.open(fnlog,"w")
end
В скрипте есть вывод в лог файл, который создается в каталоге скрипта. Результат работы скрипта на тестовом сервере QUIK
Пользователь может изменять данные лишь двумя способами -- выбирать из существующего списка или изменять значение на плюс минус 1 нажатием правой или левой кнопки мышки.. Двойным щелчком мышки можно сделать значение активным или пассивным. ----------------- Например, можно сделать неактивными все инструменты ,кроме одного, и робот будет торговать лишь этим инструментом. Можно сделать неактивным стоп или short и робот не будет ставить стоп или покупать в шорт. ------------------------- Еще в файле инициализации задается список имеющихся алгоритмов торговли , я называю их задачами.
Код
list_task="task1,task2,task3,task4" -- список существующих задач
а также список задач для всех инструментов по умолчанию
Код
user_task={"task1","task2" } --задачи для каждого инструмента по умолчанию
для каждого инструмента можно указать индивидуальные параметры или задачи, которые в списке инструментов указываются через ":" ========================= Это весь интерфейс.
Конструктивные предложения приветствуются.
Проблема с нажатием правой кнопки
Пользователь
Сообщений: Регистрация: 30.01.2015
17.11.2025 09:43:32
Когда в таблице в скрипте нажимаю правую кнопку мышки, то вылезает окно квика
можно как-то это отключить?
проблема с двойным нажатием кнопки
Пользователь
Сообщений: Регистрация: 30.01.2015
16.11.2025 10:40:06
При двойном нажатии кнопки мыши в таблице созданной в скрипте всегда сначала приходит сигнал об одинарном нажатии кнопки.
Как устранить сигнал об одинарном нажатии при двойном нажатии?
Метки на графике в скрипте
Пользователь
Сообщений: Регистрация: 30.01.2015
12.11.2025 16:00:39
По просьбе трудящихся.
Код
label_params={["TEXT"]="",["IMAGE_PATH"]="",["ALIGNMENT"]="Left",["YVALUE"]=0,["DATE"]=0,["TIME"]=0,["R"]=255,["G"]=255,["B"]=255,
["TRANSPARENCY"]=0,["TRANSPARENT_BACKGROUND"]=0,["FONT_FACE_NAME"]='Times New Roman',["FONT_HEIGHT"]=10,["HINT"]=""}
function labeldraw(price, YYYYMMDD,HMS,textlabel, texthint)
label_params.TEXT = textlabel;
label_params.DATE = YYYYMMDD;
label_params.TIME =HMS;
label_params.YVALUE = price;
label_params.HINT = texthint;
label_id_text = AddLabel(tag, label_params)
end
function main()
local i=0; local _size=0;
while true do
local size= getNumCandles(tag);
while size>i do
local t, _, _ = getCandlesByIndex(tag, 0, i, 1)
local z=t[0]; local Hi,Li,Ci,Oi=z.high,z.low,z.close,z.open;
i=i+1;
local Ti=ds:T(i)
YYYYMMDD=100*(100*Ti.year+Ti.month)+Ti.day;
HMS=100*(100*Ti.hour+Ti.min)+Ti.sec;
labeldraw(Hi, YYYYMMDD,HMS,"A", "B")
end
sleep(1)
end
end
function OnInit(p) -- инициализация функции main
sec,clas,int="SBER","QJSIM",5;
tag = "SBER_ID"
ds=CreateDataSource ("sec","QJSIM",5)
while ds==nil do ds,err=CreateDataSource(clas,sec,int); sleep(1); end
end
остановка скрипта при запуске QUIK
Пользователь
Сообщений: Регистрация: 30.01.2015
04.11.2025 06:15:38
Если Вам надо чтобы скрипт, запущенный в QUIK остановился при новом старте нужно первую строку в колбек OnInit записать так:
Код
function OnInit(path)
if os.clock()<2 then return end
индикатор на Lua
Пользователь
Сообщений: Регистрация: 30.01.2015
23.09.2025 18:58:02
Индикатор определяет максимум и минимум на заданном периоде и отображает среднее значение. Алгоритм оптимизировал для ускорения вычислений. Выкладываю для всех желающих:
Код
Settings = {Name = "*Kijun-sen",kijun_period = 6,}
function OnCalculate(i)
Hi=H(i) or H1; Li=L(i) or L1; x1=x;
if i1>i then
ma=Hi; mi=Li; jma=i; jmi=i;
end
if Hi and Li then
local j=i-Settings.kijun_period; if j<1 then j=1; end
if j>jma or j>jmi then
ma=Hi; mi=Li; jma=i; jmi=i;
while i>j do
Hi,Li=H(j),L(j)
if Hi and Hi>ma then ma = Hi jma=j; end
if Li and Li<mi then mi =Li jmi=j;end
j=j+1
end
else
if Hi>ma then ma=Hi; jma=i; end
if mi>Li then mi=Li; jmi=i; end
end
x=(ma + mi)/2; H1,L1,i1=Hi,Li,i;
end
return x1
end
function OnChangeSettings()
i1=0;jma=0; jmi=0; H1=0; L1=0;ma=0;mi=0;
end
function Init()
OnChangeSettings()
Settings.line = {{ Name=Settings.Name, Color=RGB(32,255,128), Type=TYPE_LINE, Width = 2,}}
return #Settings.line end
это мой вариант модификации этого алгоритма .
Код
Settings = {Name = "*ind_nk",period = 6,}
function OnCalculate(i)
Hi=H(i) or H1; Li=L(i) or L1;
Oi=O(i) or O1; Ci=C(i) or C1;
if i1>i then
ma=Hi; mi=Li; jma=i; jmi=i;
end
if Hi and Li and i1~=i then
local j=i-Settings.period; if j<1 then j=1; end
if j>jma or j>jmi then
ma=Hi; mi=Li; jma=i; jmi=i;
while i>j do
Hi,Li=H(j),L(j)
if Hi and Hi>ma then ma = Hi jma=j; end
if Li and Li<mi then mi =Li jmi=j;end
j=j+1
end
else
if Hi>ma then ma=Hi; jma=i; end
if mi>Li then mi=Li; jmi=i;
end
if x then
local z=2*Li-Hi; if Li>x and Ci>Oi then mi=z; jmi=i; end
z=2*Hi-Li; if x>Hi and Oi>Ci then ma=z; jma=i; end
end
end
if ma and mi then x=(ma + mi)/2; end
H1,L1,O1,C1,i1=Hi,Li,Oi,Ci,i;
end
return x
end
function OnChangeSettings()
i1=0;jma=0; jmi=0; H1=0; L1=0;ma=0;mi=0;
end
function Init()
OnChangeSettings()
Settings.line = {{ Name=Settings.Name, Color=RGB(255,255,255), Type=TYPE_LINE, Width = 2,}}
return #Settings.line end
Если нравится, можете сказать "спасибо".
Скрипты от ИИ
Пользователь
Сообщений: Регистрация: 30.01.2015
17.09.2025 14:54:10
Добрый день, Выкладываю работающий скрипт, который написал ИИ под моим руководством. ------------------------------- Скрипт позволяет загружать историю с биржи MOEX. ------------------------- Состоит из двух файлов. --------------- config.json
import os
import pandas as pd
import requests
import json
from datetime import datetime, timedelta
def load_config(config_file="config.json"):
"""
Загружает конфигурацию из JSON файла
Параметры:
config_file (str): Путь к файлу конфигурации
Возвращает:
dict: Конфигурация
"""
if not os.path.exists(config_file):
print(f"Файл конфигурации {config_file} не найден")
return None
try:
with open(config_file, 'r', encoding='utf-8') as f:
config = json.load(f)
return config
except Exception as e:
print(f"Ошибка при загрузке конфигурации: {e}")
return None
def fetch_moex_data(ticker, start_date, end_date, interval=60):
"""
Загружает данные с MOEX API для указанного тикера и временного периода
Параметры:
ticker (str): Тикер инструмента
start_date (datetime): Начальная дата загрузки
end_date (datetime): Конечная дата загрузки
interval (int): Интервал свечей в минутах (по умолчанию 60 минут)
Возвращает:
pd.DataFrame: DataFrame с загруженными данными
"""
data = []
current_start = start_date
print(f"Загрузка данных с MOEX для {ticker} (интервал: {interval} мин) с {start_date} по {end_date}...")
while True:
url = f"http://iss.moex.com/iss/engines/stock/markets/shares/boards/TQBR/securities/{ticker}/candles.json"
params = {
'from': current_start.strftime('%Y-%m-%d'),
'till': end_date.strftime('%Y-%m-%d'),
'interval': interval,
'start': 0
}
try:
response = requests.get(url, params=params)
response.raise_for_status() # Проверка на ошибки HTTP
json_data = response.json()
if 'candles' not in json_data or 'data' not in json_data['candles']:
print("Нет данных в ответе API")
break
candles = pd.DataFrame(json_data['candles']['data'],
columns=json_data['candles']['columns'])
if len(candles) == 0:
break
data.append(candles)
# Обновляем время для следующего запроса
current_start = (pd.to_datetime(candles['end'].iloc[-1]) + timedelta(minutes=interval))
if len(candles) < 500: # Если получено меньше 500 записей, это последняя порция
break
except requests.exceptions.RequestException as e:
print(f"Ошибка при запросе данных: {e}")
break
print(f"Загружено {len(data)} блоков данных")
if not data:
return pd.DataFrame()
# Объединяем все блоки данных и удаляем дубликаты
return pd.concat(data).drop_duplicates()
def process_raw_data(raw_data):
"""
Обрабатывает сырые данные: разделяет begin на date и time, удаляет end
Параметры:
raw_data (pd.DataFrame): Сырые данные с MOEX
Возвращает:
pd.DataFrame: Обработанные данные
"""
# Разделение begin на date и time
raw_data['date'] = pd.to_datetime(raw_data['begin']).dt.date
# Преобразование времени в формат без двоеточий (HHMMSS)
raw_data['time'] = pd.to_datetime(raw_data['begin']).dt.strftime('%H%M%S')
# Удаление ненужных столбцов
columns_to_keep = ['date', 'time', 'open', 'high', 'low', 'close', 'volume']
processed_data = raw_data[columns_to_keep].copy()
# Сортировка по дате и времени
processed_data = processed_data.sort_values(['date', 'time'])
return processed_data
def get_last_available_date(ticker_dir):
"""
Определяет последнюю дату, за которую есть данные в структуре каталогов
Параметры:
ticker_dir (str): Путь к каталогу тикера
Возвращает:
datetime.date: Последняя дата или None, если данных нет
"""
last_date = None
# Проходим по всем подкаталогам (годы)
if not os.path.exists(ticker_dir):
return None
for year_dir in os.listdir(ticker_dir):
year_path = os.path.join(ticker_dir, year_dir)
if not os.path.isdir(year_path):
continue
# Проходим по месяцам
for month_dir in os.listdir(year_path):
month_path = os.path.join(year_path, month_dir)
if not os.path.isdir(month_path):
continue
# Проходим по дням
for day_dir in os.listdir(month_path):
day_path = os.path.join(month_path, day_dir)
if not os.path.isdir(day_path):
continue
# Проверяем, что имя дня - это число (день месяца)
if not day_dir.isdigit():
continue
# Формируем дату: год, месяц, день
try:
year = int(year_dir)
month = int(month_dir)
day = int(day_dir)
current_date = datetime(year, month, day).date()
except:
continue
# Сравниваем с последней датой
if last_date is None or current_date > last_date:
last_date = current_date
return last_date
def update_data_file(ticker, interval, output_dir, start_date, end_date):
"""
Обновляет файлы данных, добавляя новые записи при необходимости
Параметры:
ticker (str): Тикер инструмента
interval (int): Интервал свечей в минутах
output_dir (str): Путь к каталогу для сохранения данных
start_date (datetime): Начальная дата для загрузки
end_date (datetime): Конечная дата для загрузки
Возвращает:
None
"""
# Определяем последнюю дату, за которую есть данные
last_date = get_last_available_date(output_dir)
if last_date is None:
# Данных нет, загружаем с начальной даты
new_start_date = start_date
print("Не найдено существующих данных. Загружаем все данные...")
else:
# Загружаем с последней даты (включительно)
new_start_date = datetime.combine(last_date, datetime.min.time())
print(f"Найдены данные до {last_date}. Загружаем новые данные начиная с этой даты...")
# Загружаем данные с MOEX
raw_data = fetch_moex_data(ticker, new_start_date, end_date, interval)
if raw_data.empty:
print("Нет новых данных для загрузки")
return
# Обрабатываем данные
processed_data = process_raw_data(raw_data)
# Сохраняем данные, разбивая по датам
save_data_by_date(ticker, interval, output_dir, processed_data)
print(f"Загружено и сохранено {len(processed_data)} записей")
def save_data_by_date(ticker, interval, output_dir, data):
"""
Сохраняет данные с разбиением по датам в подкаталоги
Параметры:
ticker (str): Тикер инструмента
interval (int): Интервал в минутах
output_dir (str): Базовый каталог для сохранения
data (pd.DataFrame): Данные для сохранения
Возвращает:
None
"""
# Получаем имя интервала
interval_name = get_interval_name(interval)
# Группируем данные по датам
grouped = data.groupby('date')
for date, day_data in grouped:
# Разбиваем дату на год, месяц, день
year = date.year
month = date.month
day = date.day
# Форматируем месяц и день с ведущими нулями
month_str = f"{month:02d}"
day_str = f"{day:02d}"
# Формируем путь к каталогу для этой даты
date_dir = os.path.join(output_dir, str(year), month_str, day_str)
# Создаем каталог, если он не существует
os.makedirs(date_dir, exist_ok=True)
# Создаем подкаталог для интервала
interval_dir = os.path.join(date_dir, interval_name)
os.makedirs(interval_dir, exist_ok=True)
# Удаляем столбец date перед сохранением
day_data_to_save = day_data.drop(columns=['date'])
# Сохраняем каждый параметр в отдельный файл
for column in day_data_to_save.columns:
file_path = os.path.join(interval_dir, column)
# Сохраняем столбец без индекса и без заголовка
day_data_to_save[column].to_csv(file_path, index=False, header=False)
print(f"Сохранено {len(day_data_to_save[column])} значений параметра '{column}' за {date} в {file_path}")
def get_interval_name(interval):
"""
Преобразует интервал в минутах в текстовое представление
Параметры:
interval (int): Интервал в минутах
Возвращает:
str: Текстовое представление интервала
"""
if interval == 1:
return "1min"
elif interval == 10:
return "10min"
elif interval == 60:
return "1hour"
elif interval == 1440:
return "1day"
else:
return f"{interval}min"
if __name__ == "__main__":
# Загрузка конфигурации
config = load_config()
if not config:
print("Не удалось загрузить конфигурацию. Выход...")
exit(1)
# Параметры загрузки
end_date = datetime.now()
print(f"Начало загрузки данных для тикеров: {', '.join([item['ticker'] for item in config['tickers']])}")
print(f"Период: с {min([datetime.strptime(item['start_date'], '%Y-%m-%d') for item in config['tickers']])} по {end_date.strftime('%Y-%m-%d')}")
print("-" * 50)
# Обработка каждого тикера и интервала
for ticker_config in config['tickers']:
ticker = ticker_config['ticker']
start_date_str = ticker_config['start_date']
intervals = ticker_config['intervals']
# Преобразование начальной даты из строки в datetime
try:
start_date = datetime.strptime(start_date_str, '%Y-%m-%d')
except ValueError:
print(f"Ошибка в формате даты для тикера {ticker}: {start_date_str}")
continue
print(f"\nОбработка тикера: {ticker} (начальная дата: {start_date_str})")
print(f"Интервалы: {', '.join([get_interval_name(i) for i in intervals])}")
# Формирование пути к каталогу для сохранения данных
base_dir = "moex"
ticker_dir = os.path.join(base_dir, ticker)
# Обработка каждого интервала для текущего тикера
for interval in intervals:
print(f"\nОбработка интервала: {get_interval_name(interval)}")
# Обновление файла данных
update_data_file(ticker, interval, ticker_dir, start_date, end_date)
# Проверка сохраненных данных
print("\nПроверка сохраненных данных:")
interval_name = get_interval_name(interval)
# Проверяем наличие подкаталогов с датами
if os.path.exists(ticker_dir):
# Собираем все даты из структуры каталогов
dates_found = []
# Проходим по годам
for year_dir in sorted(os.listdir(ticker_dir)):
year_path = os.path.join(ticker_dir, year_dir)
if not os.path.isdir(year_path):
continue
# Проходим по месяцам
for month_dir in sorted(os.listdir(year_path)):
month_path = os.path.join(year_path, month_dir)
if not os.path.isdir(month_path):
continue
# Проходим по дням
for day_dir in sorted(os.listdir(month_path)):
day_path = os.path.join(month_path, day_dir)
if not os.path.isdir(day_path):
continue
# Проверяем наличие подкаталога с интервалом
interval_path = os.path.join(day_path, interval_name)
if os.path.isdir(interval_path):
# Формируем дату для отображения
try:
year = int(year_dir)
month = int(month_dir)
day = int(day_dir)
date_str = f"{year:04d}-{month:02d}-{day:02d}"
dates_found.append(date_str)
except:
pass
if dates_found:
print(f"Найдено {len(dates_found)} дат с данными:")
# Показываем последние 5 дат
for date_str in sorted(dates_found)[-5:]:
# Формируем путь к каталогу интервала
interval_path = os.path.join(ticker_dir, date_str.split('-')[0],
date_str.split('-')[1], date_str.split('-')[2],
interval_name)
# if os.path.exists(interval_path):
# print(f" {date_str}:")
# Проверяем наличие файлов с параметрами
# for param in ['time', 'open', 'high', 'low', 'close', 'volume']:
# param_file = os.path.join(interval_path, param)
# if os.path.exists(param_file):
# # Читаем файл
# param_data = pd.read_csv(param_file, header=None)
# print(f" {param}: {len(param_data)} значений")
# Выводим последние 3 значения для проверки
# if len(param_data) > 0:
# print(f" Последние 3 значения: {', '.join(map(str, param_data.tail(3).values.flatten()))}")
# else:
# print(f" {date_str}: каталог интервала не найден")
else:
print("Данные не найдены")
else:
print(f"Каталог тикера {ticker} не найден")
print("-" * 30)
# print("-" * 50)
print("\nЗагрузка данных завершена для всех инструментов и интервалов")
QUIK на VPS
Пользователь
Сообщений: Регистрация: 30.01.2015
21.05.2025 16:59:53
Добрый день, Вопрос к разработчикам и знатокам. Хочу поставить торгового робота c возможностью ручной торговли на VPS. QUIK или что-то подобное вполне устроило бы. робота могу написать на любом языке, но предпочитаю С и Lua. ---------------------- Какое решение можете посоветовать. ------------------- Какие требования к железу. Если есть что-то в интернете просьба дать ссылку.
Что-то с сайтом не так.
Пользователь
Сообщений: Регистрация: 30.01.2015
17.04.2025 12:36:55
Сегодня при попытке зайти на Ваш сайт получаю это:
Данные с MOEX на Lua
Пользователь
Сообщений: Регистрация: 30.01.2015
21.03.2025 11:57:55
Для этого надо установить пакет luasocket. ------------------- Пример скрипта получить данные по Сбер для тайма 10 минут
Код
p2 = "D:/luasocket/";
package.cpath =package.cpath ..";"..p2.."?.dll";
package.path =package.path..";"..p2.."?.lua;"
local http = require("socket.http")
local header="http://iss.moex.com/iss/"
local sec="SBER"
local Date="2025-03-10"
local interval="10"
Data=http.request(header..eng..sec.."/candles.csv?from="..Date.."&interval="..interval) --не более 500 значений
print(Data)
Сбер перешел на версию 11.4.0.54. По их предложению установил. И сразу появились временами задержки при передвижении позиции мышкой. Т е двигаешь, ничего не меняется, но через секунд 20-60 позиция перемещается. Эффект проявляется не всегда, возможно при активном движении рынка. До этого работал на версии 8 как самой устойчивой и надежной. Возможно придется обратно откатиться. Что скажут разработчики. Что не так?
Имена параметров ТТП и других данных торгового сервера MOEX
Пользователь
Сообщений: Регистрация: 30.01.2015
19.02.2025 16:14:22
Вопрос разработчикам. Как указать "Время окончания срока действия заявки "?
Пользователь
Сообщений: Регистрация: 30.01.2015
20.01.2025 08:39:53
Нигде не нашел конкретное описание и установку данного параметра ------------------------ В документации по QUIK параметр указан на русском языке : Время истечения=124141 ------------------------ В документации по QLua аналогичный параметр есть лишь в описании таблицы заявок :
expiry_time
NUMBER
Время окончания срока действия заявки в формате <ЧЧММСС DESIGNTIMESP=19552>. Для GTT-заявок, используется вместе со сроком истечения заявки (Expiry)
---------------------- Просьба объяснить каким образом установить данный параметр на Lua и при ручном вводе заявки. Если это есть в документации, то дайте ссылку.
проблема с параметром "expiry_time "
Пользователь
Сообщений: Регистрация: 30.01.2015
19.01.2025 14:15:02
Добрый день, В формате таблицы "Заявки" есть параметр
expiry_time
NUMBER
Время окончания срока действия заявки в формате <ЧЧММСС DESIGNTIMESP=19552>.
Нигде не смог найти как этот параметр установить. На демо сервере его установка никак не проявляется. При ручном вводе заявки не увидел данного параметра. Кто может пояснить, возможность его применения.
Проблема с нажатием клавиш в таблице пользователя
Пользователь
Сообщений: Регистрация: 30.01.2015
14.01.2025 09:52:31
Добрый день, В своей таблице использую обработку нажатия правой и левой кнопки мыши. С левой все нормально, а справой такая проблема После обработки нажатия правой кнопки мыши в моем колбеке,терминал выдает предложение сортировки:
Как сделать так, чтобы это предложение не появлялось.
как найти единый счет
Пользователь
Сообщений: Регистрация: 30.01.2015
26.12.2024 15:43:17
Вопрос к разработчикам. Тестирую на демо поиск счата таким образом: ------------------------- str="trade_accounts"; for i=0,getNumberOf(str)-1 do x=getItem(str,i) if string.find(x.class_codes,c) then account=x.trdaccid; break; end end -------------------- Все находит замечательно. Но на реальном едином счете, счет не находится. Что не так?
Установка точного времени
Пользователь
Сообщений: Регистрация: 30.01.2015
23.12.2024 06:35:54
Добрый день, Иногда можно замечать, как время сервера брокера куда-то убегает относительно времени компьютера и времени сделок. ----------------------- В моей практике были случаи, когда сервер брокера отставал. Получалось прикольно, время сделок было в будущем. ---------------------- Особенно важным становится знание точного времени в момент открытия торгов, если Вы совершаете сделки на открытие. ------------------------- Как известно, все биржи сверяют свои часы по серверам точного времени. -------------------- Сделал это на своем компьютере. Проверить насколько точно синхронизирован ваш компьютер можно здесь:
у меня так:
Про синхронизацию ПК можно прочитать здесь:
Сервер лучше брать российский из этого пула:
Как исправить QUK без переустановки?
Пользователь
Сообщений: Регистрация: 30.01.2015
02.08.2024 20:20:35
Вопрос к разработчикам. Тестирую скрипт на версии 13. и 10. Ошибка одна и та же. При вылете скрипта возникает ошибка системы с сообщением.
Удаление файлов с расширением dat не исправляет ошибку. Исправить можно лишь переустановкой QUIK. ------------------- Переустанавливать каждый раз при такой ошибке очень муторно. ---------------------- Предполагаю, что QUIK записывает какую-то информацию в служебный файл и после этого выдает всегда эту ошибку. =============== Какой файл удалить чтобы QUIK нормально загрузился при возникновении данной ошибки. =============== Для особо непонятливых поясняю. Сообщение о нехватке памяти -является фейком, так как при переустановке с тем же или даже меньшим объемом свободной памяти QUIK нормально работает до очередного затыка в скрипте.
сообщение в Telegram - это просто
Пользователь
Сообщений: Регистрация: 30.01.2015
20.07.2024 12:53:33
Код
token="......" --робот в telegram
chat_id="....." --ваш в telegram
curl="C:/Windows/System32/curl.exe --silent --output null "..'"'.."https://api.telegram.org/bot"..token.."/sendMessage?chat_id="..chat_id.."&text="
------------------------------
mes="привет!!" --сообщение
os.execute(curl..mes..'"'); --отправляем
Циклические массивы - это просто
Пользователь
Сообщений: Регистрация: 30.01.2015
13.07.2024 12:14:28
Проблема версии 11.2.0.16
Пользователь
Сообщений: Регистрация: 30.01.2015
09.07.2024 06:57:21
Обнаружил такой эффект. делаем так:
1) Вызываем QUIK Junior, но не подключаемся к серверу. 2) Открываем любой индикатор в отдельном окне Например ADX 3) Удаляем второе окно В результате QUIK зависает, окно не удаляется. 4) Принудительно убиваем QUIK