编写自己的who命令

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

1.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命令相差无几了,哈哈。

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/131751.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 项目复盘/总结模板

    项目复盘/总结模板复盘会议的安排、步骤及意义什么是复盘?为何要复盘?WHAT?复盘,是行动后的深刻反思和经验总结,是一个不断学习、总结、反思、提炼和持续提高的过程。复盘是企业文化、是行动学习、是提升组织智慧的重

    2022年8月3日
    8
  • 列车调度(PTA)

    列车调度(PTA)7-11列车调度(25分)火车站的列车调度铁轨的结构如下图所示。两端分别是一条入口(Entrance)轨道和一条出口(Exit)轨道,它们之间有N条平行的轨道。每趟列车从入口可以选择任意一条

    2022年7月1日
    21
  • day72Django

    day72Django

    2021年6月15日
    105
  • poe交换机的供电方式_交换机需要电源吗

    poe交换机的供电方式_交换机需要电源吗前言:近年来,PoE供电技术的发展势头越来越强劲。凭借简化用电设备的安装和部署、节能,安全等一系列优势,PoE供电成为无线覆盖、安防监控、以及智能电网等场景的新宠。在技术交流中,工程商困惑最多的其中就有POE供电的问题,本文就汇总了大家最关注的问题,集中解答。 ▶ 问题一:何为PoE技术? PoE(PowerOverEthernet)指的是在现有的以太网Cat.5布线基础…

    2022年9月28日
    3
  • nio和零拷贝_零拷贝

    nio和零拷贝_零拷贝传统IO传统IO的数据拷贝流程如下图:数据需要从磁盘拷贝到内核空间,再从内核空间拷到用户空间(JVM)。程序可能进行数据修改等操作再将数据拷贝到内核空间,内核空间再拷贝到网卡内存,通过网络发送出去(或拷贝到磁盘)。即数据的读写(这里用户空间发到网络也算作写),都至少需要两次拷贝。当然磁盘到内核空间属于DMA拷贝(DMA即直接内存存取,原理是外部设备不通过CPU而直接与系统内存交换……

    2022年9月21日
    2
  • 七. 200多万元得到的创业教训–周鸿祎传授的“活法”

    七. 200多万元得到的创业教训–周鸿祎传授的“活法”

    2022年1月31日
    48

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号