13

我正在用 C# 编写一个 GUI 工具来解析和显示另一个用 C 编写的程序的数据输出。为了解析数据,我需要知道在许多 C 头文件中指定的数据结构。因此,我需要将这些 C 头文件合并到我的 C# 项目中。我的问题是:

1)经过一些研究,我得出结论,最好的方法是在我的解决方案中创建一个新的 C++/CLI 项目,将 C 头文件导入这个新项目,编写一些 C++/CLI 类作为数据的瘦包装器C 头文件中定义的结构,然后从 C# 代码引用 C++/CLI 包装类。这是最好的方法,还是有更好的方法?

2)我遇到了一个参考问题。这是我的简化代码来说明问题:

C++/CLI 项目中的原始 C 头文件

#define ABC  0x12345

C++/CLI 项目中的包装类

#include "my_c_header.h"

namespace C_Wrappers {
    public ref class MyWrapper {
        public:
            property unsigned int C_ABC {
                unsigned int get() { return ABC; }
            }
    }
 }

C# 项目中的用户类

using C_Wrappers;
using System;

namespace DataViewer {
    public partial class MainForm : Form {
        private  MyWrapper myWrapper = new MyWrapper();
        public MainForm() {
            Console.WriteLine(myWrapper.C_ABC.ToString());
        }
    }
 }

在 C# 项目中,我添加了对 C++/CLI 项目的引用(使用右键单击 > 添加引用)。在构建期间,我在 C# 项目中遇到错误:“找不到类型或命名空间 'C_Wrappers'(您是否缺少 using 指令或程序集引用?)。”

我以为我做了我应该做的一切。我应该怎么做才能修复这个错误?

4

2 回答 2

24

在我自己的解决方案中,我有 4 个项目:

  1. C++ 项目和测试代码
  2. 仅使用动态链接从第一个项目源中编译 DLL的C++ DLL 项目
  3. 包装器项目,它只是一个使用 C++/CLI 的适配器,它包装了原始 C ++
  4. C# WPF 项目,这是我的图形界面。

这是我对上面提供的链接的翻译。

C++ DLL 项目

将您的 C++ 代码转换为 DLL 库。

  1. 在 Visual Studio 中,转到File > New > Project ,从 Visual C++ 选项卡中选择Win32 项目。
  2. 为项目和解决方案选择一个名称,解决方案将包含所有项目。
  3. 在 Win32 Application 的助手中,单击next,选中DLL框,然后Empty project然后单击Finish

项目创建 工程助理

要添加的代码

这是我的 dll 的 C++ 头文件(减去很多东西)。

令牌.h

#pragma once
#define DLLEXP   __declspec( dllexport )

DLLEXP void pop_back(std::string& str);

DLLEXP std::string testReturnString();

DLLEXP int getRandomNumber();

CPP内部没有什么可以改变的。

构建项目,您应该有一个 DLL 和一个 LIB 文件以包含在 C# 项目调试目录中。

C++/CLI 包装器

该项目用作上一个项目的本机代码与 GUI 的托管代码之间的接口。

  1. 向解决方案添加一个新项目(Visual C++ 中的类库)(在本示例中称为“Wrapper”)
  2. 使用附加的 Include 目录添加到本机项目的路径
  3. 添加本机项目作为新项目的引用(右键单击 > References... > Add New Link
  4. Properties > Linker > Input中,将本机 dll 的名称放在延迟加载 DLL(本例中为 Computations.dll)字段中

C++/CLI 代码

我的包装器只是一个看起来像这样的类(减去我自己的代码)。

包装器.h

#include "Token.h" // include your C++ header

#include <string>
#include <iostream>

namespace Wrapper {

    // noticed the ref?
    public ref class TokenAnalyzer
    {

    public:
        TokenAnalyzer(){
        };

        void Init();
            // use the C++/CLI type, like String^ (handles)
        System::String^ ProcessLine(int lineNbr, System::String^ line);
    };
}

CPP 中没有什么特别之处,除了必须包含#include "stdafx.h".

它还应该构建到一个 DLL 中,您将把它包含在 C# 调试目录中。

只是我在 SO 某处找到的一个有用功能,但不记得您可能需要在哪里。它将 C++/CLI 字符串句柄转换为 C++ 标准字符串。

std::string MarshalString (String ^ s) {
        using namespace Runtime::InteropServices;
        const char* chars = 
            (const char*)(Marshal::StringToHGlobalAnsi(s)).ToPointer();
        std::string os = chars;
        Marshal::FreeHGlobal(IntPtr((void*)chars));
        return os;
    }

C# 项目

向解决方案添加一个新项目(C# Windows 窗体WPF或任何您想要的!)并将其设置为启动项目(右键单击 > 设置为启动项目)。

  1. 添加 C++/CLI 项目作为新项目的参考
  2. using Wrapper;以源代码形式添加指令

像这样使用它:

/// Store the C++/CLI Wrapper object.</summary>
private Wrapper.TokenAnalyzer mTokenAnalyzer = new TokenAnalyzer();

更新 2016/05/06

Ashwin花时间制作了一个示例项目和一篇博文教程,这可能会有所帮助。

于 2013-10-03T16:11:35.017 回答
6

对我来说,这个类似问题的答案解决了 C# 中的相同错误。

(即我的 C++\CLI 项目中有一个包装头文件,但也忘记包含一个 .cpp 文件(即使它只有两行:#include "stdafx.h" 和 #include "Wrapper.h "))

于 2013-11-17T12:43:01.440 回答