automake编写完整的项目---动态库+可执行文件

上一篇博客automake简介中简单的介绍了automake的使用流程,但是真正的项目肯定是很复杂的,包含不同的目录,有的生成动态库,有的是可执行文件,本文就用一个简单的opencv项目来介绍一下automake在大型项目的使用方法。本文中,最外层包含两个目录,一个目录是src,里面是不同模块的源文件,最后是要生成动态库的;一个目录是sample,里面是一个应用程序,调用src中的动态库生成可执行文件,而src中又包含了不同的目录,每个子目录最后生成一的so动态库,具体如下:其中,smooth、sharpen、segment分别生成三个动态库,而processManage则调用他们三个也生成一个动态库,最后sample中的main函数调用processManage提供的接口间接的调用smooth、sharpen、segment三个动态库实现一个应用程序:


具体步骤如下:

1、在最外层目录(和src、sample在同一级)执行autoscan命令生成autoscan.log和configure.scan文件

2、mv configure.scan configure.ac且修改configure.ac(早期使用configure.in)

#                                               -*- Autoconf -*-
# Process this file with autoconf to produce a configure script.

AC_PREREQ([2.68])
AC_INIT(segment, 1.0, email)            
AM_INIT_AUTOMAKE(segment,1.0)    #手动添加
AC_CONFIG_SRCDIR([sample/segment.cpp])  #用于检测源码目录的有效性,任选一个源码文件即可
AC_CONFIG_HEADERS([config.h])

# Checks for programs.
AC_PROG_CXX
AC_PROG_CC

# Checks for libraries.
# Checks for header files.
# Checks for typedefs, structures, and compiler characteristics.
# Checks for library functions.

AC_CONFIG_FILES([Makefile src/Makefile src/smooth/Makefile src/sharpen/Makefile src/segment/Makefile src/processManage/Makefile sample/Makefile])   #需要生成的所有Makefile  
AC_OUTPUT

3、使用aclocal生成aclocal.m4文件

4、执行autoconf命令生成configure文件

5、执行autoheader命令生成config.h和config.h.in 

6、在每一个需要生成Makefile的目录里创建Makefile.am文件

6.1:如果是父目录(和src、sample在同一级),则必须包含SUBDIRS = src sample  (注意顺序),相当于调用子目录去生成Makefile,如果没有其它的操作,则只有这一行即可.同理,src中的(和smooth、sharpen、segment、processManage同一级的)Makefile.am也只有一行,即SUBDIRS = smooth sharpen segment processManage。对于smooth、sharpen和segment三个目录中Makefile.am分别生成一个动态库,方法类似,这里给出smooth目录中的Makefile.am:

INCLUDES = `pkg-config opencv --cflags` -I./   #头文件的目录
projectlibdir=$(libdir)
projectlib_PROGRAMS = libsmooth.so    #动态库的名字
libsmooth_so_SOURCES = GaussSmooth.cpp MedianSmooth.cpp GaussSmooth.h Smooth.h
libsmooth_so_LDFLAGS = -shared -fpic #GCC编译动态库的选项
上面宏的具体含义,参考上一篇博客的表格

对于processManage中的Makefile.am,由于调用了另外三个so,所以稍微有点不同,具体如下:

INCLUDES = `pkg-config opencv --cflags` -I../smooth/ -I../sharpen -I../segment/ -I./
projectlibdir = $(libdir)
projectlib_PROGRAMS = libmanage.so
libmanage_so_SOURCES = processManage.cpp  processManage.h
libmanage_so_LDFLAGS = -shared -fpic #GCC编译动态库的选项
libmanage_so_LDADD= -L../smooth/ -L../sharpen/ -L../segment/ -lsmooth -lsharpen -lsegment  #依赖的库
最后是sample中可执行文件的Makefile.am,具体如下:

AUTOMAKE_OPTIONS = foreign
INCLUDES=`pkg-config opencv --cflags` -I../src/processManage/ -I../src/smooth/ -I../src/sharpen/ -I../src/segment/ -I. 
bin_PROGRAMS = segment
segment_SOURCES = segment.cpp
LIBS = `pkg-config opencv --libs` 
segment_LDADD =-L../src/smooth/ -L../src/sharpen/ -L../src/segment/ -L../src/processManage/  -lsmooth -lsharpen -lsegment -lmanage   
注意事项:由于GCC中依赖的动态库必须放在-o参数的后面,所以`pkg-config opencv --libs`必须放到LIBS变量中,因为查看后面生成的Makefile可以看到,_LDADD和_LIBS在-o的后面,而_LDFLAGS在-o的前面,而 `pkg-config opencv --libs`不能放到_LDADD中
6.2: 如果使用libtool生成动态库,可以使用libtool生成可移植性的动态库,生成的动态库为.lo文件,它会调用自己目录下子目录./libs/中的.so文件,使用.lo文件和使用.so文件的方法类似,只是需要使用-lz参数,模版如下(下面用一个简单的项目举例):

AUTOMAKE_OPTIONS=foreign
lib_LTLIBRARIES=libhello.la   #要生成的动态库的名字
libhello_la_SOURCES=test.cpp #依靠的源文件和头文件,使用空格分开
此时的可执行文件,模版如下:

AUTOMAKE_OPTIONS=foreign
INCLUDES= -I../include     #依赖的头文件目录
bin_PROGRAMS=hello    
hello_SOURCES=hello.cpp
hello_LDADD= -L../lib/ -lz ../lib/test.lo  #依赖的库文件
7、如果要生成.lo动态库,则 libtoolize --automake --copy --force

8、touch NEWS README AUTHORS ChangeLog

9、执行automake --add-missing命令生成Makefile.in文件

10、执行./configure 命令生成Makefile文件

11、执行make && make install







你可能感兴趣的:(Linux系统运维)