Unix低级编程2——who

1、序
可能是外面的炎热,身上的汗臭,逼仄的笔记本屏幕,这几种糟糕的事物聚集在一起,导致现在一点写作热情都没有。不过一会还要锻炼,趁着刚写完程序,我觉得还是稍微写写心得,也算是一个复习。希望对大家有所帮助。

本篇文章我想应该是更加强调如何善用man——在线联机文档,这个强大的工具。用好了这样一个工具,我想对以后的工作,肯定是有所裨益的。好了,闲话不多说,直接切入正题。

2、实现
这个程序我打算按照书上的安排,设计成两个版本,但是,如果两个版本,我觉得无论如何对大家学习知识都没有任何好处——绝对增加了大家的负担——本来一个简单的问题还要分两步走,事倍功半。

#include <stdio.h>
#include <unistd.h>
#include <utmp.h>
#include <fcntl.h>
#include <time.h>

void show_info(struct utmp * buffer);
void show_time(time_t *time);

int main(int argc, char **argv)
{
    struct utmp buffer_utmp;
    int buffer_len=sizeof(buffer_utmp);
    int utmp_fd=open("/var/run/utmp", O_RDONLY);//打开用户文件

    while(read(utmp_fd,&buffer_utmp, buffer_len) == buffer_len)
    {
        show_info(&buffer_utmp);
    }
    return 0;
}

void show_info(struct utmp * buffer)
{
    printf("%-10s ",buffer->ut_user);//打印用户名
    printf("%-10s  ", buffer->ut_line);//打印登录终端
    //printf("%20d  ",buffer->ut_tv.tv_sec);//打印时间
    show_time(&(buffer->ut_tv.tv_sec));//按照字符串打印时间

    int i;
    for(i=0;i<3;i++)
        printf("%d.",buffer->ut_addr_v6[i]);
    printf("%d",buffer->ut_addr_v6[i]);

    printf("\n");
}

void show_time(time_t *time)
{
    printf("%-20s ",ctime(time));
}

3、分析
大家也看到了,本身代码并不复杂,还不到50行,但是,从引用的头文件中,也可以看出来,这个程序系统系统相关性很强,涉及到很多的系统文件记忆数据结构。所以,这篇文章将用一部分篇幅来说说系统调用。

1)我们首先用指令man who来看看who程序的具体说明,注意到在文档中有提到/var/run/utmp,这个文件其实就是存用户登录信息的文件。我们也能猜到,只有在开机的状态下用户才能够登录,所以不可能把当前登录的用户写到磁盘中,因此/var/run/utmp文件实际上是驻留在内存中的一大坨数据。

说到数据,这个文件就应该有一定的结构——我们输入who命令的时候,也是逐条按顺序显示用户登录信息的。那么这个结构的定义(存储登录用户信息的数据结构)放在哪里呢?同样,还是在man who里面,说到了utmp.h——这个文件中即存储了一堆的struct结构体,来定义存储登录用户信息的类型。

2)再man utmp一下,我们就进入了utmp.h的说明文档(本人用的centos,我想这个和平台没太大相关性。)如果英文还可以的同学,我觉得看注释就完全可以明白了。当然由于我用的事centos,所以“登录时间”那一部分是一个内嵌的结构体。这点需要注意下。

3)好,程序基本上就已经成了。如我给大家的程序显示那样。main函数中,首先定义一个struct utmp类型的变量,然后从文件中,一个一个地读struct utmp类型的变量,并将变量中的内容通过函数show_info显示出来。

4)感觉差不多要完工了。不过还有个事要注意,就是打印时间的函数。由于utmp结构体中存储的时间是一个整数,其表示相对于1970年1月1日0时的秒数,因此需要将其转换成人类能读懂的时间值,所以我们又用到了一个转换函数——ctime。继续man ctime。我们就能看到关于这个函数的具体说明。
但是还有一点需要注意,就是ctime()的参数类型,由于系统相关性,所以Linux表示时间统一用到了time_t类型来表示时间,其实就是个int类型,但是为了让生成可执行文件的时候看着舒心点,不会弹出一大堆warning,note来,我觉得还是按照人家套路办事会比较好些。

5)最后注意一点,对于utmp结构体中有一个成员属性ut_type,这个指的就是上面定义的一大堆#define,先看下常量名称,我想大家就都明白了。此处需要说明的是,书中说/var/run/utmp文件中存了所有终端的信息,不管有没有用户用这些终端,都会显示出来。不过目前我的这个utmp没有产生这种状况。只显示了用户登录的终端信息。所以有的时候还是需要根据环境具体分析问题。

    原文作者:吃根香蕉压压惊
    原文地址: https://www.jianshu.com/p/29810dd53991
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞