Chinaunix首页 | 论坛 | 博客
  • 博客访问: 267290
  • 博文数量: 74
  • 博客积分: 1470
  • 博客等级: 上尉
  • 技术积分: 793
  • 用 户 组: 普通用户
  • 注册时间: 2008-11-25 21:01
文章分类

全部博文(74)

文章存档

2011年(1)

2010年(32)

2009年(32)

2008年(9)

我的朋友

分类: C/C++

2010-01-03 14:28:13

  今天由于要用到静态链接库,所以就学习了一下相关知识,总结如下:

静态链接库(一般命名为libxxx.a)就是很多.o文件的集合,在你的项目中如果有一个子模快,这个子模块只是给总控模块提供一个函数接口,那么你就可以考虑把这个子模快编译成静态链接库libxxx.a,然后在总控模块中编译的时候,只需-L包含链接库所在的目录,再-lxxx引用链接库就行.

当然,你也可以用动态链接库,具体的动态链接库创建和引用,做法和静态链接库大同小异,只是动态链接库是在程序执行的时候是动态的添加到内存的,所以可以实现进程之间的资源共享.
另外动态链接库可以做到所有的函数本着“有需求才调入”的原则,于是大大节省了系统资源:也就是说什么时候或者什么情况下,链接载入哪个动态链接库函数,完全由程序员在程序代码中控制。这样,当你有一个相当大的工程,每次运行的时候,由于不同的操作需求,就只会有一小部分程序被载入内存。

具体给一个例子,先看一下工程的目录结构:
$ ls -RF
.:
lib/  main.c  Makefile

./lib:
Makefile.a  Makefile.so  string.h  strlen.c  strnlen.c

在工程主目录下有main.c主控程序,Makefile文件和lib目录
lib目录下有string.h头文件,strlen.c和strnlen.c,这三个文件里的函数就是我们想生成的库函数
Makefile.a生成静态链接库的makefile文件
Makefile.so生成动态链接库的makefile文件



好,让我们看一看这些文件的具体内容:

头文件string.h,声明相关函数原形
$cat lib/string.h

int Strlen(char *pStr);
int StrNlen(char *pStr, unsigned long ulMaxLen);



strlen.c:函数Strlen的实现,获取给定字符串的长度
$cat lib/strlen.c
 

#include <stdio.h>
#include <assert.h>
int Strlen(char *pStr)
{
    unsigned long ulLength;
    assert(NULL != pStr);

    ulLength = 0;
    while(*pStr++)
    {
        ulLength++;
    }
    return ulLength;
}



strnlen.c:函数StrNlen的实现,获取给定字符串的长度,如果输入字符串的长度大于指定的最大长度,则返回最大长度,否者返回字符串的实际长度
$cat lib/strnlen.c

#include<stdio.h>
#include<assert.h>
int StrNlen(char *pStr, unsigned long ulMaxLen)
{
    unsigned long ulLength;
    assert(NULL != pStr);
    if(ulMaxLen <= 0)
    {
        printf("Wrong Max Length!\n");
        return -1;
    }
    ulLength = 0;
    while(*pStr++ && ulLength < ulMaxLen)
    {
        ulLength++;
    }
    return ulLength;
}


这三个文件是在lib/目录下.

Makefile.a:生成静态链接库的makefile文件
$ cat lib/Makefile.a

libstr.a: strlen.o strnlen.o
    $(AR) r $@ $^
    $(RM) $^

.PHONY : clean
clean :
    rm -f libstr.a



Makefile.so:生成动态链接库的makefile文件
$ cat Makefile.so

libstr.so: strlen.o strnlen.o
    gcc -fpic -shared -o $@ $^
    $(RM) $^

.PHONY : clean
clean :
    rm -f libstr.so


-fpic 使输出的对象模块是按照可重定位地址方式生成的
-shared指定把对应的源文件生成对应的动态链接库文件libstr.so文件


main.c:总控程序

#include <stdio.h>
#include "./lib/string.h" //静态库对应函数的头文件


int main(int argc, char* argv[])
{
    char str[] = {"hello world"};
    unsigned long ulLength = 0;

    printf("The string is : %s\n", str);
    ulLength = Strlen(str);
    printf("The string length is : %d(use Strlen)\n", ulLength);
    ulLength = StrNlen(str, 10);
    printf("The string length is : %d(use StrNlen)\n", ulLength);

    return 0;
}



总控Makefile
$ cat Makefile

CC = gcc
CFLAGS = -Wall -g
LIBPATH = -L./lib
LIB = -lstr

main: main.o
    $(CC) $(CFLAGS) -o $@ main.o $(LIBPATH) $(LIB)
    $(RM) *.o

.PHONY:clean
clean:
    -rm -f main




下面看一看怎么生成和使用静态链接库/动态链接库
1.静态链接库的生成:
$ cd lib
$ make -f Makefile.a
cc    -c -o strlen.o strlen.c
cc    -c -o strnlen.o strnlen.c
ar r libstr.a strlen.o strnlen.o
ar: creating libstr.a
rm -f strlen.o strnlen.o
这样就生成了静态链接库libstr.a了.

2.静态链接库的使用:


$ cat Makefile
CC = gcc
CFLAGS = -Wall -g
LIBPATH = -L./lib
LIB = -lstr

main: main.o
    $(CC) $(CFLAGS) -o $@ main.o $(LIBPATH) $(LIB)
    $(RM) *.o

.PHONY:clean
clean:
    -rm -f main

-L指定库文件的路径
-l引用库文件

看看结果:
$ make
gcc -Wall -g   -c -o main.o main.c
gcc -Wall -g -o main main.o -L./lib -lstr
rm -f *.o
$ ls
lib  main  main.c  Makefile
生成了可执行文件main
$ ./main
The string is : hello world
The string length is : 11(use Strlen)
The string length is : 10(use StrNlen)


3.动态链接库的生成:
先删除刚才生成的静态链接库
$ cd lib
$ make clean -f Makefile.a
rm -f libstr.a

$ make -f Makefile.so
cc    -c -o strlen.o strlen.c
cc    -c -o strnlen.o strnlen.c
gcc -fpic -shared -o libstr.so strlen.o strnlen.o
rm -f strlen.o strnlen.o
$ ls
libstr.so  Makefile.a  Makefile.so  string.h  strlen.c  strnlen.c
生成了动态链接库libstr.so

4.动态链接库的使用:
使用方法和静态链接库一样,还用的是静态链接库的那个Makefile
$ cat Makefile
CC = gcc
CFLAGS = -Wall -g
LIBPATH = -L./lib
LIB = -lstr

main: main.o
    $(CC) $(CFLAGS) -o $@ main.o $(LIBPATH) $(LIB)
    $(RM) *.o

.PHONY:clean
clean:
    -rm -f main

-L指定库文件的路径
-l引用库文件

$ make
gcc -Wall -g   -c -o main.o main.c
gcc -Wall -g -o main main.o -L./lib -lstr
rm -f *.o
$ ls
lib  main  main.c  Makefile
生成了可执行文件main
$ ./main
./main: error while loading shared libraries: libstr.so: cannot open shared object file: No such file or directory


这就是动态链接库和静态链接库使用时唯一的区别:

在程序运行期间,也需要告诉系统去哪里找你的动态链接库文件.在UNIX下是通过定义名为LD_LIBRARY_PATH 的环境变量来实现的.只需将动态链接库的目录path赋值给此变量即可。
为了让执行程序顺利找到动态库,有三种方法:
1)把库拷贝到/usr/lib和/lib目录下.
2)在LD_LIBRARY_PATH环境变量中加上库所在路径.例如动态库libstr.so在/home/xulei/test/lib目录下,以bash为例,使用命令:
$export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/xulei/test/lib
在环境变量LD_LIBRARY_PATH后添加/home/xulei/test/lib
3) 修改/etc/ld.so.conf文件,把库所在的路径加到文件末尾
然后sudo ldconfig
这样,加入的目录下的所有库文件都可见.

在我的ubuntu下是这样的:
$ cat /etc/ld.so.conf
include /etc/ld.so.conf.d/*.conf
$ ls /etc/ld.so.conf.d/
i486-linux-gnu.conf  libc.conf
$ cat /etc/ld.so.conf.d/libc.conf
# libc default configuration
/usr/local/lib

当然由于ld.so.conf包含/etc/ld.so.conf.d/*.conf,你也可以自己新建个文件vi /etc/ld.so.conf.d/myownlib.conf,然后在其中输入/home/xulei/test/lib

我用的是第二种方法:
$ export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/xulei/test/lib
$ ./main
The string is : hello world
The string length is : 11(use Strlen)
The string length is : 10(use StrNlen)


最后,使用ldd命令查看可执行文件依赖于哪些库,
$ ldd main
    linux-gate.so.1 =>  (0xb7f43000)
    libstr.so => /home/xulei/test/lib/libstr.so (0xb7f3f000)
    libc.so.6 => /lib/tls/i686/cmov/libc.so.6 (0xb7dd9000)
    /lib/ld-linux.so.2 (0xb7f44000)


reference:
linux静态链接库与动态链接库http://blog.chinaunix.net/u2/76292/showart_1274181.html
Linux静态/动态链接库的创建和使用http://dev.csdn.net/article/84/84562.shtm








阅读(1626) | 评论(3) | 转发(0) |
给主人留下些什么吧!~~

chinaunix网友2010-03-11 09:35:11

汗~,是.so,估计当时打错了。

chinaunix网友2010-01-05 14:35:39

*.o动态链接库?

chinaunix网友2010-01-05 14:10:19

师兄,我只是在实验中,用过显式调用*.o动态链接库的哦~ 当时你这种的我也看过了,不过嫌麻烦,就没有用。 blog.sina.com.cn/loverdc