1. 程式人生 > >linux下執行《UNIX環境高階程式設計》的第一個程式時原始碼編譯出錯的處理方法

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目錄下面的內容了