使用extern从Haxe访问C ++类

时间:2016-02-25 07:23:36

标签: c++ haxe

我想从Haxe程序访问C ++代码。我正在尝试使用外部课程但面临一些问题。我正在尝试使用haxe -main Cpp_Extern -cpp Cpp_Extern进行编译。

我的Haxe代码:

@:include("./Rectangle.cpp")
@:extern("Rectangle*")
extern class Rectangle
{
    @:native("set_values") public function set_values(w : Int, h : Int) : Void;
    @:native("new Rectangle") public static function create() : Rectangle;
}

class Cpp_Extern
{
    public static function main()
    {
        Rectangle.set_values(10,20);
    }
}

C ++代码

#include <iostream>

class Rectangle
{
  public:
  void set_values (int x, int y) 
    {
        std::cout << "x = " << x << "\n";
        std::cout << "y = " << y << "\n";
    }
};

int main()
{
    Rectangle one;
    return 0;
}

错误是

C:\Users\ila5\Desktop\CPP\Cpp_Extern>haxe -main Cpp_Extern -cpp Cpp_Extern
haxelib run hxcpp Build.xml haxe -Dhaxe3="1" -Dhaxe_ver="3.201" -Dhxcpp_api_level="321" -I"C:\\HaxeToolkit\\haxe\\extraLibs/" -I"" -I"C:\\HaxeToolkit\\haxe\\std/cpp/_std/" -I"C:\\HaxeToolkit\\haxe\\std/"
cl.exe -Iinclude -nologo -O2 /WX- /fp:precise -DHX_WINDOWS -D_USING_V140_SDK71_ -GR -FS -Oy- -c -EHs -GS- -arch:SSE -IC:/HaxeToolkit/haxe/lib/hxcpp/3,2,205/include -DHXCPP_VISIT_ALLOCS -DHXCPP_API_LEVEL=321 -D_CRT_SECURE_NO_DEPRECATE -D_ALLOW_MSC_VER_MISMATCH -D_ALLOW_ITERATOR_DEBUG_LEVEL_MISMATCH -wd4996 -MT ./src/Cpp_Extern.cpp -FoC:/Users/ila5/Desktop/CPP/Cpp_Extern/Cpp_Extern/obj/msvc19xp/2c1b12bd_Cpp_Extern.obj
Cpp_Extern.cpp
./src/Cpp_Extern.cpp(30): error C2440: 'initializing': cannot convert from 'Rectangle *' to 'Rectangle'
./src/Cpp_Extern.cpp(30): note: No constructor could take the source type, or constructor overload resolution was ambiguous
./src/Cpp_Extern.cpp(34): error C2819: type 'Rectangle' does not have an overloaded member 'operator ->'
C:/Users/ila5/Desktop/CPP/Cpp_Extern/Rectangle.cpp(4): note: see declaration of 'Rectangle'
./src/Cpp_Extern.cpp(34): note: did you intend to use '.' instead?
./src/Cpp_Extern.cpp(34): error C2232: '->Rectangle::set_values': left operand has 'class' type, use '.'

Error: Build failed

2 个答案:

答案 0 :(得分:4)

您需要将@:include@:extern元数据添加到您的extern类,以便hxcpp可以指向正确的文件并在编译时解析。

@:include元标记允许您指向一个文件,该文件位于您的extern类定义之上。该文件需要相对于您的构建输出目录。

@:extern元标记告诉hxcpp将类定义解析为您希望在C ++中看到的类的名称。因此,如果您正在创建指针,则将使用new关键字对该类进行实例化,并且extern定义如下:

@:include("includes/Rectangle.cpp")
@:extern("Rectangle*")
extern class Rectangle
{
    @:native("set_values") public function set_values(w : Int, h : Int) : Void;
    @:native("new Rectangle") public static function create() : Rectangle;
}

您不能将new函数定义与extern类一起使用,因为它不是常规的Haxe类。相反,你需要创建一个静态函数来执行instantion,它会将extern类类型返回到Haxe上下文中,然后允许你访问它的成员变量。

如果您遇到包括要查找hxcpp的文件的问题,您可以使用XML文件,您可以使用该文件告诉hxcpp要查找的文件夹。

在Windows上,您可以使用<files id="haxe">标记,后跟内容<compilerflag value="-I/../includes/" />。当然,关闭此标记并确保包含文件夹位于项目的根目录中。

这将生成一个额外的编译器标志,告诉Haxe编译器查找要包含的C ++源文件的附加文件夹。然后,您可以在extern类定义之前删除include/中的@:include前缀。

然后告诉hxcpp使用XML文件,您需要在主入口点上方使用@:buildXml元标记,该元标记应包含以下内容:<include name="${haxelib:myCustomRepo}/../Build.xml" />同样,这需要相对于你项目的根源。

在上面的XML示例中,我使用${haxelib:myCustomRepo}告诉Haxe查找给定haxelib的路径。您可以使用以下命令设置haxelib存储库开发目录:haxelib dev myCustomRepo ./可以在这种情况下帮助您。这完全取决于您是否使用该路线。

您不必使用XML文件,但建议大型项目使用,尤其是在跨平台和目标构建时。

我还注意到你在C ++代码中缺少一个构造函数,所以不要忘记把它放进去。另外,你的主入口点代码现在应该是这样的:

@:buildXml('<include name="${haxelib:myCustomRepo}/../Build.xml" />') //only if you are using XML
class Main {
    public static function main() {
        var rect = Rectangle.create();
        rect.set_values(10, 20);
    }
}

有两种方法可以在C ++中实例化对象。您可以使用new关键字,在这种情况下,返回的对象应该是指向该类型的指针。因此new Rectangle()将在C ++中返回Rectangle*。如果您在Haxe中使用Rectangle.create(),并将其分配给变量,那就是您实际获得的。

您还可以在C ++中使用Rectangle rect;进行实例化,或者更确切地说声明。这可以像结构一样访问,因此不使用->来访问成员,而是使用句号'。'

为了在Haxe中工作,您需要在extern上添加@:structAccess元数据。此外,您需要将@:native@:native("Rectangle*")更改为@:native("Rectangle&")。这确保了初始变量将始终用作参考。

&表示使用变量作为引用,与指针不同, 是引用。

答案 1 :(得分:0)

https://haxe.io/roundups/wwx/c++-magic/

> toJSON(container, pretty = TRUE)
[
  {
    "data": {
      "name": "foo",
      "moogs": [
        {
          "x": 1,
          "y": 2
        },
        {
          "x": 2,
          "y": 1
        }
      ]
    }
  }
]