编写自己的who命令

今天自己照着书一步步敲了who命令的实现。老外写的有些书就是不错,一步步启发你告诉你怎么思考,怎么根据已有的线索查询联机帮助,怎么一步步最终解决问题。真不错。
下面我就根据书上的思想,来回顾一下这将近2个小时的工作。

1.who命令能做什么
2.who命令是怎么工作的
3.怎样编写who命令


1.who命令能做什么

我们可以在命令行下输入who命令,结果如下:
caoli@caoli-laptop:~$ who
caoli    tty2         2010-11-23 15:41
caoli    tty7         2010-11-23 15:30 (:0)
caoli    pts/0        2010-11-23 16:12 (:0.0)
caoli    pts/1        2010-11-23 16:51 (:0.0)
caoli@caoli-laptop:~$


通过联机帮助
man who

:  Print information about users who are currently logged in.


who可以显示出当前系统中已经登录的用户的信息。
可以大概知道who的作用以及对它的描述和使用的一些选项等。。。。

2.who命令是怎么工作的
step1:
根据man who中:       If FILE is not specified, use /var/run/utmp.

可以知道who是从/var/run/utmp中获得用户登录信息的。

step2:
现在我们知道的是一个目录,下一步就是要找到跟他相关的信息。
man -k 选项可以根据关键字查找。
所以man -k utmp
就能找到utmp(4) 的相关信息,:::我是直接man utmp的。

step3:
再来查看utmp里的详细信息:
more /usr/include/utmp.h
这时候,我们会发现utmp里面保存的是结构数组。记录的用户登录的相关信息。

step4:
根据以上步骤可以推测who的工作原理:
把记录有用户登录信息的文件里的内容一条条读出。我们接着就可以根据这个思想来实现了。

3.怎样编写who命令
关键点:如何从文件里读取数据结构
这里就要用到一些文件操作相关的知识了
(这里要说的是系统调用和大一时学到的fopen等等不是一回事,其实个人觉得功能差不多)

接着我们就开始编码把
主函数部分:
#include <utmp.h>
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>

#define SHOWHOST
void show_info(struct utmp *utbufp);

int main(int argc, char** argv)
{
    struct utmp current_record;//一个缓冲区
    int utmpfd;  //文件描述符
    int reclen=sizeof(current_record);
   
    if((utmpfd=open(UTMP_FILE,O_RDONLY))==-1)
    {
        perror(UTMP_FILE);
        return 1;
    }
    while(read(utmpfd,&current_record,reclen)==reclen)
        show_info(&current_record);  //显示utmp的内容
    close(utmpfd);   
    return 0;
}


显示登录信息部分:
void show_info(struct utmp *utbufp)
{
     if(utbufp->ut_type!=USER_PROCESS)  // 清除空白记录
       return;  
    printf("%-8.8s/t",utbufp->ut_name);
    printf("%-8.8s/t",utbufp->ut_line);
//    printf("%10ld/t",utbufp->ut_time);
    show_time(utbufp->ut_time);
    #ifdef SHOWHOST
    if(utbufp->ut_host[0]!='/0')
        printf("(%s)",utbufp->ut_host);
    #endif
   
    printf("/n");
}


运行结果:
caoli@caoli-laptop:~/workspace/test$ ./who1
reboot      ~           1290497376    (2.6.32-26-generic)
runlevel    ~           1290497376    (2.6.32-26-generic)
LOGIN       tty4        1290497376    ()
LOGIN       tty5        1290497376    ()
caoli       tty2        1290498098    ()
LOGIN       tty3        1290497376    ()
LOGIN       tty6        1290497376    ()
LOGIN       tty1        1290497384    ()
caoli       tty7        1290497409    (:0)
caoli       pts/0       1290499920    (:0.0)
caoli       pts/1       1290501832    (:0.0)
caoli       pts/2       1290501833    ()
caoli       pts/1       1290502298    (:0.0)
caoli       pts/2       1290506795    (:0.0)

这里的结果跟上面一开始who命令的结果做个比较,我们会发现
一,我们要过滤掉不是用户的名字
二,要正确的显示时间
三,不需要显示主机名的可以省略

根据第1点:做出改动如下

utmp结构体中有一项是ut_type,当他的值为7时,表示是一个已经登录的用户,所以可以对显示函数show_info中的显示用户稍作修改
void show_info(struct utmp *utbufp)utmp
{
     if(utbufp->ut_type!=USER_PROCESS)  // 清除空白记录
       return;  
    printf("%-8.8s/t",utbufp->ut_name);


根据第2点:我们根据time.h中的内容做改动,关于时间函数方面的内容,我已经在Linux程序设计第4章中学习过了,所以这里我就直接跳过去了。
ctime(##)中需要一个指向time_t的指针。返回的时间字符串类似下面
Wed Jun 30 21:49:09 1993/n
我们要从4位开始输出12个字符   printf("%12.12s",ctime(&t)+4);
所以添加show_time函数如下
void show_time(long timeval)
{
    char *cp;
    cp=ctime(&timeval);
   
    printf("%12.12s",cp+4);
}

注意ctime是在time.h里面的。所以程序开头部分要添加头文件。

第3点 这里有个预处理的定义,我不太懂,哥要是看到了,就评论一下呗!
    if(utbufp->ut_host[0]!='/0')
        printf("(%s)",utbufp->ut_host);



最后的代码如下:



结果如下:
caoli@caoli-laptop:~/workspace/test$ ./who1
caoli       tty2        Nov 23 15:41
caoli       tty7        Nov 23 15:30(:0)
caoli       pts/0       Nov 23 16:12(:0.0)
caoli       pts/1       Nov 23 16:51(:0.0)

到这里,已经跟原始的who命令相差无几了,哈哈。

通过这次实践,我最大的收获就是不能什么事都问别人,能自己找答案的就尽量自己找,这也是一种能力。
在这方面,我觉得老公做的很好,崇拜他一下!
  好哥,向你致敬。
      爱你的老婆。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值