Ruby FFI 入门教程
FFI是一个可以让用户使用Ruby调用C代码的gem。如果你需要执行一些系统底层调用,或者做一些高性能运算的话,FFI是一个很不错的选择。
1. 安装
执行gem install ffi即可。非常标准的安装过程,期间会做一些本地编译。
2. 超简易入门
现在我们已经可以调用C代码了。我们可以自己写一些C代码来调用,但是更简单的办法是直接调用C标准库里的东西。
我们试试看调用puts函数。随便创建一个文件hello_world.rb:
require 'ffi'
module Hello
extend FFI::Library # 获得FFI::Library相关函数
ffi_lib FFI::Library::LIBC # 加载libc
attach_function :puts, [ :string ], :int # 将puts方法挂到当前Hello模块上,指明入参和返回值的类型
end
Hello.puts("Hello, World") # 现在可以调用puts这个模块方法了
运行得到输出结果:
Hello, World
Cool,调用C函数成功。
3. 调用自己的C函数
能调用C标准库是远远不够的。我们的目标是能够调用自己的C代码。
首先,我们写一个测试用的C源文件,比如叫做c_base.c:
#include <string.h>
int add(int a, int b){
return a+b;
}
对,就是一个简单到不能再简单的C函数,做了个加法。
我们先将它编译:
gcc -c -fPIC c_base.c -o c_base.o
gcc c_base.o -shared -o c_base.so
执行即可得到c_base.so文件。
当然,为了方便起见,建议把这些放在一个Makefile里面,直接用make编译。
比如写成这样:
c_base.so: c_base.o
gcc c_base.o -shared -o c_base.so
c_base.o: c_base.c
gcc -c -fPIC c_base.c -o c_base.o
.PHONY: clean
clean:
rm *.o *.so
执行后一样可以得到c_base.so文件。
然后我们就可以在Ruby里面调用了:
require 'ffi'
module Hello
extend FFI::Library
ffi_lib File.join(File.dirname(__FILE__) + '/c_base.so') # 加载动态库,只要提供路径就行
attach_function :add, [:int,:int ], :int
end
res = Hello.add(10, 7)
p res
运行得到输出结果:
17
好的!我们成功调用了自己的C函数!
4. 有关指针
到目前为止看起来不错。唯一需要注意的是attach_function里面参数类型的指定。
我们用:int来指定int,用:string来指定char *类型(需要以\0结尾)
完整的类型映射关系可以参考: https://github.com/ffi/ffi/wiki/Types
之前我们在Hello World的例子中指定了:string来输出字符串,传给C代码的时候变成了char *类型。
但是对于这种指针操作实际上是有隐患的。考虑以下代码:
static char* my_name;
void bad_set_my_name(char* name) {
my_name = name;
}
在这个代码里我们可以给my_name设置值。
我们假设使用以下Ruby代码调用:
module Bar
extend FFI::Library
ffi_lib File.join(File.dirname(__FILE__) + '/set_name.so') # 加载动态库
attach_function :bad_set_my_name, [ :string ], :void
end
module Foo
name = "Robin"
Bar.bad_set_my_name(name) # 调用C代码
end
调用后我们可以认为现在C代码中的my_name应该指向Robin这个C字符串。
现在问题来了,如果Ruby的GC把name变量回收掉了,那么C里面的my_name指针将成为一个无效指针。另一方面由于这个指针是C在管理,因此Ruby也没有办法越俎代庖地去回收C里面的变量。因此这就成为了一个隐患。
还有一种更复杂的情况是,如果是在使用JRuby的情况下,由于JVM对于新生代使用的是复制GC算法,因此会使得堆里面的对象地址发生变化,这就使得即使没有回收,C里面的指针也可能会变成无效的。
解决方法也很简单,只需要让传给C的指针所指向的数据位置不要变就可以了。做法也很简单,只要把数据复制到native memory里面,然后把native memory里的地址传给C代码就可以了。
修改后的代码如下:
module Bar
extend FFI::Library
ffi_lib File.join(File.dirname(__FILE__) + '/set_name.so')
attach_function :bad_set_my_name, [ :pointer ], :void # 注意类型改为了pointer
end
module Foo
name = FFI::MemoryPointer.from_string("Robin") # 将Robin字符串拷贝到native memory上面去
Bar.bad_set_my_name(name)
end
由于FFI::MemoryPointer实际上代表了一个内存地址,相当于void *,因此在attach_function的时候需要修改入参类型为:pointer。
反正传给C代码的指针在运行时都是纯粹的地址,char *和void *并没有区别,所以C代码不需要做修改。
Ruby FFI 入门教程的更多相关文章
- 《Ruby语言入门教程v1.0》学习笔记-01
<Ruby语言入门教程v1.0> 编著:张开川 邮箱:kaichuan_zhang@126.com 想要学习ruby是因为公司的自动化测试使用到了ruby语言,但是公司关于ruby只给了一 ...
- 《Ruby语言入门教程v1.0》学习笔记-03
10.09 第七章 7.1 模块 Ruby标准包里的 Math 模块提供了许多方法,比如:求平方根 sqrt ,使用的时候要这么写:模块名.方法名(参数).如:Math.sqrt( a*5+b ) M ...
- 《Ruby语言入门教程v1.0》学习笔记-02
9.18 第四章 一切都是对象 这个章节的例子都举得很浅显易懂,而且作者的语言= =噗,委实生动有趣啊是~~ 4.1 两种思维方式 初期的编程思想是:以“如何做”为指导来编写代码.这时期的编程语言叫 ...
- Ruby入门教程和技巧
转自:http://blog.csdn.net/cqfz123/article/details/1349050 Ruby真的比Java更好? Ruby On Rails 创始人:对Java 说再见 ...
- 转:几十种编程语言的快速入门教程- learnxinyminutes.com
原文来自于:http://top.jobbole.com/15551/ 这家网站的名称是 Learn X in Y minutes,包括了几十种编程语言的快速学习入门教程.打开几种编程语言来看了一下, ...
- nodejs【伪】入门教程
声明: 本文适合白的不能再白的小白 不要被标题误导,本文不会讲nodejs基础,只是本人学习流程和资料的一个整合 如果想找一大堆教程自己看,没有电梯,自己拉到文章最下方吧 一.nodejs是什么 ...
- TensorFlow 中文资源全集,官方网站,安装教程,入门教程,实战项目,学习路径。
Awesome-TensorFlow-Chinese TensorFlow 中文资源全集,学习路径推荐: 官方网站,初步了解. 安装教程,安装之后跑起来. 入门教程,简单的模型学习和运行. 实战项目, ...
- laravel 中CSS 预编译语言 Sass 快速入门教程
CSS 预编译语言概述 CSS 作为一门样式语言,语法简单,易于上手,但是由于不具备常规编程语言提供的变量.函数.继承等机制,因此很容易写出大量没有逻辑.难以复用和扩展的代码,在日常开发使用中,如果没 ...
- Apache Solr入门教程(初学者之旅)
Apache Solr入门教程(初学者之旅) 写在前面:本文涉及solr入门的各方面,建议边思考边实践,相信能帮助你对solr有个清晰全面的了解并能简单实用. 在Apache Solr初学者教程的这个 ...
随机推荐
- HDU-4869 Turn the pokers
原题: Turn the pokers 思路:假设正面为0,反面为1.牌就像这样 000000....... .考虑到假如可以实现最终反面个数为m, 牌共n张, 则这n张排任取m个为反面 ...
- IOS开发——使用数据库
IOS开发——使用FMDB数据库 简介 需求作用: 如果需要保存大量的结构较为复杂的数据的时候,使用数据库,例如交规考试项目 1.数据库的基本介绍 数据库(DB)是一种数据模型组织起来并存放存储管理的 ...
- C# 的析构
首先介绍下关于C#的GC垃圾回收器,有了这个垃圾回收器c#的开发人员可以不用像C++开发人员那样关心垃圾回收! 但是GC是把双刃剑,GC仅仅对于托管资源进行管理,对非托管资源却无能为力,并且C#的开发 ...
- SQL Server选取本周或上一周数据
有关SQL Server中有关周的数据查询主要思路来自下面这个语句 select getdate(), dateadd(wk, datediff(wk, 0, DateAdd(Day,-1,getda ...
- 批量修改vss工作目录
vss作为源代码版本控制工具,可以针对不同的文件夹设置不同的本地工作目录,这样可以方便我们不同的个性化需求.但是往往实际情况是,我们设置了不同的工作目录,后来却发现导致引用混乱,每个人每次获取项目文件 ...
- ASP.NET MVC 学习笔记(一)
很久很久没有在博客园写过东西了,很多大虾也说过展示自己最好的地方就是有一个博客作为笔记,展示一下自己的学习和研究成果. 最近决心将公司的一款产品改用MVC的方式实现,于是乎就开始在园子里面疯狂的寻找各 ...
- EQueue - 详细谈一下消息持久化以及消息堆积的设计
前言 之前写了一篇文章,总体介绍了EQueue.在看这篇文章之前如果还没看过那篇文章,可能会看不懂这篇文章.所以建议没看过的朋友务必先看一下那篇文章中所提到的各种概念,这样才能更好的理解本文所说的内容 ...
- 可在广域网部署运行的QQ高仿版 -- GG叽叽V1.8(源码)
距离的GG 1.0发布已经三周了,这三周内,我利用业余时间为GG增加了视频聊天的功能.个人觉得进展有些缓慢,主要是因为大多数时间都花在了UI上.由于本人不会PS,所以图片素材都是从网上一个一个搜下来的 ...
- Windows Phone 8.1上的开发人员请看
1)SDK选择:如果你是在Windows Phone 8.1上做一个新App, 或者想把7.x/8.0的App移植到8.1上,请使用WinRT SDK,而不是Silverlight.当然Silverl ...
- Handlebars.js循环中索引(@index)使用技巧(访问父级索引)
使用Handlebars.js过程中,难免会使用循环,比如构造数据表格.而使用循环,又经常会用到索引,也就是获取当前循环到第几次了,一般会以这个为序号显示在页面上. Handlebars.js中获取循 ...