Alexey написал: Прошу помощи у знающих: Кто-нибудь может сказать в чем ошибка: Код условный, суть в том, что когда в качестве второго оператора цикла используется Size индиктор пропадает из вкладки добавления индикатора терминала Quik. Я так понимаю идет какая-то ошибка...
Size = Size()
function Q() dVal = {}; for i = 1, Size, 1 do dVal = 100*i; end return dVal end
dVal = Q()
без полного кода сказать трудно. Но у Вас куча дыр в этом условном коде, которые могут привести к сюрпризам при выходе из этой функции. -------------------- бесплатный совет. ------------------ Не надо писать условных кодов, если проблема реальная. Т к суть проблемы в том, что вы реально, а не условно, не понимаете как писать индикаторы. И это видно из вашего условного кода, -----------------
Можно и так через скрипт получить код инструмента с графика, но лучше, конечно, чтобы была функция встроенная в quik, без всяких редактирований таблицы tool как в скрипте:
Код
is_run = true
chart_tag = "tag"
local tool = {
[ "Сбербанк [Price]" ] = "SBER" ,
[ "ГАЗПРОМ ао [Price]" ] = "GAZP" ,
[ "ЛУКОЙЛ [Price]" ] = "LKOH" ,
[ "Роснефть [Price]" ] = "ROSN"
}
function candle ()
allcandles = getNumCandles (chart_tag)
tab, count, leg = getCandlesByIndex (chart_tag, 0 , 0 , allcandles)
for key, val in pairs(tool) do
if key = = leg then
codeSec = val
end
end
return codeSec
end
function main ()
while is_run do
candle()
message (tostring(codeSec), 1 )
sleep ( 1000 )
end
end
Вы что-то путаете или невнимательно читали документацию qLua. Если вы хотите получить данные по инструментам в скрипте (как в вашем примере), а не в индикаторе, то для этого уже есть функции ----------------------- CreateDataSource и O, H, L, C, V, T - Функции в качестве параметра принимают индекс свечи и возвращают соответствующее значение. ----------------------------- и tag вообще не нужен для этого.
nikolz написал: если это индикатор в окне на графике, то можно прочитать значения через идентификатор
Каким образом через идентификатор можно получить значения scd и cld в примере выше? Например, можно через следующий код получить имя линии индикатора: is_run = true
chart_tag = "code" numcandles = 3
-- получение всех свечей function candle() allcandles = getNumCandles(chart_tag) -- диапазон свечей, по которым идёт поиск tab, count, leg = getCandlesByIndex(chart_tag, 0, 0, allcandles) message(tostring(leg), 1) end
function main() while is_run do candle() sleep(1000) end end
Просьба к разработчикам: добавить в LUA функции для определения вкладок в QUIK Это нужно для того, чтобы одного и того же робота можно было запускать для разных инструментах, в разных вкладках. Тогда в названии вкладки можно было бы писать код инструмента.
Еще вторая просьба, дать возможность при создании окна указать название вкладки. Потому что-то если ты после запуска робота, переключился на другую вкладку, а робот при этом пересоздал окно, то оно выводится на текущую вкладку, а это не очень удобно.
Заранее спасибо!
Я решаю эту проблему так: Идентификатор вкладки пишите по имени инструмента В индикаторе делает tag по имени инструмента которое читаете в индикаторе(роботе) ------------- В итоге робот-индикатор будет автоматически настраивать на тот инструмент, в окно которого вы установите этот робот-индикатор
Спасибо большое. Теперь мне понятно, что такое сбор за адресную сделку и клиринговая комиссия, жаль что московская биржа так же ловко как вы не могут этого объяснить. Остался небольшой вопрос по сбору за регистрацию сделки. Ответьте пожалуйста ответ цифрой сколько с меня снимут сбор за регистрацию сделки при покупки 10 лотов и продаже 10 лотов на примере индекса РТС. 1) (купил 10 лотов РТС и продал эти 10 лотов через 5 мин (внутри дня)) = ? 2) (купил 10 лотов РТС и продал эти 10 лотов через 5 дней) = ?
Добавлю к предыдущему ответу. Чтобы ответить на Ваш вопрос, вам надо взять регламент брокера и там есть приложение - тарифы В этом приложении находите свой тариф и считаете. Как ранее сказал если торгуете внутри дня то тариф обычно меньше, все расчеты делаются в клиринг по сальдо всех сделок брокером. ---------------- Обращаю ваше внимание на тот факт, что на самом деле торгуете не вы, а брокер. Вы лишь даете поручение ему, а он их либо выполняет либо нет, но за ваши деньги.
nikolz написал: надо собирать 5.3 и пример я Вам дал для 5.3 и ссылку на документацию на 5.3
Собсвенно для 5.3.5 - я тоже собирал - ошибка точно такая же.
Я вам все написал как делать. Вы собираетесь делать или рассуждать? если у вас 8.7 то все должно быть 5.3 ------------------- анекдот: Мадам, там кирпич и обрыв,туда нельзя. Но Вам можно.
гадать не видя весь код бессмысленно. Сообщение говорит, что функции нет. еще раз скажу, отлаживайте свою функцию без квик. Научитесь сначала писать dll для луа. Полагаю Вы знаете как отладить софт.
аналог - это любое решение, которое может выполнять те же функции. Open Office Calc - аналог Excel в функции создания таблиц. ---------------- В плане экспорта по DDE Open Office вообще не причем.
Вы где в документации прочитали о возможности эксопрта в OpenOffice? Причем же здесь поддержка КВИК? Они что на все придуманные Вами вопросы должны отвечать?
nikolz написал: у вас ошибка в ---------------------- extern "C" LUALIB_API int luaopen_lua_dll_x64_name(lua_State * L) { lua_newtable(L); luaL_setfuncs(L, ls_lib, 0); lua_pushvalue(L, -1); lua_setglobal(L, "lua_dll_x64_name"); return 0; }
-- для LUA5.3 надо так:
//=== Регистрация реализованных в dll функций, чтобы они стали "видимы" для Lua ================================// luaL_Reg ls_lib[] = { {"TestFunc", forLua_TestFunc}, {NULL, NULL} };
//=== Регистрация названия библиотеки, видимого в скрипте Lua ==================================================// extern "C" LUALIB_API int luaopen_lua_dll_x64_name(lua_State * L) { luaL_newlib(L,ls_lib); lua_setglobal(L, "lua_dll_x64_name"); return 1; }
Вы напишите, как я написал или разбирайтесь как правильно раскрыть макрос -------------- (luaL_newlibtable(L,l), luaL_setfuncs(L,l,0)) и нужно ли удалять значение из стека lua_pushvalue(L, -1); -------------------- у меня один оператор, а умник наплодил три.
Andrey написал: Добрый день уважаемые форумчане. Помогите пжлста разобраться какая сумма списывается со счета при совершении сделки на срочном рынке московской бирже. Звонил брокеру, сотрудники горячей линии говорят, что согласно моему тарифу комиссия брокера составляет 1р при покупке или при продаже 1 лота на фьючерсном контракте. Т.е. купил затем продал заплатил 1+1=2р. Больше никакой информации по комиссиям они дать не могут. Позвонил в московскую биржу, чтобы узнать какую комиссию снимают они. Мне сказали писать в тех поддержку. Написал в тех поддержку. Тех поддержка дала ссылку, мол читайте и разбирайтесь сами, на сайте все написано. Прошу помочь разобраться на примере фьючерса РТС (RIZ2). На сайте имеется следующая инфа:
Сбор за регистрацию сделки*, руб.
9,27
Сбор за адресную сделку*, руб.
3,09
Клиринговая комиссия за исполнение контракта*, руб.
3,09
Со сбором за регистрацию сделки всё понятно. Что такое сбор за адресную сделку и клиринговая комиссия за исполнения контракта? Хотелось бы понимать, если я купил 10 лотов на фьючерсном контракте РТС (RIZ2) и через 3 минуты продал их. Какова будет комиссия? Если я купил 10 лотов на фьючерсном контракте РТС и продал через 3 дня, то какая будет комиссия?
для Вас актуален лишь Сбор за регистрацию сделки Кроме того сбор на сделки внутри дня ниже чем на сделки результат которых переходит через клиринг. Так как внутри дня до клиринга сделки лишь фиксируются но расчета по ним нет. Это как игра в преф - внутри дня -пишем пульку а расчет в конце игры. ---------------- Сбор за адресную сделку - это когда в заявке вы укажете, что продаете Васе Пупкину, а тот укажет что покупает у Вас --------------- Клиринговая комиссия за исполнение контракта- это когда контракт закончился т е происходит экспирация, а у Вас открытый этот контракт на счете.
Создает новую таблицу с размером, оптимизированным для хранения всех записей в массиве l(но фактически не сохраняет их). Оно предназначено для использования в сочетании с luaL_setfuncs(см. luaL_newlib).
Он реализован в виде макроса. Массив lдолжен быть фактическим массивом, а не указателем на него.
я использую 5.3 так как это работает для всех версий КВИК а 5.4 лишь начиная с 9.X но пока там много ошибок и там можно включить 5.3 для 5.4 не помню, возможно и так как у вас.
у вас ошибка в ---------------------- extern "C" LUALIB_API int luaopen_lua_dll_x64_name(lua_State * L) { lua_newtable(L); luaL_setfuncs(L, ls_lib, 0); lua_pushvalue(L, -1); lua_setglobal(L, "lua_dll_x64_name"); return 0; }
-- для LUA5.3 надо так:
//=== Регистрация реализованных в dll функций, чтобы они стали "видимы" для Lua ================================// luaL_Reg ls_lib[] = { {"TestFunc", forLua_TestFunc}, {NULL, NULL} };
//=== Регистрация названия библиотеки, видимого в скрипте Lua ==================================================// extern "C" LUALIB_API int luaopen_lua_dll_x64_name(lua_State * L) { luaL_newlib(L,ls_lib); lua_setglobal(L, "lua_dll_x64_name"); return 1; }
Serg_ написал: Подскажите, есть ли какие ни будь условно официальные примеры, как связать С++ с Quik ? Как сделать dll ?
Все есть безусловно официально. В КВИК встроена виртуальная машина луа без каких либо изменений исходников поэтому читайте API C для lua. ------------------------- Я так пишу функции для QUIK на СИ от колбек до многопоточной обработки данных в LUA. ---------------------- QLUA - это библиотека написанная на API C. -------------------- Или в чем то другом проблема?
Не могу понять, как через dll - все это делать. Откроено говоря - это неМного геморрой. Продукту 20 лет, а нормального API почему то нет.
Вообще-то есть даже учебники о том как использовать API C ------------ как писать dll поясняю ------------------- берете любой компилятор СИ --------------- Я например последнее время для dll луа использую Pelles C Очень хороший пакет именно для СИ -------------- иногда использую MVC. ================ Далее в любимом IDE выбираете создать DLL и пишите программу на CB Функции СИ пишите как обычно для СИ А для вызова их из луа делаете обертку -------------- типа такой: -------------------- static int stop(lua_State *L) { //здесь вставляете преобразование данных луа в данные СИ QueryPerformanceCounter(&count1);
lua_pushinteger(L,(count1.QuadPart-count.QuadPart)); //это преобразование целого в формат луа и запись в стек return 1; -- это число возвращаемых параметров } Все функции API C здесь http://antirek.github.io/luabook/api.htm если любите C++ то читайте это: https://habr.com/ru/post/237503/ -------
Alexander написал: Я не рассчитывал на то, что колбэки могут отправляться по нескольку раз. Прочитал эту тему, понял, что могут и причём неограниченное число раз. Я не заморачивался и не тестил чем конкретно отличаются пришедшие в колбэках таблицы, но вижу, что OnOrder() где order.balance == 0 - 2 раза, и OnTrade() 3 раза и слава богу цена в нём одна и та же.
Вопросы: 1) Возможен ли вызов OnTransReply() на мою поданную заявку с моим TRANS_ID, с trans_reply.trans_id == 0, nil, "0", "" или же trans_reply.trans_id всегда содержит номер заявки? Этот номер заявки измениться не может? 2) OnTransReply() вызывается только один раз? 3) Возможен ли случай, когда сначала первой будет вызов OnOrder(), а потом уже вызов OnTransReply()? Вроде бы как-то один раз такое проскочило, когда скрипт вылетел с ошибкой исполнения. Но точно не помню, возможно было и так, даже скорее всего так, что был вызов только OnTrade(), а OnTransReply() не вызвался, хотя по факту заявка выставилась. Из-за этого дублирую сохранение номера заявки не только в OnTransReply(), но и в OnOrder(). 4) Могу ли я быть уверен, что имея самый первый OnOrder() c order.balance == 0, последующие OnOrder() не изменят значение order.balance? 5) Могут ли приходить сначала OnTrade() без trade.price, а потом с trade.price? От этого зависит какие именно я могу использовать вызовы OnTrade(), т.е. нужно ли вообще проверять установлено ли в них поле trade.price, или это поле заполнено ценой всегда и не меняется в следующих вызовах? 6) Могу ли я быть уверен, что имея самый первый OnTrade() с ценой исполнения trade.price, последующие OnOrder() не изменят значение trade.price? 7) Могу ли я с учётом своего алгоритма контроля исполнения использовать только самые первые вызовы OnOrder(), OnTrade(), OnTransReply(), отфильтровывая все последующие, где я фактически смотрю только на изменение баланса в заявке и жду появления сделки с номером заявки? 8) Может быть вообще как-то по другому изменить алгоритм контроля исполнения заявки(приход сделки)? Какой алгоритм контроля посоветуют разработчики с учётом всех этих многочисленных вызовов?
Все эти 0, nil, "0", "", которые могут прийти в trans_id в разных частных случаях в OnTrade() или в OnOrder() как я понял для меня не критичны, так как во всех колбэках я проверяю поля таблиц trans_id со своим номером транзакции TRANS_ID
1) OnTransReply - это колбек об отправке транзакции. Т е транзакция ушла с сервера брокера и попала на сервер биржи. единственный параметр в ней который не изменится это trans_id. ------------------------ Его надо использовать для идентификации транзакций. Все остальное может бить или не быть в зависимости от того как обрабатывается эта транзакция на сервере биржи. --------------------- 2) Надо делать программу так, чтобы число срабатывания колбеков не влияло на результат. Для этого надо не считать число срабатываний, а реагировать на события о которых сообщается в информации принятой колбеком. Если новая информация не изменяет состояния Вашего робота (конечного автомата) то какая разница, сколько раз сработает колбек. состояние не изменится. ------------------ 3) OnOrder может приходить много-много раз. Например Вы поставили лимитную заявку 100 лотов , а встречные заявки содержат по 1 лоту. Сколько раз сработает OnOrder? Правильно... много. Поэтому читайте ответ 2) ------------------- 4) см 2) 5)см 2) 6) см 2) 7) нет 8) ДА ------------------
Serg_ написал: Подскажите, есть ли какие ни будь условно официальные примеры, как связать С++ с Quik ? Как сделать dll ?
Все есть безусловно официально. В КВИК встроена виртуальная машина луа без каких либо изменений исходников поэтому читайте API C для lua. ------------------------- Я так пишу функции для QUIK на СИ от колбек до многопоточной обработки данных в LUA. ---------------------- QLUA - это библиотека написанная на API C. -------------------- Или в чем то другом проблема?
Добрый день, Предлагаю добавить в библиотеку QLUA функцию установки идентификатора для графика цена и индикатора . --------------------- Объясняю зачем: -------------------- 1) Для установки идентификатора не надо будет лазить в меню окна. Причем очень достает устанавливать этот идентификатор для индикатора при отладке индикатора. ---------------------------- 2) идентификатор графика цены нужен для вывода меток на график. Если это график с якорем, то отображаемый инструмент будет изменяться при изменении активной строки в таблице, а идентификатор при этом не меняется. С помощью предлагаемой функции можно в скрипте динамически изменять идентификатор, например, указывая в нем имя инструмента. В итоге можно метки на графике разделить по инструментам. ------------------------------ 3) идентификатор индикатора нужен для чтения параметров индикатора в скрипте луа. Сейчас для этого надо лазить в меню окна и прописывать идентификатор в Settings. В итоге идентификатор для индикатора можно будет динамически привязывать к инструменту, интервалу и алгоритму индикатора.
Добрый день, --------------------------- Пример простого, универсального советника на луа на основе любых индикаторов. ------------------------------ Советник - это программа, которая формирует сигналы "купить/продать" и показывает их, но не совершает сделки. --------------------- Пример на основе стратегии пересечения двух скользящих средних. Помещаем два индикатора на график цены инструмента Присваиваем им идентификаторы MOV1 и MOV2 как на рисунке:
далее пишем индикатор "nk_bot"
Код
Settings={ i1="MOV2",i2="MOV1", Name = 'nk_bot', }
------------------
local function gI(s,j,i)
local t=getCandlesByIndex(s,j,i-1,1); if t then return t[0].close; else return 0 end
end --значение индикатора
-------------------------
local function cross(s1,j1,s2,j2,i)
local m=i; local x= gI(s1,j1,m) local x1=gI(s2,j2,m);
while m>1 and x~=0 and x1~=0 and x>x1 do m=m-1; x=gI(s1,j1,m); x1=gI(s2,j2,m) end
if m>1 and i>m then return m end
end --пересечение
--------------------------
function OnCalculate(i)
local Bu,Se; local i1=i-1;
if i>1 then
local jU_,jD_=jU,jD;
jU=cross(Settings.i1,0,Settings.i2,0,i1); --MOV1 пересекает MOV2 снизу вверх
jD=cross(Settings.i2,0,Settings.i1,0,i1); --MOV1 пересекает MOV2 сверху вниз
if jU and jU_==nil then jBu=i; Bu=L(i)-0.1; end
if jD and jD_==nil then jSe=i; Se=H(i)+0.1; end
if i==Size() then -- последняя свеча
end
end
return Se,Bu;
end
-----
function Init()
local t={};
t[#t+1]={Name = "Se",Color = RGB(255,0,0),Type = 11,Width =3}; ---sell
t[#t+1]={Name = "Bu",Color = RGB(0,255,0),Type = 10,Width =3}; ---buy
Settings.line=t;
return #t;
end
и помещаем его на график инструмента В результате получим на истории торгов сигналы "купить/ продать" на графике инструмента --------------------- На основе данного примера Вы можете построить советник для любых индикаторов, которые встроены в терминал КВИК или написаны кем-то.