设为首页 收藏本站
查看: 1010|回复: 0

[经验分享] perl C/C++ 扩展(四)

[复制链接]

尚未签到

发表于 2015-12-26 14:35:12 | 显示全部楼层 |阅读模式
  在前面三篇博客中,我们了解到如何使用c/c++ 扩展自己的perl 库,但是博主在学习过程中,对动态库或静态库的加载不是十分了解,后来自己又细挖一下。后来就有了这篇博文,再后来,没有再后来了,囧!!
  我们先来看看 perl c/c++ 扩展(二)中的 Makefile.PL



WriteMakefile(
NAME              => 'two_test',
VERSION_FROM      => 'lib/two_test.pm', # finds $VERSION
PREREQ_PM         => {}, # e.g., Module::Name => 1.1
($] >= 5.005 ?     ## Add these new keywords supported since 5.005
(ABSTRACT_FROM  => 'lib/two_test.pm', # retrieve abstract from module
AUTHOR         => 'root <root@>') : ()),
LIBS              => [''], # e.g., '-lm'
DEFINE            => '', # e.g., '-DHAVE_SOMETHING'
INC               => '-I.', # e.g., '-I. -I/usr/include/other'
# Un-comment this if you add C files to link with later:
# OBJECT            => '$(O_FILES)', # link all the C files too
MYEXTLIB => 'mylib/libchen.a'
);
  注意:
  我们第一次写时,是没有填写 LIBS 与 INC 字段的,但是我们增加了一个 MYEXTLIB 的字段,指定我们要增加‘mylib/libchen.a'的静态库
  
  由于在perl c/c++ 扩展(三)中,我们是使用了***.so 动态库来编译,而且也能正常工作,所以我就想,在(二)里,肯定也能调用动态库。
  
  于是对原程序改动了一下,编译出一个libchen.so的文件,放在mylib/ 目录下。
  修改一下Makefile.PL 脚本



MYEXTLIB => 'mylib/libchen.so'
  
  还是使用之前的命令



perl Makefile.Pl
make
make install
  编译通过,但在运行测试脚本时出错,说是找不到动态库。
  这个问题简单,给环境变量LD_LIBRARY_PATH 添加一下libchen.so路径即可。再次运行测试脚本,通过。
  
  由于需要添加环境变量这个细节又引起我的注意,在使用中,我知道perl c/c++ 扩展(三) 的编译方法不依赖环境变量的,也就是说,无论切换到什么窗口都可成功运行测试脚本。
  
  我们来看看(三)是怎么写Makefile.PL的



WriteMakefile(
NAME              => 'three_test',
VERSION_FROM      => 'lib/three_test.pm', # finds $VERSION
PREREQ_PM         => {}, # e.g., Module::Name => 1.1
($] >= 5.005 ?     ## Add these new keywords supported since 5.005
(ABSTRACT_FROM  => 'lib/three_test.pm', # retrieve abstract from module
AUTHOR         => 'root <root@>') : ()),
LIBS              => ['-L./ -lduck'], # e.g., '-lm'
DEFINE            => '', # e.g., '-DHAVE_SOMETHING'
'CC'              => $CC,
'LD'              => '$(CC)',
INC               => '-I.', # e.g., '-I. -I/usr/include/other'
# Un-comment this if you add C files to link with later:
# OBJECT            => '$(O_FILES)', # link all the C files too
'XSOPT'           => '-C++',
'TYPEMAPS'        => ['perlobject.map']
);
  我们看到,除了新增了XSOPT  TYPEMAPS  CC  LD字段外,还修改了 LIBS  INC字段。
  
  而且LIBS 字段的内容就是g++ 的lib 库,INC 字段就是头文件的地址。
  
  所以我们将(二) 的Makefile.PL修改一下,变成



WriteMakefile(
NAME              => 'two_test',
VERSION_FROM      => 'lib/two_test.pm', # finds $VERSION
PREREQ_PM         => {}, # e.g., Module::Name => 1.1
($] >= 5.005 ?     ## Add these new keywords supported since 5.005
(ABSTRACT_FROM  => 'lib/two_test.pm', # retrieve abstract from module
AUTHOR         => 'root <root@>') : ()),
LIBS              => ['-Lmylib -lchen'], # e.g., '-lm'
DEFINE            => '', # e.g., '-DHAVE_SOMETHING'
INC               => '-Imylib', # e.g., '-I. -I/usr/include/other'
# Un-comment this if you add C files to link with later:
# OBJECT            => '$(O_FILES)', # link all the C files too
#MYEXTLIB => 'mylib/libchen.so'
);
  修改了 LIBS  INC 字段,指定动态库的地址为-Lmylib,使用chen的库-lchen
  INC字段写chen.h 头文件的地址。
  
  注意,这时候,我们还是沿用libchen.so的动态库。
  
  编译并按装



perl Makefile.PL
make
make install
  运行测试脚本,OK
  切换一个窗口(等于没有执行export LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/learn/perl/two_test/mylib) ,再运行测试脚本,依然OK。
  
  但是当我们将mylib目录下的动态库改名,变成libchen.so.bak。再运行测试脚本,失败,报错说找不到libchen.so的库



错误信息:/usr/local/lib/perl/5.14.2/auto/two_test/two_test.so' for module two_test: libchen.so: 无法打开共享对象文件: 没有那个文件或目录 at /usr/share/perl/5.14/XSLoader.pm line 71
  
  到这里,我们就可以知道了,实质上在LIBS 字段写上库名,应该是在生成Makefile时,将绝对路径包含了,并且自动添加到环境变量了,所以使用动态库生成的perl 扩展,运行环境都需要将动态库加入环境变量里。
  
  上面的测试,扩展包还需要运行环境有动态库,对于移植性来说,多有不便。所以我们还是希望使用静态库编译。
  方法很简单,只要将libchen.so 动态库替换成 libchen.a 静态库,在重新编译扩展包就好了。



perl Makefile.PL
make
make install
  这次,无论你怎造,perl 的测试程序都能正确运行。
  
  我们再回头看看(二)的Makefile.PL



WriteMakefile(
NAME => 'two_test',
VERSION_FROM => 'lib/two_test.pm', # finds $VERSION
PREREQ_PM => {}, # e.g., Module::Name => 1.1
($] >= 5.005 ? ## Add these new keywords supported since 5.005
(ABSTRACT_FROM => 'lib/two_test.pm', # retrieve abstract from module
AUTHOR => 'root <root@>') : ()),
LIBS => [''], # e.g., '-lm'
DEFINE => '', # e.g., '-DHAVE_SOMETHING'
INC => '-I.', # e.g., '-I. -I/usr/include/other'
# Un-comment this if you add C files to link with later:
# OBJECT => '$(O_FILES)', # link all the C files too
MYEXTLIB => 'mylib/libchen.a'
);
  它也是使用了libchen.a静态库,经过测试,运行时也能不再依赖库了。
  
  总结:
  无论是使用MYEXTLIB 字段,还是通过指定LIBS  INC来指向库文件,只要是使用静态库编译,扩展包都能脱离库的依赖。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-156633-1-1.html 上篇帖子: perl 下篇帖子: Perl的命令行参数
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表