C ++访问冲突在mql4中写入dll中的0x00000000

时间:2015-06-09 06:05:14

标签: c++ dll mql4

首先,我是C ++的新手(将近一周),如果这很明显,请原谅我。此外,我已经搜索了许多类似问题的帖子。要么我的理解不够发达,要么没有相关信息来帮助我理解这个问题。

在Metatrader 4中,我试图弄清楚如何将结构变量传递给dll,并修改存储在所述结构中的变量。到目前为止,即使在处理结构数组时,我也取得了很大的成功。然后我遇到了一个问题。

我已将问题缩小到使用字符串。如果你愿意的话,请看看下面的代码,我已经过去专注于解决这个问题,并帮助我理解为什么我一直得到这个'访问冲突写入0x00000000'每当我尝试在mt4中运行脚本时出错。

mql4代码:

struct Naming
{
  string word;
} name;

#import  "SampleDLLtest.dll"
bool     NameTest(Naming &name);
#import

int init() { return(0); }

int start()
{
   Print("original name: ", name.word);
   if( NameTest( name ) )
   {
     Print("new name: ", name.word);
   }

   //---
   return(0);
}

这是相关的dll代码:

#define WIN32_LEAN_AND_MEAN
#include "stdafx.h"
#include <windows.h>
#include <stdlib.h>
#include <stdio.h>
#include <iostream>
#include <string>

BOOL APIENTRY DllMain(HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved)
{
   //---
   switch (ul_reason_for_call)
   {
      case DLL_PROCESS_ATTACH:
      case DLL_THREAD_ATTACH:
      case DLL_THREAD_DETACH:
      case DLL_PROCESS_DETACH:
      break;
   }

   //---
   return(TRUE);
}

struct Naming
{
   std::string n_name;
};

bool __stdcall NameTest(Naming *name)
{
   name->n_name = "Captain Success";

   return true;
}

2 个答案:

答案 0 :(得分:2)

来自mql4的文档:http://docs.mql4.com/basis/preprosessor/import

  

以下内容不能用于导入功能中的参数:

     
      
  • 指针(*);
  •   
  • 链接到包含动态数组和/或指针的对象。
  •   
     

包含字符串和/或的类,字符串数组或复杂对象   任何类型的动态数组都不能作为参数传递给   从DLL导入的函数。

导入的函数采用指针,mql4显然不支持。

你应该使用固定大小的字符数组来传递数据到dll:

像:

struct Naming {
  char m_name[255];
}

该函数需要接受对此结构的引用(但可能不支持)或直接接受结构并返回结构。

Naming NameTest(Naming name) {

  strncpy(name.m_name, "New Content", sizeof(name.m_name) -1);
  if (sizeof(name.m_name) > 0) {
      name.m_name[sizeof(name)-1] = 0;
  }
  return name;
}

调用它将如下所示:

name = NameTest(name);

答案 1 :(得分:1)

我知道这有点奇怪,但我回答了我自己的问题,因为我弄清楚发生了什么......至少大部分都是。

所以,这是交易。从技术上讲,您可以传递包含字符串的结构。你不能做的是编辑字符串。结构中没有字符串自动转换为char []。因此,当dll尝试编辑字符串时,它会抛出访问冲突,因为字符串实际上不是C ++中的字符串,而是伪装成字符串的字符数组。

那就是说,我确实解决了如何传递包含字符串的结构,并修改了dll中的值。我就是这样做的。

---从mql4代码开始--- 首先,我使用char []而不是字符串声明结构。

struct Naming
{
  char word[65];
} name;

然后我用空值初始化char [],检查它,传递结构,并检查值是否设置正确。

ArrayInitialize(name.word, '\0');
Print("original name: ", CharArrayToString(name.word));
if( NameTest( name ) )
{
   Print("new name: ", CharArrayToString(name.word));
}

---现在转到C ++代码--- 我声明了相同的结构。

struct Naming
{
    char n_name[65];
};

然后是功能。我首先必须在临时char []中捕获字符串文字。我循环了一个for循环,将元素分配给struct中的char []。问题是,struct中的char []不是const,而char temp []是。我通过将每个char捕获到char变量,然后将该变量值存储在struct char []中来解决这个问题。

bool __stdcall NameTest(Naming *name)
{
    char temp[] = "Captain Success";

    for (int i = 0; temp[i] != '\0'; i++)
    {
        char t = temp[i];
        name->n_name[i] = t;
    }

    return true;
}

此代码非常有效。