将C结构包装到Ruby

时间:2012-04-19 20:26:27

标签: c ruby

如何初始化C Struct并将其作为Ruby类包装为另一个Ruby对象的参数?我正在重写内存,但不知道,如何修复它。

Ruby代码,我想创建Person类的实例并将Address变量添加到它们中,这是另一个类:

require_relative 'my_extension'

class Address
  def inspect
    "Replaced #inspect: <Address: town:#{town}>"
  end
end

class Person
    def initialize
        puts "init"
    end

    def print()
        puts "Addr class #{@addr.inspect}"
    end
end

foo1=Person.new
foo1.add_address("London")
foo1.print
foo2=Person.new
foo2.add_address("Paris")
foo1.print
foo2.print
foo1.print

C代码,扩展Ruby:

#include <stdio.h>
#include "ruby.h"

struct Address {
    char * town;
};

static VALUE get_addr(VALUE self) {
    return rb_iv_get(self,"@addr");
}

static VALUE wrap_address_get_town(VALUE self) {

    struct Address * address;
    Data_Get_Struct(self, struct Address, address);
    return rb_str_new2(address->town);
}


VALUE foo_class;
VALUE address_wrapper_class;


void free_m(){
    printf("free\n");//just for test
}

void add_address_t(VALUE self,VALUE new_town){
    printf("add_address\n");
    /*init new struct and add value to it*/
    struct Address addr;
    addr.town=StringValuePtr(new_town);

    /*wrap struct*/
    VALUE wrapped_address=Data_Wrap_Struct(address_wrapper_class, 0, free_m,&addr);

    /*set it as instance variable*/
    rb_iv_set(self,"@addr",wrapped_address);
}

static VALUE foo_class_alloc(VALUE self){
    return self;
}


void Init_my_extension(){
    foo_class = rb_define_class("Person", rb_cObject);

    address_wrapper_class = rb_define_class("Address", rb_cObject);

    rb_define_method(address_wrapper_class, "town", wrap_address_get_town, 0);

    rb_define_method(foo_class, "add_address", add_address_t, 1);

}

输出产生意外结果:

init
Addr class Replaced #inspect: <Address: town:London>
init
Addr class Replaced #inspect: <Address: town:Paris> //London expected
Addr class Replaced #inspect: <Address: town:�)> //another problem
Addr class Replaced #inspect: <Address: town:�)>
run
run
free
free

2 个答案:

答案 0 :(得分:1)

FFI gem可以提供更好的方法来完成您想要做的事情:

require 'ffi'  
module AddressModule
   extend FFI::Library
   ffi_lib '<path to your c library>'

   class Address < FFI::Struct
      layout :address, :string,
   end
end

person = AddressModule::Address.new
person[:address] = "an address"

或者其他什么。查看https://github.com/ffi/ffi/wiki

上的FFI文档

答案 1 :(得分:0)

所有C例程都应定义为静态返回VALUE。这应该是Qnil,如果不应该返回任何相关的。使用add_address_t例程还有另一个问题 - 你要包装一个本地定义的结构 - 它应该用ALLOC(struct Address);分配