linux下執行《UNIX環境高階程式設計》的第一個程式時原始碼編譯出錯的處理方法
前幾天買了《UNIX環境高階程式設計》這本書,想好好學習下linux的程式設計。誰知道看到第一個列出指定目錄的內容的那個例子,其實就是shell中 ls 的內容,打好程式碼要執行時一直出問題。後來在網上找了挺多的解決方法,終於解決了。先把方法貼上。
先在終端裡面輸入 vi ls.c在裡面編輯如下程式碼
#include "apue.h"
#include <dirent.h>
int main(int argc, char *argv[])
{
DIR *dp;
struct dirent *dirp;
if (argc != 2)
err_quit("usage: ls directory_name"); //請輸入檔名
if ((dp = opendir(argv[1])) == NULL)
err_sys("can't open %s", argv[1]); //不能開啟檔案
while ((dirp = readdir(dp)) != NULL)
printf("%s\n", dirp->d_name);
closedir(dp);
exit(0);
}
再儲存退出.進行編譯 如gcc ls.c會出現如下的錯誤。
ls.c:1:19: apue.h: No such file or directory
ls.c: In function `main':
ls.c:13: error: `NULL' undeclared (first use in this function)
ls.c:13: error: (Each undeclared identifier is reported only once
ls.c:13: error: for each function it appears in.)
解決方法:
因為apue.h是作者自定義的一個頭檔案,包括程式所需的常用標頭檔案及出錯處理函式。所以因該將它放入系統標頭檔案中(Linux下是 /usr/include),這樣gcc編譯器就可以找到它了。所以先去作者提供網站http://www.apuebook.com/apue2e.html下的
Suorce code 下 的src.2e.tar.gz包,然後解壓至電腦中的某個目錄,比如我的是在/home/xxx(你的登入名)/下,然後進入解壓目錄apue.2e,修改Make.defines.linux中的WKDIR=/home/xxx/apue.2e,為WKDIR=/home/user/apue.2e,這就是我們將要make的工作目錄,然後再進入std目錄,用vi開啟linux.mk,將裡面的nawk全部改為awk
接下里拷貝apue.h到系統預設標頭檔案目錄中(這個目錄是/usr/include)
首先要先卻換到root使用者
然後把apue.2e/include下的apue.h拷貝到 /usr/include
執行下面的命令 #cp /home/xxx/apue.2e/include/apue.h /usr/include
回到ls.c檔案在的目錄執行程式
會出現下面的錯誤:
: undefined reference to `err_quit'
: undefined reference to `err_sys'
解決辦法:
因為err_quit跟err_sys是作者自己定義的錯誤處理函式,需要單獨定義標頭檔案
在/usr/include 下新建一個名為myerr.h的檔案
裡面的內容為
#include "apue.h"
#include <errno.h> /* for definition of errno */
#include <stdarg.h> /* ISO C variable aruments */
static void err_doit(int, int, const char *, va_list);
/*
* Nonfatal error related to a system call.
* Print a message and return.
*/
void
err_ret(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
}
/*
* Fatal error related to a system call.
* Print a message and terminate.
*/
void
err_sys(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
exit(1);
}
/*
* Fatal error unrelated to a system call.
* Error code passed as explict parameter.
* Print a message and terminate.
*/
void
err_exit(int error, const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, error, fmt, ap);
va_end(ap);
exit(1);
}
/*
* Fatal error related to a system call.
* Print a message, dump core, and terminate.
*/
void
err_dump(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
abort(); /* dump core and terminate */
exit(1); /* shouldn't get here */
}
/*
* Nonfatal error unrelated to a system call.
* Print a message and return.
*/
void
err_msg(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(0, 0, fmt, ap);
va_end(ap);
}
/*
* Fatal error unrelated to a system call.
* Print a message and terminate.
*/
void
err_quit(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(0, 0, fmt, ap);
va_end(ap);
exit(1);
}
/*
* Print a message and return to caller.
* Caller specifies "errnoflag".
*/
static void
err_doit(int errnoflag, int error, const char *fmt, va_list ap)
{
char buf[MAXLINE];
vsnprintf(buf, MAXLINE, fmt, ap);
if (errnoflag)
snprintf(buf+strlen(buf), MAXLINE-strlen(buf), ": %s",
strerror(error));
strcat(buf, " ");
fflush(stdout); /* in case stdout and stderr are the same */
fputs(buf, stderr);
fflush(NULL); /* flushes all stdio output streams */
}
接下來在ls.c裡面引用#include<myerr.h>就好了。
在執行程式
gcc ls.c
./a.out /home
就可以看到你home目錄下面的內容了