编写自己的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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • c++入门教程–-21C++类访问修饰符

    c++入门教程–-21C++类访问修饰符

    2021年3月12日
    120
  • 安防监控基础知识

    安防监控基础知识针对安防视频监控方面的基础知识UDP:用户数据报协议(无连接,封装实时性强的网络音频数据)TCP:传输控制协议(面向连接,传输实时性强的音频流)HTTP:超文本传输协议,网络摄像机通过HTTP提供web访问功能,将音频数据经过复杂网络传输.RTP:实时传输协议,提供时间信息流和实现流同步(本身不提供可靠的传输机制和流量控制)RTCP:实时传输控制协议,提供可靠的…

    2022年6月28日
    35
  • mysql查看数据库表数量

    mysql查看数据库表数量

    2022年2月9日
    40
  • spring事务的传播机制

    spring事务的传播机制什么是事务 数据库事务是指一系列严密操作 要么全部成功 要么全部失败 它有四种特性 原子性 一致性 隔离性和持久性 而 spring 事务是封装在数据库事务之上的一种事务处理机制 它有两种管理方式 编程式事务和声明式事务 在平时使用中 我们大多使用 Transactiona 声明式事务来管理 这也是 spring 推荐的方式 下面例子也统一采用此种方式 下面我们主要来看看 spring 事务的传播机制 spring 事务的传播机制 spring 事务的传播机制有七种 REQUIRED REQUIRES NEW NES

    2025年6月5日
    0
  • JSONArray 转list

    JSONArray 转listJSONArray 转list 可以通过如下简便方法进行JSONArrayprogramsArray=jsonobject.getJSONArray(“programs”);Listlist=(List)JSONArray.toCollection(programsArray,Programs.class);转换过程中容易产生NoSuchMethodEx

    2022年6月23日
    30
  • 什么是Unix时间戳?(转载)[通俗易懂]

    什么是Unix时间戳?(转载)[通俗易懂]Unix时间戳(Unixtimestamp),或称Unix时间(Unixtime)、POSIX时间(POSIXtime),是一种时间表示方式,定义为从格林威治时间1970年01月01日00时00

    2022年8月6日
    6

发表回复

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

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