P / Invoke in ASP.NET(从dll读取/写入文本文件)

时间:2012-04-25 09:45:57

标签: c# c++ asp.net pinvoke

我有一个C++ Win32程序,我正在编写和阅读文本文件。这个C ++程序生成一个DLL,我在ASP.NET Web应用程序中引用了这个dll 使用P / Invoke,我正在调用从这个dll读取和写入文件的方法。

当我在WPF应用程序中使用P / invoke测试时,dll工作正常。 引用dll位于此WPF应用程序的bin/Debug文件夹中,调用时dll中的write方法在同一文件夹中生成文本文件。 此外,from the same folder,我可以使用dll的read方法来读取文本文件。

然而,当我从我的ASP.NET Web应用程序调用Dll方法时,genearted文件转到其他目录(很可能是因为dll被加载到其他地方执行)并且我无法找到这个生成的位置文件去(没有任何错误)

与桌面应用程序类似,是否有一些方法可以将文件夹写入bin文件夹本身,以便我可以从bin文件夹本身读取?

示例代码:
.cpp文件

extern "C" D_API int Write1()
{
    ofstream myfile;
    myfile.open ("example.txt");
    myfile << "Writing this to a file.\n";
    myfile.close();
    return 1;
}

extern "C" D_API char* Read1()
{
    ifstream myReadFile;
    myReadFile.open("test.txt");
    char output[100];
    if (myReadFile.is_open())
    {
        while (!myReadFile.eof()) 
        {
        myReadFile >> output;       
        }
    }
    return output;
}

C#.aspx.cs

[DllImport("Testing1.dll", EntryPoint = "fnTest", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
        public static extern int Write1();


        [DllImport("Testing1.dll", EntryPoint = "ReadTest", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
        public static extern StringBuilder Read1();

1 个答案:

答案 0 :(得分:2)

由于您使用的是相对路径,因此该文件将相对于您调用本机代码时进程的工作目录。你发现这是一个相当脆弱的安排。

我会通过向本机代码添加额外的字符串参数来解决问题,该参数指定要使用的文件的完整路径。您可以从托管代码中轻松生成此内容。我确信。

原生代码

extern "C" D_API int WriteTest(char *filename)
{
    ....
    myfile.open(filename);
    ....
}

托管代码

[DllImport("Testing1.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int WriteTest();

要做的另一点是您的数据读取功能不正确。它尝试返回堆栈分配的缓冲区。您需要在托管代码中分配缓冲区,然后将其传递给本机代码。也许是这样的:

extern "C" D_API int ReadTest(char *filename, char* buffer, int len)
{
     //read no more than len characters from filename into buffer
}

在管理方面:

[DllImport("Testing1.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern int ReadTest(string filename, StringBuilder buffer, int len);
....
StringBuilder buffer = new StringBuilder(100);
int retval = ReadTest(FullySpecifiedFileName, buffer, buffer.Capacity);