NS3 使用 waf 工具添加外部库

2024-03-22 21:44
文章标签 工具 使用 外部 waf ns3

本文主要是介绍NS3 使用 waf 工具添加外部库,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我最近在写 NS3 的时候想要把他人写好的外部库添加到 NS3 中一起编译,在 Linux 系统中,添加外部库往往通过编译选项 -l<外部库名> 来添加,而在大型项目中往往需要把外部库写到 Makefile 文件中通过 make 来编译。奈何 NS3 的早期版本都是使用 waf 编译的,这导致项目中是没有 Makefile 文件的,无法直接通过修改 Makefile 文件实现(如果使用的是 NS3 晚期的版本——例如3.39等,官方文档就提供了如何使用外部库 HOWTO use ns-3 with other libraries)。

建议了解本文之前先弄明白 C++ 是怎么通过 Makefile 编译的,传送门:在Linux中开发C++

在我尝试的过程中,我是看了 NS3使用waf工具添加外部库(static library or shared library)亲测可用,已成功添加ZMQ库 这一篇文章成功编译的,但我感觉这种做法还不够优雅,因此我读过 waf 的代码之后重新说一下怎么做,如果赶时间可以直接阅读上面这篇文章。

在本文,我们则通过修改 waf 的具体代码文件——wscript 来引入第三方库。我们直接修改 NS3 根目录下的 wscript 文件即可。点开该文件可以发现,这个其实就是 python 代码,而作者在 938-946行其实给出了添加用户自定义编译 flags 的线索,如下:

    # append user defined flags after all our onesfor (confvar, envvar) in [['CCFLAGS', 'CCFLAGS_EXTRA'],['CXXFLAGS', 'CXXFLAGS_EXTRA'],['LINKFLAGS', 'LINKFLAGS_EXTRA'],['LINKFLAGS', 'LDFLAGS_EXTRA']]:if envvar in os.environ:value = shlex.split(os.environ[envvar])conf.env.append_value(confvar, value)

这一段代码的意思就是额外对 conf.env 的 “CCFLAGS”、“CXXFLAGS”、“LINKFLAGS” 变量添加值,对应值的来源是环境变量 “CCFLAGS_EXTRA”、 “CXXFLAGS_EXTRA”、 “LINKFLAGS_EXTRA”、 “LDFLAGS_EXTRA”。根据我们对 C++ 编码的经验,这里添加的三个变量含义如下:

  • CCFLAGS:编译 C语言文件(.c)时额外添加的选项,例如我们可以添加 -Wall 等等。
  • CXXFLAGS:编译 C++语言文件(.cc、.cpp)时额外添加的选项,同上。
  • LINKFLAGS:连接库时给出的共享链接库,格式即 -l<外部库名>。

注:C 语言编译分为编译和连接两个过程,如不了解先得去看看 在Linux中开发C++ ,我们常用的 IDE 以及 gcc编译 都把这两个过程给合并了而已。

显然,我们要添加外部库,就需要把外部库文件交给 LINKFLAGS 变量,因此我们只需要设置 “LINKFLAGS_EXTRA” 和 “LDFLAGS_EXTRA” 两个环境变量就行。在 C++ 编译中,“LIBRARY_PATH” 环境变量给出编译过程中所需要的动态库路径,而 “LD_LIBRARY_PATH” 则给出运行过程中所需要的动态库路径。只不过在 waf 这里把这两个路径给合并了而已,所以也推荐为了避免错误,我们把第三方库生成的编译库和运行库都放一起,然后指定同一个路径就行。

在上面那篇文章中,作者给出引入第三方库 -lzmq 的成功经验为在NS3根目录的 wscript 中,def configure 函数添加如下代码:

   #Add extenal lib zmqconf.env.append_value('INCLUDES', '/usr/local/include')# conf.env.append_value('LINKFLAGS', '-lzmq' )conf.env.append_value("LINKFLAGS", ["-lzmq","-L/usr/local/lib/libzmq.a"])# conf.env.append_value("LIB", ["Geographic"])#仅添加上面代码并不行,会报出找不到函数的未定义错误#添加这些后,突然就成功了哈哈哈哈conf.env.append_value("LINKFLAGS", ["-L/usr/local/lib"])  #链接库地址conf.env.append_value('CXXFLAGS', '-I/usr/local/include/zmq') #头文件地址conf.env.append_value("LIB", ["zmq"]) #用于列出链接库列表

这里的几句话其实只有两句是有用的:

    conf.env.append_value("LINKFLAGS", ["-L/usr/local/lib"])  #链接库地址conf.env.append_value("LIB", ["zmq"]) #用于列出链接库列表

而至于添加 INCLUDES 当且仅当你是使用 #include<> 来引用的时候才有必要,但我其实不太推荐这样,你把外部库的 include 文件夹放 workspace 下我认为是更合理一些的,在根目录下单独建一个 include 文件夹即可。而前面一句直接 “-L/usr/local/lib/libzmq.a” 这是不符合逻辑的呀,怎么可能直接 Link 到库名上呢,只需要 Link 到放库的文件夹即可。

而上面说过,LINKFLAGS 实际上在 wscript 也提供了通过 “LINKFLAGS_EXTRA” 和 “LDFLAGS_EXTRA” 环境变量添加的方法,因此对于作者的例子来说,更优雅的做法是在运行前通过 shell 添加环境变量:

export LINKFLAGS_EXTRA="-lzmq -L/usr/local/lib"

当然如果这样添加关掉 Shell 就没了,如果希望永久保留就放到 .bashrc 文件中,这里就不展开了,具体自己百度一下:Linux 如何设置永久环境变量。当然其实临时变量问题也不大,我们这部分代码仅在 ./waf configure 的时候会运行(写在 def configure 函数中的),而我们生成了配置文件之后一般直接 ./waf build 进行编译就行了,没必要总是生成新的配置文件,所以后续我们再次编译代码的时候是没有影响的!

再说一点,就是经过我的测试, conf.env.append_value(“LIB”, [“zmq”]) 这一句话不加是会报错的,具体原因我到目前还没有完全理解,可能要完整看一下所有 wscript 才能清楚。但 wscript 里面又没有提供添加 “LIB” 的方法,因此我认为比较优雅的改动方式是,直接在它的 for 循环里增加添加 “LIB” 的方法,即把 wscript 第 938-946行 修改为:

    # append user defined flags after all our onesfor (confvar, envvar) in [['CCFLAGS', 'CCFLAGS_EXTRA'],['CXXFLAGS', 'CXXFLAGS_EXTRA'],['LINKFLAGS', 'LINKFLAGS_EXTRA'],['LINKFLAGS', 'LDFLAGS_EXTRA'],['LIB', 'LIB_EXTRA']]:if envvar in os.environ:value = shlex.split(os.environ[envvar])conf.env.append_value(confvar, value)

然后我们就可以同样在 shell 中添加环境变量:

export LIB_EXTRA="zmq"

即可成功编译,这样我们修改 wscript 的次数比较少,尽量把我们自己实现的东西通过系统环境变量的方式传入会显得更优雅一些。

总结一下,如果要添加一个新的链接库,首先需要把 NS3 根目录下的 wscript 的 938-946行 修改一下;然后分别设置 LINKFLAGS_EXTRA 和 LIB_EXTRA 环境变量,最后记得 ./waf clean 之后重新 ./waf configure 再编译(./waf build)。

最后说一点:还是推荐使用 Makefile,写 CMakeLists.txt 比写这个 waf 文件要舒服多了。

这篇关于NS3 使用 waf 工具添加外部库的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/836248

相关文章

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

Spring Security简介、使用与最佳实践

《SpringSecurity简介、使用与最佳实践》SpringSecurity是一个能够为基于Spring的企业应用系统提供声明式的安全访问控制解决方案的安全框架,本文给大家介绍SpringSec... 目录一、如何理解 Spring Security?—— 核心思想二、如何在 Java 项目中使用?——

springboot中使用okhttp3的小结

《springboot中使用okhttp3的小结》OkHttp3是一个JavaHTTP客户端,可以处理各种请求类型,比如GET、POST、PUT等,并且支持高效的HTTP连接池、请求和响应缓存、以及异... 在 Spring Boot 项目中使用 OkHttp3 进行 HTTP 请求是一个高效且流行的方式。

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完