4

我正在尝试为 Ruby 编译一个 c++ 扩展,编译没有返回错误,但似乎编译不正确。我究竟做错了什么?

我有主要的 cpp 脚本foo.cpp

#include <iostream>
#include <ruby.h>
extern "C"

VALUE cFoo;
void Init_foo(){cFoo = rb_define_class("Foo", rb_cObject);}

extconf.rb

require "mkmf"
$libs += " -lstdc++ "
create_makefile("foo")

在这些文件的目录中,我做了

$ ruby extconf.rb
creating Makefile
$ make
compiling foo.cpp
linking shared-object foo.so
$ ls
Makefile extconf.rb foo.cpp foo.o foo.so

然后,我有一个 Ruby 脚本test.rb

#!/usr/bin/env ruby
require "path_to_this_directory/foo"

我跑test.rb。它返回一个错误:

... in `require': .../foo.so: undefined symbol: cFoo - .../foo.so (LoadError)

我究竟做错了什么?

环境

  • 操作系统:Ubuntu Linux 11.10
  • 红宝石:1.9.3
4

1 回答 1

1

您声明cFoo存在:

extern "C"

VALUE cFoo;

但你从来没有定义它。extern "C" blahblah只是说存在blahblah,它是外部可见的,并且它的名称不会被破坏(即“C”链接),但extern声明实际上并没有定义任何东西;extern说某物存在,但它并没有带来它的存在。

你的 C++ 应该看起来更像这样:

#include <iostream>
#include <ruby.h>

extern "C" VALUE cFoo;
extern "C" void Init_foo();

VALUE cFoo;
void Init_foo(){cFoo = rb_define_class("Foo", rb_cObject);}

您需要两个extern "C"来告诉 C++ 编译器不要管名称,然后单独定义符号。你也可以这样说:

#include <iostream>
#include <ruby.h>

extern "C" VALUE cFoo;

VALUE cFoo;
extern "C" void Init_foo(){cFoo = rb_define_class("Foo", rb_cObject);}

我不确定第二个版本是标准 C++ 还是 GCC 扩展。

我不是一个 C++ 人,所以我希望我没有把这些术语弄得太糟糕。

于 2012-04-16T00:21:10.023 回答