使用 extern 从 Haxe 访问 C++ class
Access C++ class from Haxe using extern
我想从 Haxe 程序访问 C++ 代码。我正在尝试使用 extern class 但遇到了一些问题。我正在尝试使用 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
您需要将 @:include
和 @:extern
元数据添加到您的 extern class 以便 hxcpp 可以指向正确的文件并在编译时解析。
@:include
元标记允许您指向一个文件,该文件位于您的 extern class 定义之上。该文件需要相对于您的构建输出目录。
@:extern
元标记告诉 hxcpp 将 class 定义解析为您希望在 C++ 中看到的 class 的名称。因此,如果您要创建一个指针,class 将使用 new
关键字实例化,并且外部定义如下:
@: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 class 一起使用,因为它不是常规的 Haxe class。相反,您需要创建一个静态函数来执行实例化,它将 return extern class 键入 Haxe 上下文,然后允许您访问其成员变量。
如果您在包含供 hxcpp 查找的文件时遇到问题,您可以使用 XML 文件,您可以使用该文件告诉 hxcpp 要查找的文件夹。
在 Windows 上,您可以使用 <files id="haxe">
标签,后跟内容 <compilerflag value="-I/../includes/" />
。当然,关闭此标签并确保包含文件夹位于项目的根目录中。
这将生成一个额外的编译器标志,告诉 Haxe 编译器在一个额外的文件夹中查找要包含的 C++ 源文件。然后,您可以删除 @:include
中的 include/
前缀,就在您的 extern class 定义之前。
然后告诉 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
关键字,在这种情况下 returning 对象应该是指向该类型的指针。所以在 C++ 中 new Rectangle()
将 return 变成 Rectangle*
。如果你在 Haxe 中使用 Rectangle.create()
,并将它赋给一个变量,这就是你有效得到的。
您还可以在 C++ 中使用 Rectangle rect;
实例化,或者说声明。这是像结构一样访问的,所以不是使用 ->
来访问成员,而是使用句号 '.'
要使其在 Haxe 中运行,您需要在外部添加 @:structAccess
元数据。此外,您需要将 @:native
从 @:native("Rectangle*")
更改为 @:native("Rectangle&")
。这确保初始变量将始终用作参考。
&
表示将变量作为引用,不像指针是引用
https://haxe.io/roundups/wwx/c++-magic/
// Avoids generating dynamic accessors.
@:unreflective
// Marks an extern class as using struct access(".") not pointer("->").
@:structAccess
@:include("string")
@:native("std::string")
extern class StdString {
@:native("new std::string")
public static function create(inString:String):cpp:Pointer<StdString>;
public function size():Int;
public function find(str:String):Int;
}
class Main {
public static function main() {
var std = StdString.create("my std::string");
trace( std.value.size() );
std.destroy();
}
}
我想从 Haxe 程序访问 C++ 代码。我正在尝试使用 extern class 但遇到了一些问题。我正在尝试使用 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
您需要将 @:include
和 @:extern
元数据添加到您的 extern class 以便 hxcpp 可以指向正确的文件并在编译时解析。
@:include
元标记允许您指向一个文件,该文件位于您的 extern class 定义之上。该文件需要相对于您的构建输出目录。
@:extern
元标记告诉 hxcpp 将 class 定义解析为您希望在 C++ 中看到的 class 的名称。因此,如果您要创建一个指针,class 将使用 new
关键字实例化,并且外部定义如下:
@: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 class 一起使用,因为它不是常规的 Haxe class。相反,您需要创建一个静态函数来执行实例化,它将 return extern class 键入 Haxe 上下文,然后允许您访问其成员变量。
如果您在包含供 hxcpp 查找的文件时遇到问题,您可以使用 XML 文件,您可以使用该文件告诉 hxcpp 要查找的文件夹。
在 Windows 上,您可以使用 <files id="haxe">
标签,后跟内容 <compilerflag value="-I/../includes/" />
。当然,关闭此标签并确保包含文件夹位于项目的根目录中。
这将生成一个额外的编译器标志,告诉 Haxe 编译器在一个额外的文件夹中查找要包含的 C++ 源文件。然后,您可以删除 @:include
中的 include/
前缀,就在您的 extern class 定义之前。
然后告诉 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
关键字,在这种情况下 returning 对象应该是指向该类型的指针。所以在 C++ 中 new Rectangle()
将 return 变成 Rectangle*
。如果你在 Haxe 中使用 Rectangle.create()
,并将它赋给一个变量,这就是你有效得到的。
您还可以在 C++ 中使用 Rectangle rect;
实例化,或者说声明。这是像结构一样访问的,所以不是使用 ->
来访问成员,而是使用句号 '.'
要使其在 Haxe 中运行,您需要在外部添加 @:structAccess
元数据。此外,您需要将 @:native
从 @:native("Rectangle*")
更改为 @:native("Rectangle&")
。这确保初始变量将始终用作参考。
&
表示将变量作为引用,不像指针是引用
https://haxe.io/roundups/wwx/c++-magic/
// Avoids generating dynamic accessors.
@:unreflective
// Marks an extern class as using struct access(".") not pointer("->").
@:structAccess
@:include("string")
@:native("std::string")
extern class StdString {
@:native("new std::string")
public static function create(inString:String):cpp:Pointer<StdString>;
public function size():Int;
public function find(str:String):Int;
}
class Main {
public static function main() {
var std = StdString.create("my std::string");
trace( std.value.size() );
std.destroy();
}
}