4

我已经从 iqoption 获得了 OHLC 值,并试图找到一种将其与 MT5 一起使用的方法。
这是我获得值的方式:

import time
from iqoptionapi.stable_api import IQ_Option
I_want_money=IQ_Option("email","password")
goal="EURUSD"
print("get candles")
print(I_want_money.get_candles(goal,60,111,time.time()))

上面的代码库在这里:iqoptionapi

该行:I_want_money.get_candles(goal,60,111,time.time())输出json为:命令的输出

现在我在输出中得到 json,所以它像 API 一样工作,我猜是这样。

同时,我尝试在 MT5 中创建一个自定义符号为iqoption. 现在我只想将来自 API 的 OHLC 的数据添加到它,以便它会继续从 Iqoption 获取数据并在图表窗口中显示自定义交易品种的图表iqoption

但我无法将其加载到自定义符号中。请帮助我。

已编辑

这是来自 iqoption 的实时流数据的代码:

from iqoptionapi.stable_api import IQ_Option
import logging
import time
logging.basicConfig(level=logging.DEBUG,format='%(asctime)s %(message)s')
I_want_money=IQ_Option("email","password")
I_want_money.start_candles_stream("EURUSD")
thread=I_want_money.collect_realtime_candles_thread_start("EURUSD",100)
I_want_money.start_candles_stream("USDTRY")
thread2=I_want_money.collect_realtime_candles_thread_start("USDTRY",100)
time.sleep(3)
#Do some thing
ans=I_want_money.thread_collect_realtime.items()
for k, v in ans:
    print (k, v)
I_want_money.collect_realtime_candles_thread_stop(thread)
I_want_money.stop_candles_stream("EURUSD")
I_want_money.collect_realtime_candles_thread_stop(thread2)
I_want_money.stop_candles_stream("USDTRY")
4

2 回答 2

4

好的,您需要
1. 接收来自代理的提要(希望您成功)
2. 将其写入文件
**(两者 - python)**
3. 读取并解析它
4. 将其添加到历史中心/ marketWatch
**(两者 - mt5)**

  1. 因此,在 I_want_money.get_candles(goal,60,111,time.time()) 此字符串可能是 json 或 json-array 之后,您会以字符串形式接收数据。

  2. 重要的问题当然是你要放置数据的路径。MQL45 专家只能访问两个文件夹(如果不应用 dll):C:\Users\MY_NAME_IS_DANIEL_KNIAZ\AppData\Roaming\MetaQuotes\Terminal\MY_TERMINAL_ID_IN_HEX_FORMAT\MQL4\Files 和 C:\Users\MY_NAME_IS_DANIEL_KNIAZ\AppData\Roaming\MetaQuotes\ Terminal\Common\Files 在后一种情况下,您需要使用 const int handle=FileOpen( , |*| FILECOMMON);打开文件

为了解析 json,您可以使用 jason.mqh https://www.mql5.com/en/code/13663库(其他库很少)但据我记得它有一个错误:它无法解析数组正确的对象。为了克服这个问题,我建议将每个刻度写在单独的行上。最后,您将随机从您的 python 应用程序接收数据,并将其写入 Common 或 direct 文件夹。MT5 机器人将读取并删除它。为了避免混淆,最好保证文件具有唯一的名称。随机 (random.randint(1,1000)) 或日期时间的毫秒数都可以提供帮助。

到目前为止,你有 python 代码:

 receivedString = I_want_money.get_candles(goal,60,111,time.time())  
 filePath = 'C:\Users\MY_NAME_IS_DANIEL_KNIAZ\AppData\Roaming\MetaQuotes\Terminal\MY_TERMINAL_ID_IN_HEX_FORMAT\MQL4\Files\iqoptionfeed'  
 fileName = os.path.join(filePath,"_"+goal+"_"+str(datetime.now())+".txt")   
 file = open(fileName, "w")  
 for string_ in receivedString:  
     file.write(string_)  
 file.close()  

如果您创建了一个线程,则每次从线程收到答案时,您都会编写这样一个文件。

接下来,您需要 MT5 中的数据。最简单的方法是遍历现有文件,确保您可以阅读它们并阅读(或者如果不能,则放弃)并在阅读后删除,然后继续接收到的数据。最简单快捷的方法当然是使用 0MQ,但让我们不用 dll 来做。为了读取文件,您需要设置一个可以尽可能快地工作的计时器,然后放手。由于您不能让 Windows 应用程序的睡眠时间少于 15.6 毫秒,因此您的计时器应该睡眠这个时间。

string path;
int OnInit()
  {
   EventSetMillisecondTimer(16);    
   path="iqoptionfeed\\*";
  }
void OnDeinit(const int reason) {   EventKillTimer();   }

string _fileName;
long _search_handle;
void OnTimer()
  {
   _search_handle=FileFindFirst(path,_fileName);
   if(_search_handle!=INVALID_HANDLE)
      {
       do
         {
          ResetLastError();
          FileIsExist(_fileName);
          if(GetLastError()!=ERR_FILE_IS_DIRECTORY)
             processFile(path+_fileName);
         }
       while(FileFindNext(_search_handle,_fileName));
       FileFindClose(_search_handle);
      }
  }  

这段代码循环文件夹并处理它设法找到的每个文件。现在读取文件(两个函数)并处理其中的消息:

void processFile(const string fileName)
  {
   string message;
   if(ReadFile(fileName,message))
      processMessage(message,fileName);
  }
bool ReadFile(const string fileName,string &result,const bool common=false)
  {
   const int handle = FileOpen(fileName,common?(FILE_COMMON|FILE_READ):FILE_READ);
   if(handle==INVALID_HANDLE)
     {
      printf("%i - failed to find file %s (probably doesnt exist!). error=%d",__LINE__,fileName,GetLastError());
      return(false);
     }
   Read(handle,result);
   FileClose(handle);
   if(!FileDelete(fileName,common?FILE_COMMON:0))
      printf("%i - failed to delete file %s/%d. error=%d",__LINE__,fileName,common,GetLastError());
   return(true);
  }
void Read(const int handle,string &message)
  {
   string text="";
   while(!FileIsEnding(handle) && !IsStopped())
     {
      text=StringConcatenate(text,FileReadString(handle),"\n");
     }
   //printf("%i %s - %s.",__LINE__,__FUNCTION__,text);
   message=text;
  }

最后但并非最不重要的一点:处理获得的文件。正如上面所建议的,它为每个新的刻度有一个 json 格式的刻度,用 \r\n 分隔。
我们的目标是将其添加到符号中。为了解析 json,jason.mqh 是一个可用的解决方案,但您当然可以手动解析它。

void processMessage(const string message,const string fileName)
  {
   string symbolName=getSymbolFromFileName(fileName);
   if(!SymbolSelect(symbolName,true))
     {
      if(!CustomSymbolCreate(symbolName))
         return;
     }
   string lines[];
   int size=StringSplit(message,(ushort)'\n',lines);
   for(int i=0;i<size;i++)
     {
      if(StringLen(lines[i])==0)
        continue;
      CJAVal jLine(jtUNDEF,NULL);
      jLine.Deserialize(lines[i]);
      MqlTick mql;
      //here I assume that you receive a json file like " { "time":2147483647,"bid":1.16896,"ask":1.16906,"some_other_data":"someOtherDataThatYouMayAlsoUse" } "
      mql.time=(datetime)jLine["time"].ToInt();
      mql.bid=(double)jLine["bid"].ToDbl();
      mql.ask=(double)jLine["ask"].ToDbl();
      ResetLastError();
      if(CustomTicksAdd(symbolName,mql)<0)
          printf("%i %s - failed to upload tick: %s %s %.5f %.5f. error=%d",__LINE__,__FILE__,symbolName,TimeToString(mql.time),mql.bid,mql.ask,GetLastError()); 
     }
  }
string getSymbolFromFileName(const string fileName)
  {
   string elements[];
   int size=StringSplit(fileName,(ushort)'_',elements);
   if(size<2)
      return NULL;
   return elements[1];
  }

不要忘记添加调试信息和请求GetLastError()是由于某种原因你得到错误。

这可以在回测器中工作吗?当然不是。拳头,OnTimer()在 MQL 测试器中不支持。接下来,您需要一些历史记录才能使其运行。如果您没有任何历史记录 - 除非经纪人可以提供给您,否则没有人可以帮助您;最好的想法可能是立即开始收集和存储它,当项目准备好(可能再过几个月)时,您将准备好它并能够使用可用的数据集测试和优化策略。您可以将收集到的集合应用到测试器中(与 MQL4 相比,MQL5 确实是算法交易开发的下一步),可以手动或使用诸如 tickDataSuite 及其 Csv2Fxt.ex4 文件之类的工具,该文件生成测试器可以读取和处理的 HST 二进制文件;无论如何,这是另一个问题,没有人可以告诉您您的经纪人是否将他们的数据存储在某个地方以提供给您。

于 2018-07-05T21:10:33.927 回答
1

在第二次阅读您编写(和编辑)的内容后,我可以看到您想要:

  • 与 iqoption 同步的符号[通过您的代理/远程]
  • 该符号可用于回测
  • 该符号可用于屏幕实时/策略/指标运行

这意味着 MT 平台不允许以自动化方式在策略/指标之外进行操作 - 您可以通过提供数据包、将其解析为 CSV 并导入自定义符号创建器来手动实现。有据可查here

不幸的是,您选择的平台设计代表独立的策略和指标,更适合初学者而不是专业人士认真对待它。

请参阅我提供的链接并亲自查看。官方文档声明您可以通过 mql ref创建自定义符号,但即使他们在前言中声明它允许 3rd 方提供者 - 它没有在其他任何地方引用,也没有显示任何集成可能性。


自定义指标

自定义符号属性

于 2018-07-04T07:49:01.323 回答