UNIX环境高级编程学习——列出一个目录中的所有文件

来源:互联网 发布:网络插画班比较好的 编辑:程序博客网 时间:2024/05/18 12:38

        初次接触UNIX环境高级编程,发现很多都已经被作者定义到头文件apue.h,我们根本无法知道那个函数是在那个头文件里面,对于初学者来说无法了解一个程序整个执行的过程。既然说到这里了,那就从第一个程序开始。

/*************************** * 列出一个目录中的所有文件 *     2013-07-22 *      wjy***************************/#include <stdio.h>#include <stdlib.h>#include <errno.h>#include <dirent.h>#include <stdarg.h>#include <string.h>#define MAXLINE     4096static 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, "/n");    fflush(stdout);    fputs(buf, stderr);    fflush(NULL);}void err_quit(const char *fmt, ...){    va_list ap;    va_start(ap, fmt);    err_doit(0, 0, fmt, ap);    va_end(ap);    exit(1);}void err_sys(const char *fmt, ...){    va_list ap;    va_start(ap, fmt);    err_doit(1, errno, fmt, ap);    va_end(ap);    exit(1);}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);}

程序执行结果:

uestc@Ubuntu:~/code/UNIX环境高级编程/01$ ./main /home/uestc/code/test_net.get_time..UNIX环境高级编程
       从上面的结果我们可以知道这个程序是正确运行了的,虽然很简单,但是包含的确实有一些值得深究的,我们从主函数一个一个往下看:

DIR是一个结构体,用来保存当前读取目录的有关信息。

dirent结构如下:

struct dirent{long d_ino; // inode number 索引节点号    off_t d_off; // offset to this dirent 在目录文件中的偏移    unsigned short d_reclen; // length of this d_name 文件名长    unsigned char d_type; // the type of d_name 文件类型   char d_name[NAME_MAX+1]; // 文件名,最长255字符}

继续向下,接触到的是err_quit函数,这个时候不得不说可变参数函数中用到的va_start、va_arg、va_end三个宏和一个va_list类型,这三个宏在头文件stdarg.h中。

对于可变参数,注意参数列表中的省略号:它提示此处可能传递数量和类型未确定的参数。函数声明了一个名叫ap的va_list类型的变量,用于访问参数列表的未确定部分。这个变量通过调用va_start来初始化。它的第一个参数是va_list变量的名字,就是上面代码中ap。第二个参数是省略号前最后一个有名字的参数。初始化过程把ap变量设置为指向可变参数部分的第一个参数。

为了访问参数,需要使用va_arg,这个宏接受两个参数:va_list变量和参数列表中下一个参数的类型。va_arg返回这个参数的值,并使ap指向下一个可变参数。最后,当访问完毕最后一个可变参数之后,需要调用va_end。可变参数必须从头到尾按照这个顺序访问,如果你在访问几个可变参数后想中途中止,这是可以的。但是,如果你想一开始就访问参数列表中间的参数,那是不行的。另外,由于可变参数列表中并没有原型,所以,所有作为可变参数传递给函数的值都将执行缺省参数类型提升。


void BubbleSort(int a[], int n)//n为数组a的元素个数{int i, j, temp;for (j = 0; j < n-1 ; j++)for (i = 0; i < n-1-j; i++){if (a[i] > a[i+1])//数组元素大小按升序排列{temp = a[i];a[i] = a[i + 1];a[i + 1] = temp;}}}


//快速排序  void quick_sort(int s[], int low, int high)  {  if (low < high)  {  int i = low, j = high, x = s[low];  while (i < j)  {  while(i < j && s[j] >= x) // 从右向左找第一个小于x的数  j--;    if(i < j)   s[i++] = s[j];  while(i < j && s[i] < x) // 从左向右找第一个大于等于x的数  i++;    if(i < j)   s[j--] = s[i];  }  s[i] = x;  quick_sort(s, low, i - 1); // 递归调用   quick_sort(s, i + 1, high);  }  }  

void InsertionSort(int A[ ], int N){int j, P;int Tmp;for(P = 1; P < N; P++){Tmp = A[ P ];for(j = P; j > 0 && A[ j - 1 ] > Tmp; j--)A[ j ] = A[ j - 1 ];A[ j ] = Tmp;}}


void Shellsort(int A[ ], int N){int i, j, Increment;int Tmp;for(Increment = N / 2; Increment > 0; Increment /= 2)for(i = Increment; i < N; i++){Tmp = A[i];for(j = i; j >= Increment; j -= Increment)if(Tmp < A[j - Increment])A[j] = A[j - Increment];elsebreak;A[j] = Tmp;}}


#define LeftChild( i )  ( 2 * ( i ) + 1 )void Swap(int *Lhs, int *Rhs ){int Tmp = *Lhs;*Lhs = *Rhs;*Rhs = Tmp;}void PercDown(int A[ ], int i, int N){int Child;int Tmp;for( Tmp = A[ i ]; LeftChild( i ) < N; i = Child ){Child = LeftChild( i );if( Child != N - 1 && A[ Child + 1 ] > A[ Child ] )Child++;if( Tmp < A[ Child ] )A[i] = A[Child];elsebreak;}A[ i ] =Tmp;}void Heapsort(int A[ ], int N){int i;for(i = N / 2; i >= 0; i--)  /* BuildHeap */PercDown( A, i, N );for(i = N - 1; i > 0; i--){Swap(&A[ 0 ], &A[ i ]);  /* DeleteMax */PercDown(A, 0, i);}}

void Merge(int A[ ], int TmpArray[ ], int Lpos, int Rpos, int RightEnd){int i, LeftEnd, NumElements, TmpPos;LeftEnd = Rpos - 1;TmpPos = Lpos;NumElements = RightEnd - Lpos + 1;/* main loop */while( Lpos <= LeftEnd && Rpos <= RightEnd )if( A[ Lpos ] <= A[ Rpos ] )TmpArray[ TmpPos++ ] = A[ Lpos++ ];elseTmpArray[ TmpPos++ ] = A[ Rpos++ ];while( Lpos <= LeftEnd )  /* Copy rest of first half */TmpArray[ TmpPos++ ] = A[ Lpos++ ];while( Rpos <= RightEnd ) /* Copy rest of second half */TmpArray[ TmpPos++ ] = A[ Rpos++ ];/* Copy TmpArray back */for( i = 0; i < NumElements; i++, RightEnd-- )A[ RightEnd ] = TmpArray[ RightEnd ];}void MSort(int A[ ], int TmpArray[ ], int Left, int Right){int Center;if(Left < Right){Center = ( Left + Right ) / 2;MSort( A, TmpArray, Left, Center );MSort( A, TmpArray, Center + 1, Right );Merge( A, TmpArray, Left, Center + 1, Right );}}void Mergesort(int A[ ], int N){int *TmpArray;TmpArray = malloc(N * sizeof( int ));if( TmpArray != NULL ){MSort( A, TmpArray, 0, N - 1 );free( TmpArray );}elsereturn;}

int MyStrlen(const char *str){int ret = 0;if (NULL == str)return 0;while (*str++ != '\0')ret++;return ret;}


/*** client客服端*/#include <stdio.h>#include <unistd.h>#include <stdlib.h>#include <string.h>#include <sys/socket.h>#include <netinet/in.h>#define MAXLINE     1024int main(int argc, char *argv[]){    int sockfd, n;    char buff[MAXLINE];    struct sockaddr_in servaddr;    if (argc != 2)    {        printf("Usage: ./client [IP_ADDR]\n");        return 0;    }    if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)    {        printf("Create socket failed!\n");        return 0;    }    bzero(&servaddr, sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_port = htons(8989);    if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0)    {        printf("inet_pton failed!\n");        return 0;    }    if (connect(sockfd, (struct sockaddr *) &servaddr, sizeof(servaddr)) < 0)    {        printf("connect server failed!\n");        return 0;    }    while ((n = read(sockfd, buff, MAXLINE)) > 0)    {        buff[n] = '\0';        printf("%s\n", buff);        memset(buff, 0, MAXLINE);    }    if (n < 0)    {        printf("read data from socket failed!");        return 0;    }    return 0;}

/*** server服务器*/#include <stdio.h>#include <unistd.h>#include <stdlib.h>#include <string.h>#include <time.h>#include <sys/socket.h>#include <netinet/in.h>#define LISTENQ     1024#define MAXLINE     1024int main(int argc, char *argv[]){    int listenfd, connfd;    struct sockaddr_in servaddr;    char buff[MAXLINE];    time_t ticks;    listenfd = socket(AF_INET, SOCK_STREAM, 0);    bzero(&servaddr, sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);    servaddr.sin_port = htons(8989);    if (bind(listenfd, (struct sockaddr *) &servaddr, sizeof(servaddr)) < 0)    {        printf("bind socket failed!");        return 0;    }    if (listen(listenfd, LISTENQ) != 0)    {        printf("set max listen socket failed!\n");        return 0;    }    for ( ; ; )    {        connfd = accept(listenfd, (struct sockaddr *) NULL, NULL);        ticks = time(NULL);        snprintf(buff, sizeof(buff), "%.24s\r\n", ctime(&ticks));        if (write(connfd, buff, strlen(buff)) < 0)        {            printf("write data to socket failed!\n");            return 0;        }        close(connfd);    }    return 0;}