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

相关文章

Nginx中配置使用非默认80端口进行服务的完整指南

《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令

Python WebSockets 库从基础到实战使用举例

《PythonWebSockets库从基础到实战使用举例》WebSocket是一种全双工、持久化的网络通信协议,适用于需要低延迟的应用,如实时聊天、股票行情推送、在线协作、多人游戏等,本文给大家介... 目录1. 引言2. 为什么使用 WebSocket?3. 安装 WebSockets 库4. 使用 We

python中的显式声明类型参数使用方式

《python中的显式声明类型参数使用方式》文章探讨了Python3.10+版本中类型注解的使用,指出FastAPI官方示例强调显式声明参数类型,通过|操作符替代Union/Optional,可提升代... 目录背景python函数显式声明的类型汇总基本类型集合类型Optional and Union(py

Java使用正则提取字符串中的内容的详细步骤

《Java使用正则提取字符串中的内容的详细步骤》:本文主要介绍Java中使用正则表达式提取字符串内容的方法,通过Pattern和Matcher类实现,涵盖编译正则、查找匹配、分组捕获、数字与邮箱提... 目录1. 基础流程2. 关键方法说明3. 常见场景示例场景1:提取所有数字场景2:提取邮箱地址4. 高级

使用SpringBoot+InfluxDB实现高效数据存储与查询

《使用SpringBoot+InfluxDB实现高效数据存储与查询》InfluxDB是一个开源的时间序列数据库,特别适合处理带有时间戳的监控数据、指标数据等,下面详细介绍如何在SpringBoot项目... 目录1、项目介绍2、 InfluxDB 介绍3、Spring Boot 配置 InfluxDB4、I

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定

使用Python实现无损放大图片功能

《使用Python实现无损放大图片功能》本文介绍了如何使用Python的Pillow库进行无损图片放大,区分了JPEG和PNG格式在放大过程中的特点,并给出了示例代码,JPEG格式可能受压缩影响,需先... 目录一、什么是无损放大?二、实现方法步骤1:读取图片步骤2:无损放大图片步骤3:保存图片三、示php

使用Python实现一个简易计算器的新手指南

《使用Python实现一个简易计算器的新手指南》计算器是编程入门的经典项目,它涵盖了变量、输入输出、条件判断等核心编程概念,通过这个小项目,可以快速掌握Python的基础语法,并为后续更复杂的项目打下... 目录准备工作基础概念解析分步实现计算器第一步:获取用户输入第二步:实现基本运算第三步:显示计算结果进

python之uv使用详解

《python之uv使用详解》文章介绍uv在Ubuntu上用于Python项目管理,涵盖安装、初始化、依赖管理、运行调试及Docker应用,强调CI中使用--locked确保依赖一致性... 目录安装与更新standalonepip 安装创建php以及初始化项目依赖管理uv run直接在命令行运行pytho

C#使用Spire.XLS快速生成多表格Excel文件

《C#使用Spire.XLS快速生成多表格Excel文件》在日常开发中,我们经常需要将业务数据导出为结构清晰的Excel文件,本文将手把手教你使用Spire.XLS这个强大的.NET组件,只需几行C#... 目录一、Spire.XLS核心优势清单1.1 性能碾压:从3秒到0.5秒的质变1.2 批量操作的优雅