/*  
    SList.h  
    单链表的实现
*/


#ifndef SLIST_H_
#define SLIST_H_
#include 
<iostream>

class Node
{
public:
    
int info;
    Node 
*next;
    Node(
int el,Node* ptr=0)
    
{
        info 
= el;
        next 
= ptr;
    }

}
;

class SList{
public:
    SList()
{
        head 
= tail = 0;
    }

    
int isEmpty(){
        
return head == 0;
    }

    
void addToHead(int);
    
int deleteFromHead();
    friend std::ostream
& operator<<(std::ostream&const SList&);
    
virtual ~SList();
private:
    Node 
*head,*tail;
}
;

#endif  //SLIST_H_

/*  SList.cpp  */

#include 
"SList.h"

SList::
~SList(){
    
for(Node* p;!isEmpty();){
        p 
= head->next;
        delete p;
        head 
= p;
    }

}


void SList::addToHead(int el){
    head 
= new Node(el,head);
    
if(tail == 0)
        tail 
= head;
}


int SList::deleteFromHead(){
    
int el = head->info;
    Node 
*tmp = head;
    
if(head == tail)
        head 
= tail = 0;
    
else head = head->next;
    delete tmp;
    
return el;
}


std::ostream
& operator<<(std::ostream& os, const SList& ss){
    Node
* p;
    p
=ss.head;
    os
<<p->info;
ign="top" />    
for(p=p->next;p;p=p->next)
        os
<<"->"<<p->info;
    
return os;
}



/*  main.cpp  */

#include 
<iostream>
#include 
"SList.h"

using namespace std;
int main(int argc, char* argv[])
{    
    
  
int ss[]={20,11,21,23,10,19,17,4};
    size_t n 
= sizeof(ss)/sizeof(int);
    SList sl;
    
for(int i=0;i<n;++i)
        sl.addToHead(ss[i]);
    cout
<<sl<<endl;
    
return 0;
}

一、创建共享库
1 单独编译SList.cpp,编译时需要传入-fPIC选项,告诉编译器生成位置无关代码.
位置无关代码可以被加载到地址空间的任意位置而不需要修改.

[root@LEE src]# ls
main.cpp  SList.cpp  SList.h
[root@LEE src]# g++ -fPIC -g -c SList.cpp

2 当链接到库时,为链接器传入-shared选项,把目标文件SList.o链接为共享对象
libSList.so.1.0.1
每个共享库都有一个特定的搜索名(soname).搜索名约定如下:
lib+库名+.so+.版本号
在文件系统中,搜索名是一个指向实名的符号连接.每个共享库也有一个特定的实名,约定如下:
搜索名+.子版本号+.发布号
你可以使用一个特殊的编译器选项-Wl,option,将option传给ld,用逗号分隔多个option,
为了在所有的系统上得到最好结果,链接libSList到标准C++库上

[root@LEE src]# g++ -g -shared -Wl,-soname,libSList.so.1 -o libSList.so.1.0.1 SList.o -lstdc++
[root@LEE src]# ls
libSList.so.1.0.1  main.cpp  SList.cpp  SList.h  SList.o

libSList.so.1是搜索名,libSList.so.1.0.1是实名,SList.o是目标文件(也可以是多个目标文件的列表)
 ,-lstdc++是库需要访问的库(也可以是库的列表-llibrary,关于此选项参考附录.)

3 创建一个从soname链接到库
[root@LEE src]# ln -fs libSList.so.1.0.1 libSList.so.1
[root@LEE src]# ln -fs libSList.so.1 libSList.so
 
4  使用-L使链接器在当前目录中查找库,并且使用-lSList告诉它要链接哪个库
[root@LEE src]# g++ -g -c main.cpp -o main.o
[root@LEE src]# g++ -g -o main main.o -L. -lSList
[root@LEE src]# ls
libSList.so    libSList.so.1.0.1  main.cpp  SList.cpp  SList.o
libSList.so.1  main               main.o    SList.h

5 运行命令
[root@LEE src]# LD_LIBRARY_PATH=$(pwd) ./main
4->17->19->10->23->21->11->20

LD_LIBRARY_PATH
  提供用来搜索库的目录路径,以冒号作为间隔.正常情况下它不应该被设置,因为系统文件
/ect/ld.so.conf提供了默认的路径.

二、使用链接库
 当运行一个程序时,动态装载器通常在/ect/ld.so.conf.d目录查找程序所需要的库.但是,如果
LD_LIBRARY_PATH 环境变量被设置,它首先扫描在LD_LIBRARY_PATH 中列出的目录.对于上一节5,如果直接运行
命令 ./main,会出现找不到库的现象.如:
[root@LEE src]# ./main
./main: error while loading shared libraries: libSList.so.1: cannot open shared object file: No such file or directory

上述问题的解决方案有两种(我能想到的):
1 如果LD_LIBRARY_PATH 没设定:
拷贝libSList.so到目录 /usr/local/lib (同理也可以拷到/usr/lib下)
[root@LEE src]# cp libSList.so /usr/local/lib
[root@LEE src]# ldconfig /usr/local/lib
[root@LEE src]# ./main
4->17->19->10->23->21->11->20

2 如果LD_LIBRARY_PATH 设定:
  编辑.bash_profile文件
  添加LD_LIBRARY_PATH:=$LD_LIBRARY_PATH:/path/to/libSList.so(不包括 libSList.so)
  执行.bash_profile 文件.
[root@LEE src]# . /root/.bash_profile
[root@LEE src]# ./main
4->17->19->10->23->21->11->20

 

  这些天看了一些关于静态/动态链接库的贴子和书,将这几天的研究整理成文,决定发表一下.我很早就注册了CSDN,也很想成为一名软件工程师,这个愿望在不久前实现了,所以更加努力,登陆CSDN的次数也有了数量级的增加.我有过一些Linux下的C++程序开发经历,相对于同专业(数学类)的同学,我有着较好的Linux和C++技能.最近在CSDN上看了很多优秀的文章,从中受益良多.这是我第一次发表文章,希望各位多多指教.