在 C/C++ 项目中自动检测库依赖关系的最佳方法是啥?

Posted

技术标签:

【中文标题】在 C/C++ 项目中自动检测库依赖关系的最佳方法是啥?【英文标题】:What is the best way to auto detect library dependencies in a C/C++ project?在 C/C++ 项目中自动检测库依赖关系的最佳方法是什么? 【发布时间】:2010-12-08 11:12:21 【问题描述】:

在 C/C++ 项目中自动检测库依赖关系的最佳方法是什么?

我有一个项目,我拥有机器上的所有依赖项。它构建并运行。现在我想组合一个自动工具构建系统。我正在寻找一种自动检测所需的所有依赖项的好方法,例如使用的头文件和链接所需的库。

图书馆位似乎是我最难弄清楚的。我想说,为列表或其他东西中的每个函数生成 AC_CHECK_LIB 命令。我可能可以在 Perl 中做到这一点,但我必须想象它已经存在于其他地方。

我知道我可以使用 objdump 和 nm 查看符号,我可以使用这些实用程序找到函数所属的库,然后我可以在 configure.ac 中手动输入 AC_CHECK_LIB 命令来检查它。在这一点上,自动化会很棒。

谢谢, 陈兹

【问题讨论】:

【参考方案1】:

在 Windows 上,我使用 Dependency Walker 来处理类似的事情。它的输出很详细,但通常会显示可执行文件所需的每个库。

我不知道 linux 或 mac 有什么类似的东西,但我确信一定存在一些东西。

【讨论】:

【参考方案2】:

这种详尽的测试(即每个功能)是不必要的。更不用说它很难维护并且需要一段时间才能运行。

测试您知道需要测试的功能。如果您只是测试库的存在,请选择一个常用函数在测试中使用。如果您想确保某些功能仅在较新版本中可用,请使用仅在这些较新版本中提供的功能进行测试。

【讨论】:

【参考方案3】:

我现在也遇到了类似的挑战。 autoconf 对于 C++ 技巧来说真的不是那么方便,但它有基本的砖块来构建顶部的功能。我在这里和那里看了之后的建议:

阅读this article,它会给你带来新鲜的想法 在ac-archive 上查看 autoconf 宏的源代码(它包含在 Debian 中,因此您可以按原样使用它) 我个人写过simple helper,抄自AC_CHECK_LIBAX_CXX_CHECK_LIB。是的,您需要编写一个迷你测试程序,但这允许您测试类型、类(sizeof 可能有效,但构造函数呢?)、内联函数(这在链接器的帮助下是无法做到的)和外部函数(nm 无法做到这一点)。

来自aclocal.m4

# SYNOPSIS
#
# AX_TRY_LINK(library, includes, function-body [, action-if-true [, action-if-false]])
#
# DESCRIPTION
#
# This function is a wrapper around AC_ARG_WITH, which adds -I"value" to CPPFLAGS.
# "--with-" variable is initialized to default value, if it is passed.
#
AC_DEFUN([AX_TRY_LINK], [
    dnl Below logic is a workaround for the limitation, that variables may not allow
    dnl symbols like "+" or "-". See AC_CHECK_LIB source comments for more information.
    m4_ifval([$4], , [AH_CHECK_LIB([$1])])
    AS_LITERAL_IF([$1],
        [AS_VAR_PUSHDEF([ac_Lib], [ac_cv_lib_$1_$2])],
        [AS_VAR_PUSHDEF([ac_Lib], [ac_cv_lib_$1''_$2])])

    AC_CACHE_CHECK([for -l$1], [ac_Lib], [
        dnl Save the current state
        AC_LANG_SAVE
        AC_LANG_CPLUSPLUS
        ax_try_link_save_LIBS=$LIBS
        LIBS="-l$1 $LIBS"

        AC_TRY_LINK([$2], [$3], [AS_VAR_SET([ac_Lib], [yes])], [AS_VAR_SET([ac_Lib], [no])])

        dnl Restore the state to original regardless to the result
        LIBS=$ax_try_link_save_LIBS
        AC_LANG_RESTORE
    ])

    dnl If the variable is set, we define a constant and push library to LIBS by default or execute $4, otherwise execute $5.
    AS_VAR_IF([ac_Lib], [yes],
        [m4_default([$4], [
            AC_DEFINE_UNQUOTED(AS_TR_CPP(HAVE_LIB$1))
            dnl Do not prepend a library, if it is already in the list:
            (echo $LIBS | grep -q -- "-l$1 ") || LIBS="-l$1 $LIBS"
        ])],
        [$5]
    )
    AS_VAR_POPDEF([ac_Lib])
]) # AX_ARG_WITH

现在在configure.ac:

AC_INIT([ABC], [1.2.3])
AC_LANG([C++])
AC_PROG_CXX
AC_CXX_HAVE_STL

if test "x$ac_cv_cxx_have_stl" != "xyes"; then
    AC_MSG_ERROR([STL was not found; make sure you have installed libstdc++-dev])
fi

...

dnl openbabel library

sr_openbabel_lib=yes

AC_CHECK_HEADERS([openbabel/mol.h openbabel/obconversion.h openbabel/builder.h], [], [sr_openbabel_lib=no])
AX_TRY_LINK([openbabel], [
    #include <openbabel/mol.h>
    #include <openbabel/obconversion.h>
    #include <openbabel/builder.h>
], [
    OpenBabel::OBAtom atom;
    OpenBabel::OBMol mol;
    OpenBabel::OBConversion conversion;

    atom.IsHeteroatom();
    atom.IsCarbon();

    mol.NumAtoms();
    mol.NumBonds();
    mol.NumRotors();
    mol.GetAtom(0);

    conversion.ReadString(&mol, "");
    conversion.WriteString(&mol, false);
], [], [sr_openbabel_lib=no])

if test $sr_openbabel_lib != yes; then
    AC_MSG_ERROR([openbabel headers or library was not found (use --with-openbabel to define custom header location)])
fi

【讨论】:

以上是关于在 C/C++ 项目中自动检测库依赖关系的最佳方法是啥?的主要内容,如果未能解决你的问题,请参考以下文章

最佳 C/C++ 网络库

更新旧的 Angular nodeJS 项目的依赖关系解决问题和最佳实践?

(转)Java 日志框架解析(下) - 最佳实践

从现有 jar 创建 maven 工件的最佳方法

装Kingbase ES V7时为啥依赖检测时,总是提示我说Microsoft Visual C++ 2008未通过检测???

如何检测未使用的 Maven 依赖项