标准的系统库经常放在 /lib 和/usr/lib 下面。
库的名字命名是有规范的
以lib开头,静态库的结尾是.a ,共享库的结尾是.so
位于 /usr/lib/libm.a 就是系统的一个数据函数库
通过下面的命令 -l选项来告诉c编译器函数库的位置 m 是libm.a的缩写
$ gcc -o fred fred.c -lm
如果我们的库没有放在/lib 和/user/lib目录下面,那么我们需要通过-L选择来告诉编译器我们的库的位置
$ gcc -o x11fred -L/usr/openwin/lib x11fred.c -lX11
编写并运行c程序的步骤
1、 编写c源代码
2、将我们写的.c后缀的源文件编译成.o的文件
gcc -c bill.c fred.c
gcc -c program.c
3、将.o的文件链接成可执行的文件
gcc -o program grogram.o bill.o
4、运行我们写好的程序
./program
5 用ar命令来打包我们自己的库函数 ,将bill.o fred.o打成一个包
$ar crv libfoo.a bill.o fred.o
就可以通过连接库的方式来编译我们的程序了
$ gcc -o program program.o libfoo.a
也可以通过下面的方式
$ gcc –o program program.o –L. –lfoo
-L 告诉c编译器我们的库函数查找位置为当前目录,库函数的名字叫libfoo.a
我们有时候想查看我们的程序用到了哪些共享库,我们可以通过ldd(loader dynamic lib)
$ ldd program
对文件操作的API
#include <stdio.h>
FILE *fopen(const char *filename, const char *mode);
mode的取值可以为:
“r”or “rb”: Open for reading only
“w”or “wb”: Open for writing, truncate to zero length
“a”or “ab”: Open for writing, append to end of file
“r+”or “rb+”or “r+b”: Open for update (reading and writing)
“w+”or “wb+”or “w+b”: Open for update, truncate to zero length
“a+”or “ab+”or “a+b”: Open for update, append to end of file
如果打开成功,返回FILE* 类型的指针,如果失败,则返回NULL
#include <stdio.h>
size_t fread(void *ptr, size_t size, size_t nitems, FILE *stream);
这个函数用来从文件流中读取数据到给定的数据缓存区中ptr。fread,fwrite都是处理records,size指定的是一个record的大小。nitems指定的是要读取的记录的数目,
返回的是实际读取到缓存区中记录的数据
#include <stdio.h>
size_t fwrite (const void *ptr, size_t size, size_t nitems, FILE *stream);
和fread的语法是类似的,只不过他是将缓存区中的数据写到文件流中,
#include <stdio.h>
int fclose(FILE *stream);
用fclose函数是为了确保文件流中的数据已经写完了。
#include <stdio.h>
int fflush(FILE *stream);
调用fclose的时候就显示的调用fflush函数
#include <stdio.h>
int fseek(FILE *stream, long int offset, int whence);
#include <stdio.h>
int fgetc(FILE *stream);
int getc(FILE *stream);
int getchar();
getchar是和getc(stdin)等价的,从标准的输入设备中读取一个字符
#include <stdio.h>
int fputc(int c, FILE *stream);
int putc(int c, FILE *stream);
int putchar(int c);
上面的都是读取一共字符,下面的是读取一个字符串
#include <stdio.h>
char *fgets(char *s, int n, FILE *stream);
char *gets(char *s);
#include <stdio.h>
int printf(const char *format, ...);
int sprintf(char *s, const char *format, ...);
int fprintf(FILE *stream, const char *format, ...);
printf 是输出道标准的输入设备,fprintf是输出到指定的文件流,而sprintf输出到一个字符串 s。
#include <stdio.h>
int scanf(const char *format, ...);
int fscanf(FILE *stream, const char *format, ...);
int sscanf(const char *s, const char *format, ...);
fgetpos: Get the current position in a file stream.
fsetpos: Set the current position in a file stream.
ftell: Return the current file offset in a stream.
rewind: Reset the file position in a stream.
freopen: Reuse a file stream.
简单的一共文件复制的例子
#include<stdio.h>
#include<stdlib.h>
int main()
{
int c;
FILE *in,*out;
in = fopen("file.in","r");
out = fwrite("file.out","w");
//一次读取一个字符
while((c=fgetc(in))!=EOF)
{
fputc(c,out);
}
//正常退出
exit(0);
}
#include <errno.h>
extern int errno;
当函数执行失败的时候,errno才是有效的。我们需要保存这个值到一耳光变量中取,因为其他的任何函数都可以改变这个errno。
#include <stdio.h>
int ferror(FILE *stream);
int feof(FILE *stream);
void clearerr(FILE *stream);
ferror函数用来测试文件流是否出现错误,如果是就返回非0,否则返回0
feof 函数用来测试文件是否到了末尾,如果是就返回非0,否则返回0
if(feof(stream))
{
printf("at the end of file");
}
#include <stdio.h>
int fileno(FILE *stream);
FILE *fdopen(int fildes, const char *mode);
对文件和目录的维护
chmod 改变文件的读,写权限
#include <sys/stat.h>
int chmod(const char *path, mode_t mode);
chown 改变一个文件的所有者,所属组
#include <sys/types.h>
#include <unistd.h>
int chown(const char *path, uid_t owner, gid_t group);
建立目录
#include <sys/types.h>
#include <sys/stat.h>
int mkdir(const char *path, mode_t mode);
删除目录
#include <unistd.h>
int rmdir(const char *path);
改变目录
#include <unistd.h>
int chdir(const char *path);
获取当前的目录名到buf中
#include <unistd.h>
char *getcwd(char *buf, size_t size);
打开目录
#include <sys/types.h>
#include <dirent.h>
DIR *opendir(const char *name);
读取目录中的内容
#include <sys/types.h>
#include <dirent.h>
struct dirent *readdir(DIR *dirp);
dirent包含了 ino_t d_ino: 文件的i节点值,char d_name[] 文件的名字
#include <sys/types.h>
#include <dirent.h>
long int telldir(DIR *dirp);
telldir返回的是记录在文件流中的当前位置的值。
设置目录实体指针在文件流(dirp)中的位置,loc用来设置位置
#include <sys/types.h>
#include <dirent.h>
void seekdir(DIR *dirp, long int loc);
#include <sys/types.h>
#include <dirent.h>
int closedir(DIR *dirp);
下面是一个目录扫描的程序
#include <unistd.h>
#include <stdio.h>
#include <dirent.h>
#include <string.h>
#include <sys/stat.h>
#include <stdlib.h>
void printdir(char *dir, int depth)
{
DIR *dp;
struct dirent *entry;
struct stat statbuf;
if((dp = opendir(dir)) == NULL) {
fprintf(stderr,”cannot open directory: %s\n”, dir);
}
chdir(dir);
while((entry = readdir(dp)) != NULL) {
lstat(entry->d_name,&statbuf);
if(S_ISDIR(statbuf.st_mode)) {
/* Found a directory, but ignore . and .. */
if(strcmp(“.”,entry->d_name) == 0 ||
strcmp(“..”,entry->d_name) == 0)
continue;
printf(“%*s%s/\n”,depth,”“,entry->d_name);
/* Recurse at a new indent level */
printdir(entry->d_name,depth+4);
}
else printf(“%*s%s\n”,depth,”“,entry->d_name);
}
chdir(“..”);
closedir(dp);
}
int main()
{
printf(“Directory scan of /home:\n”);
printdir(“/home”,0);
printf(“done.\n”);
exit(0);
}
#include <unistd.h>
int getopt(int argc, char *const argv[], const char *optstring);
extern char *optarg;
extern int optind, opterr, optopt;
用来解析程序传进来的参数和可选项
1、如果可选项是带有一个值,则这个值放到了optarg这个变量中
2、如果return -1 ,表明没有可选项要处理了,这里有个特殊的选项那就是--,它就会让
getopt停止扫描可选项
3、如果可选择输入了不是我们定义的-i,-j 则会返回?
4、如果可选项要求需要值,而调用的时候没有传递值,则会返回?:,这个值可以通过传替一个字符替换。下面的“:if:lr”中的第一个冒号就是用来替换的。
optwin是记录下一个要 处理参数的索引值
下面是一个具体的例子
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(int argc, char *argv[])
{
int opt;
while((opt = getopt(argc, argv, “:if:lr”)) != -1) {
switch(opt) {
case ‘i’:
case ‘l’:
case ‘r’:
printf(“option: %c\n”, opt);
break;
case ‘f’:
printf(“filename: %s\n”, optarg);
break;
case ‘:’:
printf(“option needs a value\n”);
break;
case ‘?’:
printf(“unknown option: %c\n”, optopt);
break;
}
}
for(; optind < argc; optind++)
printf(“argument: %s\n”, argv[optind]);
exit(0);
}
设置环境变量
#include <stdlib.h>
char *getenv(const char *name);
int putenv(const char *string);
下面是一个利用getenv ,putenv的例子程序
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
int main(int argc, char *argv[])
{
char *var, *value;
if(argc == 1 || argc > 3) {
fprintf(stderr,”usage: environ var [value]\n”);
exit(1);
}
var = argv[1];
value = getenv(var);
if(value)
printf(“Variable %s has value %s\n”, var, value);
else
printf(“Variable %s has no value\n”, var);
if(argc == 3) {
char *string;
value = argv[2];
string = malloc(strlen(var)+strlen(value)+2);
if(!string) {
fprintf(stderr,”out of memory\n”);
exit(1);
}
strcpy(string,var);
strcat(string,”=”);
strcat(string,value);
printf(“Calling putenv with: %s\n”,string);
if(putenv(string) != 0) {
fprintf(stderr,”putenv failed\n”);
free(string);
exit(1);
}
value = getenv(var);
if(value)
printf(“New value of %s is %s\n”, var, value);
else
printf(“New value of %s is null??\n”, var);
}
exit(0);
}
操作时间的函数
#include <time.h>
struct tm *gmtime(const time_t timeval);
这个函数返回的时间是UTC时间,为了同步世界时间
tm Member Description
int tm_sec Seconds, 0-61
int tm_min Minutes, 0-59
int tm_hour Hours, 0-23
int tm_mday Day in the month, 1-31
int tm_mon Month in the year, 0-11 (January = 0)
int tm_year Years since 1900
int tm_wday Day in the week, 0-6 (Sunday = 0)
int tm_yday Day in the year, 0-365
#include <time.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
struct tm *tm_ptr;
time_t the_time;
(void) time(&the_time);
tm_ptr = gmtime(&the_time);
printf(“Raw time is %ld\n”, the_time);
printf(“gmtime gives:\n”);
printf(“date: %02d/%02d/%02d\n”,
tm_ptr->tm_year, tm_ptr->tm_mon+1, tm_ptr->tm_mday);
printf(“time: %02d:%02d:%02d\n”,
tm_ptr->tm_hour, tm_ptr->tm_min, tm_ptr->tm_sec);
exit(0);
}
下面是获取本地的时间
#include <time.h>
struct tm *localtime(const time_t *timeval);
将日期结构体转换成时间戳
#include <time.h>
time_t mktime(struct tm *timeptr);
格式化日期成size_t类型
#include <time.h>
size_t strftime(char *s, size_t maxsize, const char *format, struct tm *timeptr);
格式化日期成一个字符串
#include <time.h>
char *strptime(const char *buf, const char *format, struct tm *timeptr);
常用的时间格式 format
“%a %b %d %H:%M:%S %Y”
下面是日期转成字符串的例子
#include <time.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
struct tm *tm_ptr, timestruct;
time_t the_time;
char buf[256];
char *result;
(void) time(&the_time);
tm_ptr = localtime(&the_time);
strftime(buf, 256, “%A %d %B, %I:%S %p”, tm_ptr);
printf(“strftime gives: %s\n”, buf);
strcpy(buf,”Thu 26 July 2007, 17:53 will do fine”);
printf(“calling strptime with: %s\n”, buf);
tm_ptr = ×truct;
result = strptime(buf,”%a %d %b %Y, %R”, tm_ptr);
printf(“strptime consumed up to: %s\n”, result);
printf(“strptime gives:\n”);
printf(“date: %02d/%02d/%02d\n”,
tm_ptr->tm_year % 100, tm_ptr->tm_mon+1, tm_ptr->tm_mday);
printf(“time: %02d:%02d\n”,
tm_ptr->tm_hour, tm_ptr->tm_min);
exit(0);
}
getuid获取用户的id
#include <sys/types.h>
#include <unistd.h>
uid_t getuid(void);
char *getlogin(void);
获取密码的详细信息
#include <sys/types.h>
#include <pwd.h>
struct passwd *getpwuid(uid_t uid);
struct passwd *getpwnam(const char *name);
char *pw_name The user’s login name
uid_t pw_uid The UID number
gid_t pw_gid The GID number
char *pw_dir The user’s home directory
char *pw_gecos The user’s full name
char *pw_shell The user’s default shell
下面是一个获取当前登录用户信息的程序
#include <sys/types.h>
#include <pwd.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
uid_t uid;
gid_t gid;
struct passwd *pw;
uid = getuid();
gid = getgid();
printf(“User is %s\n”, getlogin());
printf(“User IDs: uid=%d, gid=%d\n”, uid, gid);
pw = getpwuid(uid);
printf(“UID passwd entry:\n name=%s, uid=%d, gid=%d, home=%s, shell=%s\n”,
pw->pw_name, pw->pw_uid, pw->pw_gid, pw->pw_dir, pw->pw_shell);
pw = getpwnam(“root”);
printf(“root passwd entry:\n”);
printf(“name=%s, uid=%d, gid=%d, home=%s, shell=%s\n”,
pw->pw_name, pw->pw_uid, pw->pw_gid, pw->pw_dir, pw->pw_shell);
exit(0);
}
查看系统日志信息
#include <syslog.h>
void syslog(int priority, const char *message, arguments...);
Priority Level Description
LOG_EMERG An emergency situation
LOG_ALERT High-priority problem, such as database corruption
LOG_CRIT Critical error, such as hardware failure
LOG_ERR Errors
LOG_WARNING Warning
LOG_NOTICE Special conditions requiring attention
LOG_INFO Informational messages
LOG_DEBUG Debug messages
#include <syslog.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
FILE *f;
f = fopen(“not_here”,”r”);
if(!f)
syslog(LOG_ERR|LOG_USER,”oops - %m\n”);
exit(0);
}
宁外一个增强的日记操作函数
#include <syslog.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
int logmask;
openlog(“logmask”, LOG_PID|LOG_CONS, LOG_USER);
syslog(LOG_INFO,”informative message, pid = %d”, getpid());
syslog(LOG_DEBUG,”debug message, should appear”);
logmask = setlogmask(LOG_UPTO(LOG_NOTICE));
syslog(LOG_DEBUG,”debug message, should not appear”);
exit(0);
}
mysql 接口
设置mysql的日志存放位置,数据库的引擎,字符编码
vi /etc/my.cnf
设置密码
1、mysqladmin -u root password newpassword
这种方式是不安全的,因为其他的人可以通过ps命令查看,也可以通过history
2、进入mysql设置密码
>mysql -u root
>user mysql;
>SET password=PASSWORD('root');
删除安装时候mysql自动创建的用户
1、首先查看当前的用户
SELECT user, host, password FROM user;
2、删除没有用到的用户
DELETE FROM user WHERE user != ‘root’;
DELETE FROM user WHERE host != ‘localhost’;
让rick只可以通过localhost登录
GRANT ALL ON *.* TO rick@localhost IDENTIFIED BY ‘secretpassword';
让rick可以通过子网掩码为192.168.0.0 网络号为255.255.0.0的登录
GRANT ALL ON *.* TO rick@'192.168.0.0/255.255.255.0' IDENTIFIED BY '123456';
让nick可以通过wiley.com访问
GRANT ALL ON *.* TO rick@'%.wiley.com' IDENTIFIED BY '123456';
mysql
mysql的命令行工具
连接mysql服务器
mysql -u rick -p foo
以非交互模式执行sql命令,通过从文件中读取命令,foo是数据库
$mysql -u rick --password=secretpassword foo < sqlcommands.sql
mysqladmin
可以通过
$mysqladmin --help 来查看mysqladmin的一些命令信息
mysqlbug
给开发者和管理员调试mysql服务器的一种工具
$mysqlbug --help
mysqldump
用来导入或者导出mysql数据
$mysqldump --help
mysqldump -u rick -p myplaydb > myplaydb.dump
mysqlimport
用来一次导入大量的数据到某个表中
grant <privilege> on <object> to <user> [identified by user-password] [with grant option];
alter Alter tables and indexes.
create | Create databases and tables.
delete | Delete data from the database.
drop | Remove databases and tables.
index | Manage indexes.
insert | Add data to the database.
lock tables | Allow locking of tables.
select | Retrieve data.
update | Modify data.
all All| the above.
object的格式 :databasename.tablename
foo.* 表示foo数据库下面所有的表
收回权限
revoke <a_privilege> on <an_object> from <a_user>
删除用户
mysql> use mysql
mysql> DELETE FROM user WHERE user = “rick”
mysql> FLUSH PRIVILEGES;
查看mysql安装的位置
先用rpm -qa \*mysql\*看看你安装了哪些包,然后用 rpm -ql xxxxxxx 去查看具体的包安装位置
#include <mysql.h>
MYSQL *mysql_init(MYSQL *);
在这个是用来初始化mysql连接,通常参数传递的是NULL
返回值给下面的mysql_real_connect(MYSQL *conn)
MYSQL *mysql_real_connect(MYSQL *connection,
const char *server_host,
const char *sql_user_name,
const char *sql_password,
const char *db_name,
unsigned int port_number,
const char *unix_socket_name,
unsigned int flags);
这里需要说明的是后面3个参数的取值
一般设置成 0,NULL,0 如果你的端口是用的mysql默认的端口3306
关闭mysql连接,释放占用的系统资源
void mysql_close(MYSQL *connection);
int mysql_options(MYSQL *connection, enum option_to_set,
const char *argument);
MYSQL_OPT_CONNECT_TIMEOUT const unsigned int * The number of seconds to wait
before timing out a connection.
MYSQL_OPT_COMPRESS None, use NULL Use compression on the network connection.
MYSQL_INIT_COMMAND const char * Command to send each time a connection is established
错误处理
1、下面的报的是客户端的常见错误
unsigned int mysql_errno(MYSQL *connection);
2、返回的是与服务器规范定义的相关错误
char *mysql_error(MYSQL *connection);
执行查询
int mysql_query(MYSQL *connection, const char *query)
处理返回的结果
my_ulonglong mysql_affected_rows(MYSQL *connection);
返回的上面UPDATE,INSERT,DELETE query影响的记录数
#include <stdlib.h>
#include "mysql.h"
int main(int argc,char * argv[])
{
MYSQL *conn;
int result;
conn = mysql_init(NULL);
conn = mysql_real_connect(conn,"localhost","rick","secret","foo",0,NULL,0);
if(conn){
printf("connection success\n");
/*执行对数据库的CRUD*/;
char * sql = "insert into children(fname,age) values('anna',33)";
result = mysql_query(conn,sql);
if(!result) {
/*非0说明操作成功,打印出影响的记录数据*/
printf("insert %lu rows\n",(unsigned long)mysql_affected_rows(conn));
mysql_close(conn);
}
else{
printf("connection fail,%d,%s",mysql_errno(conn),mysql_errno(conn));
}
return EXIT_SUCCESS;
}
mysql接口
LAST_INSERT_ID()
获取最进插进去的id
SELECT LAST_INSERT_ID();
Just as you did with the INSERTand DELETEstatements, you’ll use mysql_queryto send the SQL. Next you’ll retrieve the data using either mysql_store_resultor mysql_use_result, depending on how you want the data retrieved. Next you’ll use a sequence of mysql_fetch_rowcalls to process the data. Finally,you’ll use mysql_free_resultto clear the memory you used for your query.
The difference between mysql_use_resultand mysql_store_resultbasically amounts to whether you want to get your data back a row at a time, or get the whole result set in one go. The latter is more appropriate in circumstances where you anticipate a smaller result set.
存储所有返回给客户端的数据
MYSQL_RES *mysql_store_result(MYSQL *connection);
获取返回的记录数
long mysql_num_rows(MYSQL_RES *result);
获取某一行数据
MYSQL_ROW mysql_fetch_row(MYSQL_RES *result);
让行记录的指针跳转到指定的offset位置,offset在0-num_record ,不包括num_record
void mysql_data_seek(MYSQL_RES *result, my_ulonglong offset);
告诉当前行记录指针的位置
MYSQL_ROW_OFFSET mysql_row_tell(MYSQL_RES *result);
移动当前记录指针的位置,同事返回上次记录指针的位置
MYSQL_ROW_OFFSET mysql_row_seek(MYSQL_RES *result, MYSQL_ROW_OFFSET offset);
检索一行一行的数据
MYSQL_RES *mysql_use_result(MYSQL *connection);
获取具体某个字段的值
MYSQL_FIELD *mysql_fetch_field(MYSQL_RES *result);
unsigned int mysql_field_count(MYSQL *connection);
释放资源
void mysql_free_result(MYSQL_RES *result);
int mysql_select_db(MYSQL *connection, const char *dbname);
int mysql_shutdown(MYSQL *connection, enum mysql_enum_shutdown_level);
下面是一个完整的例子
#include <stdlib.h>
#include <stdio.h>
#include “mysql.h”
MYSQL my_connection;
MYSQL_RES *res_ptr;
MYSQL_ROW sqlrow;
void display_header() {
MYSQL_FIELD *field_ptr;
printf(“Column details:\n”);
while ((field_ptr = mysql_fetch_field(res_ptr)) != NULL) {
printf(“\t Name: %s\n”, field_ptr->name);
printf(“\t Type: “);
if (IS_NUM(field_ptr->type)) {
printf(“Numeric field\n”);
} else {
switch(field_ptr->type) {
case FIELD_TYPE_VAR_STRING:
printf(“VARCHAR\n”);
break;
case FIELD_TYPE_LONG:
printf(“LONG\n”);
break;
default:
printf(“Type is %d, check in mysql_com.h\n”, field_ptr->type);
} /* switch */
} /* else */
printf(“\t Max width %ld\n”, field_ptr->length);
if (field_ptr->flags & AUTO_INCREMENT_FLAG)
printf(“\t Auto increments\n”);
printf(“\n”);
} /* while */
}
void display_row() {
unsigned int field_count;
field_count = 0;
while (field_count < mysql_field_count(&my_connection)) {
if (sqlrow[field_count])
printf(“%s “, sqlrow[field_count]);
else
printf(“NULL”);
field_count++;
}
printf(“\n”);
}
int main(int argc, char *argv[]) {
int res;
mysql_init(&my_connection);
if (mysql_real_connect(&my_connection, “localhost”, “rick”,
“secret”, “foo”, 0, NULL, 0)) {
printf(“Connection success\n”);
res = mysql_query(&my_connection, “SELECT childno, fname,
age FROM children WHERE age > 5”);
if (res) {
printf(“SELECT error: %s\n”, mysql_error(&my_connection));
} else {
res_ptr = mysql_store_result(&my_connection);
if (res_ptr) {
printf(“Retrieved %lu rows\n”, (unsigned long)mysql_num_rows(res_ptr));
while ((sqlrow = mysql_fetch_row(res_ptr))) {
printf(“Fetched data...\n”);
}
if (mysql_errno(&my_connection)) {
fprintf(stderr, “Retrive error: %s\n”, mysql_error(&my_connection));
}
mysql_free_result(res_ptr);
}
}
mysql_close(&my_connection);
} else {
fprintf(stderr, “Connection failed\n”);
if (mysql_errno(&my_connection)) {
fprintf(stderr, “Connection error %d: %s\n”,
mysql_errno(&my_connection), mysql_error(&my_connection));
}
}
return EXIT_SUCCESS;
}
Make 语法
target: fil1.c file2.o
gcc -o myapp file2.o
1、Target
target是自己定义的一些任务,其中如果make指定target,则会自动找第一个target
一般的程度都定义一个all的target,这样当用户没有指定target的时候就会自动执行我们定义的target。
all: myapp myapp.1
2、Dependencies
依赖指定他们在应用中各源文件的关系。
myapp: main.o 2.o 3.o
main.o: main.c a.h
2.o: 2.c a.h b.h
3.o: 3.c b.h c.h
上面的一个例子就是 myapp的target 依赖main.o 2.o 3.o这3个target,当这3个中任何一个更新了,那么myapp会自动重新编译,这是通过make自身的机制实现的,这也是我们为什么使用make的原因
3、Rules
规则制定了怎么样去创建target
规则一般是由一些的命令组成的
myapp: main.o 2.o 3.o
gcc -o myapp main.o 2.o 3.o
main.o: main.c a.h
gcc -c main.c
2.o: 2.c a.h b.h
gcc -c 2.c
3.o: 3.c b.h c.h
gcc -c 3.c
make文件中的注释用 #表示
Macros in a Makefile
# Which compiler
CC = gcc
宏是通过键值对的形式
引用改变量的时候可以通过$CC,${CC},$(CC),3种不同的形式,根据linux支持版本不一样
内置的宏
Macro Definition
$? List of prerequisites (files the target depends on) changed more recently
than the current target
$@ Name of the current target
$< Name of the current prerequisite
$* Name of the current prerequisite, without any suffix
- 横线表示不管出现的错误,程序继续执行
@ 让make在执行命令之前不要打印命令的到标准的输出设备
可以使用make -p 来查看当前系统内置的规则
版本控制
通过rcs命令来初始化RCS对文件的控制
$ rcs -i important.c
通过ci 来查看当前的版本
$ ci important.c
check out 文件
$ co -l important.c
获取1.1那个版本文件
$ co -r1.1 important.c
check in 文件
$ ci important.c
$ ci -r2 important.c
查看文件的摘要信息
$ rlog important.c
比较2个文件的不同
$ rcsdiff -r1.1 -r1.2 important.c
CVS Subversion
cvs –d /usr/local/repository init svnadmin create /usr/local/repository
cvs import wrox/chap9-cvs svn import cvs-sp file:///usr/local/repository/trunk
cvs checkout wrox/chap9-cvs svn checkout file:///usr/local/repository/trunk cvs-sp
cvs diff svn diff
cvs rdiff svn diff tag1 tag2
cvs update svn status -u
cvs commit svn commit