前言
在學(xué)習(xí)了《Linux內(nèi)核設(shè)計(jì)與實(shí)現(xiàn)》之后咒彤,對(duì)于Linux的內(nèi)核的原理有了一定認(rèn)識(shí),尤其是進(jìn)程管理咒精,地址空間等知識(shí)镶柱。
由于本人并不想從事Linux內(nèi)核的開(kāi)發(fā),只是想在Linux系統(tǒng)上進(jìn)行開(kāi)發(fā)模叙,因此從知乎歇拆,以及stackoverflow上面找到這本Unix環(huán)境高級(jí)編程(APUE)。簡(jiǎn)單翻了一下APUE這本書(shū)范咨,它在簡(jiǎn)單介紹Linux工作模式的同時(shí)故觅,通過(guò)實(shí)例演示了如何調(diào)用Linux接口。由于知識(shí)容易遺忘渠啊,同時(shí)看了網(wǎng)上的一些blog要么排版不太友好输吏,要么內(nèi)容深度不符合我這種菜鳥(niǎo)水平,因此通過(guò)這個(gè)系列文章總結(jié)自己的學(xué)習(xí)體會(huì)替蛉。
1.學(xué)習(xí)前準(zhǔn)備
APUE前兩章的內(nèi)容主要是對(duì)Unix的版本贯溅,發(fā)展歷程,已經(jīng)基本知識(shí)進(jìn)行綜述躲查。因?yàn)楹竺嬲鹿?jié)都涉及到具體實(shí)例它浅,每個(gè)實(shí)例都包含一個(gè)重要的頭文件apue.h
。apue.h
是作者常用的一些頭文件熙含,函數(shù)以及宏定義罚缕,可以從網(wǎng)址http://www.apuebook.com/下到,里面的內(nèi)容如下:
/*
* Our own header, to be included before all standard system headers.
*/
#ifndef _APUE_H
#define _APUE_H
#define _POSIX_C_SOURCE 200809L
#if defined(SOLARIS) /* Solaris 10 */
#define _XOPEN_SOURCE 600
#else
#define _XOPEN_SOURCE 700
#endif
#include <sys/types.h> /* some systems still require this */
#include <sys/stat.h>
#include <sys/termios.h> /* for winsize */
#if defined(MACOS) || !defined(TIOCGWINSZ)
#include <sys/ioctl.h>
#endif
#include <stdio.h> /* for convenience */
#include <stdlib.h> /* for convenience */
#include <stddef.h> /* for offsetof */
#include <string.h> /* for convenience */
#include <unistd.h> /* for convenience */
#include <signal.h> /* for SIG_ERR */
#define MAXLINE 4096 /* max line length */
/*
* Default file access permissions for new files.
*/
#define FILE_MODE (S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)
/*
* Default permissions for new directories.
*/
#define DIR_MODE (FILE_MODE | S_IXUSR | S_IXGRP | S_IXOTH)
typedef void Sigfunc(int); /* for signal handlers */
#define min(a,b) ((a) < (b) ? (a) : (b))
#define max(a,b) ((a) > (b) ? (a) : (b))
/*
* Prototypes for our own functions.
*/
char *path_alloc(size_t *); /* {Prog pathalloc} */
long open_max(void); /* {Prog openmax} */
int set_cloexec(int); /* {Prog setfd} */
void clr_fl(int, int);
void set_fl(int, int); /* {Prog setfl} */
void pr_exit(int); /* {Prog prexit} */
void pr_mask(const char *); /* {Prog prmask} */
Sigfunc *signal_intr(int, Sigfunc *); /* {Prog signal_intr_function} */
void daemonize(const char *); /* {Prog daemoninit} */
void sleep_us(unsigned int); /* {Ex sleepus} */
ssize_t readn(int, void *, size_t); /* {Prog readn_writen} */
ssize_t writen(int, const void *, size_t); /* {Prog readn_writen} */
int fd_pipe(int *); /* {Prog sock_fdpipe} */
int recv_fd(int, ssize_t (*func)(int,
const void *, size_t)); /* {Prog recvfd_sockets} */
int send_fd(int, int); /* {Prog sendfd_sockets} */
int send_err(int, int,
const char *); /* {Prog senderr} */
int serv_listen(const char *); /* {Prog servlisten_sockets} */
int serv_accept(int, uid_t *); /* {Prog servaccept_sockets} */
int cli_conn(const char *); /* {Prog cliconn_sockets} */
int buf_args(char *, int (*func)(int,
char **)); /* {Prog bufargs} */
int tty_cbreak(int); /* {Prog raw} */
int tty_raw(int); /* {Prog raw} */
int tty_reset(int); /* {Prog raw} */
void tty_atexit(void); /* {Prog raw} */
struct termios *tty_termios(void); /* {Prog raw} */
int ptym_open(char *, int); /* {Prog ptyopen} */
int ptys_open(char *); /* {Prog ptyopen} */
#ifdef TIOCGWINSZ
pid_t pty_fork(int *, char *, int, const struct termios *,
const struct winsize *); /* {Prog ptyfork} */
#endif
int lock_reg(int, int, int, off_t, int, off_t); /* {Prog lockreg} */
#define read_lock(fd, offset, whence, len) \
lock_reg((fd), F_SETLK, F_RDLCK, (offset), (whence), (len))
#define readw_lock(fd, offset, whence, len) \
lock_reg((fd), F_SETLKW, F_RDLCK, (offset), (whence), (len))
#define write_lock(fd, offset, whence, len) \
lock_reg((fd), F_SETLK, F_WRLCK, (offset), (whence), (len))
#define writew_lock(fd, offset, whence, len) \
lock_reg((fd), F_SETLKW, F_WRLCK, (offset), (whence), (len))
#define un_lock(fd, offset, whence, len) \
lock_reg((fd), F_SETLK, F_UNLCK, (offset), (whence), (len))
pid_t lock_test(int, int, off_t, int, off_t); /* {Prog locktest} */
#define is_read_lockable(fd, offset, whence, len) \
(lock_test((fd), F_RDLCK, (offset), (whence), (len)) == 0)
#define is_write_lockable(fd, offset, whence, len) \
(lock_test((fd), F_WRLCK, (offset), (whence), (len)) == 0)
void err_msg(const char *, ...); /* {App misc_source} */
void err_dump(const char *, ...) __attribute__((noreturn));
void err_quit(const char *, ...) __attribute__((noreturn));
void err_cont(int, const char *, ...);
void err_exit(int, const char *, ...) __attribute__((noreturn));
void err_ret(const char *, ...);
void err_sys(const char *, ...) __attribute__((noreturn));
void log_msg(const char *, ...); /* {App misc_source} */
void log_open(const char *, int, int);
void log_quit(const char *, ...) __attribute__((noreturn));
void log_ret(const char *, ...);
void log_sys(const char *, ...) __attribute__((noreturn));
void log_exit(int, const char *, ...) __attribute__((noreturn));
void TELL_WAIT(void); /* parent/child from {Sec race_conditions} */
void TELL_PARENT(pid_t);
void TELL_CHILD(pid_t);
void WAIT_PARENT(void);
void WAIT_CHILD(void);
#endif /* _APUE_H */
可以直接copy該文件到實(shí)例目錄怎静,但是為了多動(dòng)手邮弹,我采用的方法是當(dāng)用到一個(gè)函數(shù)時(shí)才添加進(jìn)去,因此暫時(shí)包含頭文件:
#ifndef _APUE_H
#define _APUE_H
#include<stdio.h> /* headers for convenience */
#include<stdlib.h> /* headers for convenience */
#include<stddef.h> /* for offsetof */
#include<string.h> /* string lib */
#include<unistd.h> /* for convenience */
#include<signal.h> /* for conSIG_ERR */
#endif
2.簡(jiǎn)單運(yùn)行實(shí)例
寫好apue.h頭文件之后蚓聘,開(kāi)始測(cè)試一下腌乡,編寫一個(gè)簡(jiǎn)單的程序調(diào)用getpid()
函數(shù),包含編寫的頭文件apue.h
:
#include "apue.h"
int main()
{
printf("This process ID is %ld \n", (long)getpid());
exit(0);
}
保存為getPid.c
夜牡,gcc(沒(méi)有下載的百度一下)編譯, 鍵入gcc getPid.c -I ../
可以得到可執(zhí)行文件a.out
:
注意与纽,-I ../
原因是因?yàn)?code>apue.h存放在../
目錄里侣签,-I
是編譯時(shí)搜索該目錄,運(yùn)行可執(zhí)行文件:
直接編譯源代碼
如何實(shí)在無(wú)法運(yùn)行實(shí)例急迂,也可以將源代碼解壓影所,閱讀README文件,根據(jù)里面要求make(一般系統(tǒng)都是直接make即可)僚碎。
小結(jié)
OK猴娩,APUE前兩章內(nèi)容主要是版本和基本概念介紹,簡(jiǎn)略過(guò)一下勺阐,主要是下載或者編寫apue.h
用于后面所有的實(shí)例卷中。通過(guò)一個(gè)getpid()
函數(shù)簡(jiǎn)單演示了編譯和運(yùn)行過(guò)程。