Makefile学习(二)----生成静态库文件
Lunix下编译静态库文件:
.o后缀文件:编译生成的中间代码文件;
.a后缀文件:静态库文件,编译的时候会合到可执行程序中,文件比较大;
.so后缀文件:动态库文件,只是一个指向,不会合到可执行程序中,当要调用函数库的时候才使用;
用一个简单的例子阐述下在lunix环境下生成静态库文件的过程:
一. 源码文件:
so_test.h:
void test_a();
void test_b();
void test_c();
test_a.c:
#include "so_test.h"
void test_a()
{
printf("this is in test_a...\n");
}
test_b.c:
#include "so_test.h"
void test_a()
{
printf("this is in test_b...\n");
}
test_c.c:
#include "so_test.h"
void test_a()
{
printf("this is in test_c...\n");
}
test.c:
#include "so_test.h"
int main()
{
test_a();
test_b();
test_c();
return 0;
}
二. makefile 文件:
makefile:
SHELL = /bin/sh
LIB_DIR = /cbs/lhbb/hcj/dy/lib/
BIN_DIR = /cbs/lhbb/hcj/dy/bin/
OBJECT_DIR = /cbs/lhbb/dy/test/obj/
APP_DIR = /cbs/lhbb/hcj/dy/testapp/
$(shell mkdir -p ${LIB_DIR})
$(shell mkdir -p ${BIN_DIR})
$(shell mkdir -p ${OBJECT_DIR})
RM = rm -fr
#****************************************************************************
CC = gcc
AR = ar rc
SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c
H_OBJECT = $(APP_DIR)so_test.h
OBJECT = test_a.o test_b.o test_c.o
DY_SRC_OBJECT = $(APP_DIR)test.c
DY_OBJECT=test.o
LIB_OBJECT = libtest.a
BIN_OBJECT = test
#****************************************************************************
.PHONY:all
all:$(LIB_OBJECT) $(BIN_OBJECT)
$(LIB_OBJECT):$(OBJECT)
$(AR) $(LIB_OBJECT) $(OBJECT)
mv $(LIB_OBJECT) $(LIB_DIR)
$(OBJECT):$(SRC_OBJECT) $(H_OBJECT)
$(CC) -c $(SRC_OBJECT)
$(BIN_OBJECT):$(DY_OBJECT)
$(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)
mv $(BIN_OBJECT) $(BIN_DIR)
$(DY_OBJECT):$(DY_SRC_OBJECT)
$(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)
mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)
clean:
$(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)
三.对makefile文件的简单阐述:
1. 路径变量赋值:
LIB_DIR = /cbs/lhbb/hcj/dy/lib/ -----存放生成的静态库文件,.a文件;
BIN_DIR = /cbs/lhbb/hcj/dy/bin/ -----存放生成的可执行文件;
OBJECT_DIR = /cbs/lhbb/dy/test/obj/ -----存放生成的中间代码文件,.o文件;
APP_DIR = /cbs/lhbb/hcj/dy/testapp/ -----存放源文件;
2.调用shell脚本,创建目录:
$(shell mkdir -p ${LIB_DIR})
$(shell mkdir -p ${BIN_DIR})
$(shell mkdir -p ${OBJECT_DIR})
3.编译变量赋值:
RM = rm –fr
CC = gcc
AR = ar rc -----生成静态库文件命令
4.文件变量赋值:
SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c
----源文件变量;
H_OBJECT = $(APP_DIR)so_test.h
----头文件变量;
OBJECT = test_a.o test_b.o test_c.o
----中间代码文件变量;
DY_SRC_OBJECT = $(APP_DIR)test.c
----生成test可执行文件所依赖的源文件变量;
DY_OBJECT=test.o
----编译test.c生成的中间代码文件变量;
LIB_OBJECT = libtest.a
----静态库文件变量;
BIN_OBJECT = test
----生成的可执行文件变量;
这个makefile中基本都是使用的变量,这样编译不同的工程的时候,就只要修改变量的值,编译的整体框架就不需要修改,当然,这个框架只是本人刚开始研究makefile写的,不够好,望大神吐槽,谢谢!
5.编译主体阐述:
.PHONY:all
----固定用.PHONY定义一个伪目标,伪目标不是文件,只是一个标签,没有依赖文件;这边定义了一个伪目标all;
all:$(LIB_OBJECT) $(BIN_OBJECT)
----目标all包含了两个需要生成的目标文件,一个是需要生成的静态库文件libtest.a,一个是最终生成的可执行文件test;但是为什么这边需要用一个all来包含两个目标文件呢?原因是通常情况下,系统只会去编译生成makefile文件开头最早一个目标文件,下面的目标文件是不会去执行编译的,所以我在这边将两个目标文件写在了一个伪目标all中,这样就可以实现生成两个目标文件;
生成静态库文件的过程:
$(LIB_OBJECT):$(OBJECT) -----目标文件:依赖文件 libtest.a:test_a.o test_b.o test_c.o
$(AR) $(LIB_OBJECT) $(OBJECT) ----编译生成静态库文件命令,特点就是需要使用
ar rc命令去编译;
mv $(LIB_OBJECT) $(LIB_DIR) ----将生成的文件移动到对应的文件夹;
$(OBJECT):$(SRC_OBJECT) $(H_OBJECT)
$(CC) -c $(SRC_OBJECT)
生成可执行文件test的过程:
$(BIN_OBJECT):$(DY_OBJECT)
$(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)
-----将静态库文件包含在命令中;
mv $(BIN_OBJECT) $(BIN_DIR)
$(DY_OBJECT):$(DY_SRC_OBJECT)
$(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)
mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)
执行make clean删除生成的文件:
clean:
$(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)
四.注意点:
1.命令行必须以tab键打头,否则会报错;
2.生成静态库文件用命令:ar rc;
3.库文件以lib打头,.a为后缀;
五.执行结果:
bin文件夹中存放可执行文件test,运行./test结果如下:
this is in test_a...
this is in test_b...
this is in test_c...
lib文件夹下存放生成的静态库文件libtest.a;
Makefile学习(二)----生成静态库文件的更多相关文章
- Android NDK入门实例 计算斐波那契数列二生成.so库文件
上一篇文章输生成了jni头文件,里面包含了本地C代码的信息,提供我们引用的C头文件.下面实现本地代码,再用ndk-build编译生成.so库文件.由于编译时要用到make和gcc,这里很多人是通过安装 ...
- IOS 生成静态库文件(.framework)
http://blog.csdn.net/zwl492454828/article/details/55095422
- IOS 生成静态库文件(.a文件)
http://www.cnblogs.com/lyy-5518/p/5459643.html
- linux 静态库文件
1.生成目标文件 gcc -o mylib.o -c mylib.c 2.生成静态库文件 ar rcs libmylib.a mylib.o 查看库信息: nm libmylib.a //====== ...
- Ubuntu下通过makefile生成静态库和动态库简单实例
本文转自http://blog.csdn.net/fengbingchun/article/details/17994489 Ubuntu环境:14.04 首先创建一个test_makefile_gc ...
- cmake利用toolchain.cmake生成makefile之后,make生成静态库失败问题
问题描述 利用toolchian.cmake设置好编译器后,利用make指令生成静态库,出现以下问题 Error running link command: No such file or direc ...
- Makefile 编译静态库文件及链接静态库
本文为原创文章,转载需指明该文链接 1.代码目录结构如下: comm/ comm/inc/apue.h 3 atexit.c Makefile 5 staticlib/lib/ staticlib ...
- 生成静态库.a文件和动态库.so文件
转载来源:https://www.cnblogs.com/hookjc/ 静态库 在linux环境中, 使用ar命令创建静态库文件.如下是命令的选项: d -----从指定的静态库文件中删除文件 m ...
- 在Linux下如何使用GCC编译程序、简单生成 静态库及动态库
最近在编写的一个Apache kafka 的C/C++客户端,,在看他写的 example中,他的编译是用librdkafka++.a和librdkafka.a 静态库编译的,,,而我们这 ...
随机推荐
- AKOJ-1265-输出二叉树
链接:https://oj.ahstu.cc/JudgeOnline/problem.php?id=1265 题意: 我们知道二叉树的先序序列和中序序列或者是中序和后序能够唯一确定一颗二叉树.现在给一 ...
- 利用Common-Fileupload上传文件图片
一,介绍 common-fileupload是appache的开源组件,基于该组件可以轻松实现文件上传的功能,strust框架的文件上传功能也是基于该组件. 二,使用 1,导入两个jar包:commo ...
- js操作表格
js 操作table: insertRow(),deleteRow(),insertCell(),deleteCell()方法 表格有几行: var trCnt = table.rows.length ...
- 119 Pascal's Triangle II 帕斯卡三角形 II Pascal's Triangle II
给定一个索引 k,返回帕斯卡三角形(杨辉三角)的第 k 行.例如,给定 k = 3,则返回 [1, 3, 3, 1].注:你可以优化你的算法到 O(k) 的空间复杂度吗?详见:https://leet ...
- C. Divide by Three DP
http://codeforces.com/contest/792/problem/C 这题奇葩题我居然用dp过了. 如果要模拟的话,可以用一个栈保存,保存每一个%3 = 2的pos,%3 = 1的p ...
- springmvc 实现原理与struts2原理的区别
spring mvc的入口是servlet,而struts2是filter,这样就导致了二者的机制不同. spring mvc是基于方法的设计,sturts2是基于类设计的. springmvc将ur ...
- Ice-cream Tycoon9(线段树)
线段树的一些基本应用,就是函数写了很多,有点繁琐. 以每个物品的单价建树,刚开始写了个裸的想水过去直接MLE了,然后又离散化了下. 离散化单价后建树,lz数组用来清零,s数组保存结点所含物品个数,co ...
- 导入动态Web项目到Eclipse中遇到的问题
问题一:创建动态网页项目时,项目报错而无文件报错 当时解决方法:直接右击项目->properties->project facets将jdk改为1.8版本即可 如图: 问题二:Tomcat ...
- Java GUI 事件监听
现在使用的仍是AWT的事件模型.涉及到3类对象: Event Source:事件源,即事件发生所在的组件 Event:事件,封装了此次事件的相关信息 Event Listener:事件监听器,监听事件 ...
- Eclipse Mars.2集成Maven 3.5.4
准备材料: Eclipse Mars.2 Release (4.5.2): 官网戳:https://www.eclipse.org/downloads/ Maven 3.5.4: http://ma ...