首先,我是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;
}
答案 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;
}
此代码非常有效。