linux系统编程:自己动手写一个who命令
who命令的作用用于显示当前有哪些用户登录到系统。
这个命令执行的原理是读取了系统上utmp文件中记录的所有登录信息,直接显示出来的
utmp文件在哪里呢?
man who的时候,在手册下面有这么一段说明:意思就是不指定文件参数,那么读取的就是/var/run/utmp,到底是不是,验证下
If FILE is not specified, use /var/run/utmp. /var/log/wtmp as FILE is
common. If ARG1 ARG2 given, -m presumed: 'am i' or 'mom likes' are
usual.
当我指定file参数为/var/run/utmp或者省略这个参数的时候,结果都是一样, 当我用一个错误的文件时,没有任何结果,从这里可以推断,who命令确实从/var/run/utmp中读取用户登录的信息
ghostwu@ubuntu:~$ who
ghostwu tty7 -- : (:)
ghostwu pts/ -- : (:)
ghostwu pts/ -- : (:)
ghostwu pts/ -- : (:)
ghostwu@ubuntu:~$ who -b
system boot -- :
ghostwu@ubuntu:~$ who -b /var/run/utmp
system boot -- :
ghostwu@ubuntu:~$ who -b /var/run/utmp2
ghostwu@ubuntu:~$ who -b /var/run/utmp3
那么utmp到底在哪里?
利用man -k utmp 查找所有的可能: 推断---> utmp (5) - login records 这里的可能性比较大,描述说,这里是记录登录信息的
ghostwu@ubuntu:~$ man -k utmp
endutent () - access utmp file entries
endutxent () - access utmp file entries
getutent () - access utmp file entries
getutent_r () - access utmp file entries
getutid () - access utmp file entries
getutid_r () - access utmp file entries
getutline () - access utmp file entries
getutline_r () - access utmp file entries
getutmp () - copy utmp structure to utmpx, and vice versa
getutmpx () - copy utmp structure to utmpx, and vice versa
getutxent () - access utmp file entries
getutxid () - access utmp file entries
getutxline () - access utmp file entries
login () - write utmp and wtmp entries
logout () - write utmp and wtmp entries
pututline () - access utmp file entries
pututxline () - access utmp file entries
sessreg () - manage utmpx/wtmpx entries for non-init clients
setutent () - access utmp file entries
setutxent () - access utmp file entries
systemd-update-utmp () - Write audit and utmp updates at bootup, runlevel ch...
systemd-update-utmp-runlevel.service () - Write audit and utmp updates at bo...
systemd-update-utmp.service () - Write audit and utmp updates at bootup, run...
utmp () - login records
utmpdump () - dump UTMP and WTMP files in raw format
utmpname () - access utmp file entries
utmpx () - login records
utmpxname () - access utmp file entries
接下来,我们去 man 5 utmp 看下,会发现有这么一段提示:
The file is a sequence of utmp structures, declared as follows in <utmp.h> (note that this
is only one of several definitions around; details depend on the version of libc):
意思是utmp文件的信息是一系列utmp结构体数据, 这个结构体定义在utmp.h文件中, 每个linux发行版可能不一样.
接下来,我用强大的find命令查找到了2个目标:
ghostwu@ubuntu:~$ find /usr/include -name "utmp.h"
/usr/include/x86_64-linux-gnu/bits/utmp.h
/usr/include/utmp.h
结构体的定义就在这个文件中( /usr/include/x86_64-linux-gnu/bits/utmp.h )
这里有两个宏要注意下( ut_time和UTMP_FILE ), 下面的程序会用到
#ifndef _NO_UT_TIME
/* We have a problem here: `ut_time' is also used otherwise. Define
_NO_UT_TIME if the compiler complains. */
# define ut_time ut_tv.tv_sec
#endif
ghostwu@ubuntu:~$ grep "UTMP_FILE" /usr/include/utmp.h
#define UTMP_FILE _PATH_UTMP
#define UTMP_FILENAME _PATH_UTMP
ghostwu@ubuntu:~$ grep "_PATH_UTMP" /usr/include/utmp.h
#define UTMP_FILE _PATH_UTMP
#define UTMP_FILENAME _PATH_UTMP
ghostwu@ubuntu:~$ grep "_PATH_UTMP" /usr/include/x86_64-linux-gnu/bits/utmp.h
ghostwu@ubuntu:~$ grep "_PATH_UTMP" /usr/include/*.h
/usr/include/paths.h:#define _PATH_UTMP "/var/run/utmp"
/usr/include/utmp.h:#define UTMP_FILE _PATH_UTMP
/usr/include/utmp.h:#define UTMP_FILENAME _PATH_UTMP
/usr/include/utmpx.h:# define UTMPX_FILE _PATH_UTMPX
/usr/include/utmpx.h:# define UTMPX_FILENAME _PATH_UTMPX
ghostwu@ubuntu:~$
UTMP_FILE的查找思路: 首先grep两个目录下面的文件utmp.h,在/usr/include/utmp.h找到一个宏定义 _PATH_UTMP,下一步就是确定 _PATH_UTMP到底是什么,利用grep "_PATH_UTMP" /usr/include/*.h
最终在paths.h头文件中,发现了他的真面目
who命令书写思路:
1)从/var/run/utmp读取文件,每次读取一个struct utmp结构体这么大,如果长度每次都有这么大,继续读取
2)格式化4个信息:用户名,主机,地址,时间
3)只打印当前活动的用户(当前登录的用户)
4)格式化时间( 小时,分钟,秒, >10的补0, <10的原样返回 )
源代码
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:mywho.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月08日
* 描 述:
*
================================================================*/ #include <stdio.h>
#include <utmp.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <time.h>
#include <string.h> #ifndef UTMP_FILE
#define UTMP_FILE "/var/run/utmp"
#endif int count = ; //格式化时间, <10 就补0, >10 原样返回
char* format_time( char* s, const char *time ) {
if( strlen( time ) < ) {
return strcat( s, time );
}
return strcpy( s, time );
} void show_info( struct utmp* t_utmp ) {
if ( t_utmp->ut_type != USER_PROCESS ) //不显示 非活跃的用户信息
return; printf( "%-8.8s", t_utmp->ut_user );
printf( " " );
printf( "%-8.8s", t_utmp->ut_line );
printf( " " ); //printf( " " );
//printf( "%12.12s", ctime( (time_t*)&(t_utmp->ut_time) ) + 4 ); //+4--->去除天(day)和后面的空格 /*测试localtime用法
//当前时间
time_t now;
struct tm* pNow;
time( &now );
pNow = localtime( &now );
printf( "%d-%d-%d %d:%d", pNow->tm_year + 1900, pNow->tm_mon + 1, pNow->tm_mday, pNow->tm_hour, pNow->tm_min );
*/ struct tm* ptm;
time_t u_time = t_utmp->ut_time;
ptm = localtime( &u_time );
int ihour = ptm->tm_hour;
int imin = ptm->tm_min; char hour[] = "";
char hour2[] = "";
sprintf( hour2, "%d", ihour );
format_time( hour, hour2 ); char min[] = "";
char min2[] = "";
sprintf( min2, "%d", imin );
format_time( min, min2 ); //printf( "%d-%d-%d %d:%d", ptm->tm_year + 1900, ptm->tm_mon + 1, ptm->tm_mday, ihour, imin );
printf( "%d-%d-%d %s:%s", ptm->tm_year + , ptm->tm_mon + , ptm->tm_mday, hour, min ); printf( " " );
printf( "%-8.8s", t_utmp->ut_host ); printf( "\n" );
} int main(int argc, char *argv[])
{
struct utmp myutmp;
int fd = -;
int reclen = sizeof( myutmp ); fd = open( UTMP_FILE, O_RDONLY ); if( - == fd ) {
perror( "open utmp" );
exit( - );
} //printf( "fd = %d\n", fd ); while( read( fd, &myutmp, reclen ) == reclen ) {
count++;
show_info( &myutmp );
}
printf( "文件读取的次数:%d\n", count );
close( fd ); return ;
}
总结:
一个非常小的功能,囊括以下知识点:
1)文件读取
2)man手册与系统命令使用技巧
3)指针用法
4)字符串函数用法
5)时间函数用法
6)宏与typedef的用法
linux系统编程:自己动手写一个who命令的更多相关文章
- Linux系统编程【2】——编写who命令
学到的知识点 通过实现who命令,学到了: 1.使用man命令寻找相关信息 2.基于文件编程 3.体会到c库函数与系统调用的不同 4.加深对缓冲技术的理解 who命令的作用 who命令的使用 在控制终 ...
- linux系统编程综合练习-实现一个小型的shell程序(一)
之前已经花了不少篇幅学习了linux系统编程的很多知识点:文件与io.进程.信号.管道,而零散的知识点,怎么能够综合的串接起来是学习的一个很重要的目的,当然最好的方式就是用所学的知识点做一个项目了,所 ...
- Linux系统编程【3.2】——ls命令优化版和ls -l实现
前情提要 在笔者的上一篇博客Linux系统编程[3.1]--编写ls命令中,实现了初级版的ls命令,但是与原版ls命令相比,还存在着显示格式和无颜色标记的不同.经过笔者近两天的学习,基本解决了这两个问 ...
- linux系统编程:自己动手写一个cp命令
cp命令的基本用法: cp 源文件 目标文件 如果目标文件不存在 就创建, 如果存在就覆盖 实现一个cp命令其实就是读写文件的操作: 对于源文件: 把内容全部读取到缓存中,用到的函数read 对于目标 ...
- linux系统编程综合练习-实现一个小型的shell程序(四)
上节中已经对后台作业进行了简单处理,基本上要实现的功能已经完了,下面回过头来,对代码进行一个调整,把写得不好的地方梳理一下,给代码加入适当的注释,这种习惯其实是比较好了,由于在开发的时候时间都比较紧, ...
- linux系统编程综合练习-实现一个小型的shell程序(三)
上节中已经实现了对普通命令的解析,包括输入重定向,输出重定向,管道,后台作业,这次就来执行已经解析好的命令,对应的函数为:execute_command(),首先对带有管道的命令进行执行: 比如:&q ...
- linux系统编程综合练习-实现一个小型的shell程序(二)
上节minishell当中,已经初步实现了一个简单命令的解析,这节来继续对更加复杂命令进行解析,包含:输入重定向的解析.管道行的解析.输出重定向的解析以及是否有后台作业的解析,如下: 下面对其进行实现 ...
- Linux系统编程【1】——编写more命令
背景介绍 笔者知识背景 笔者接触Linux快一年了.理论知识方面:学习了操作系统基础知识,了解进程调度.内存分配.文件管理.磁盘I/O这些基本的概念. 实操方面:会使用Linux简单命令,在嵌入式系统 ...
- Linux系统编程(14)——shell常用命令
1. ls命令 ls命令是列出目录内容(ListDirectory Contents)的意思.运行它就是列出文件夹里的内容,可能是文件也可能是文件夹. "ls -l"命令已详情模式 ...
随机推荐
- javascript设计模式——组合模式
前面的话 在程序设计中,有一些和“事物是由相似的子事物构成”类似的思想.组合模式就是用小的子对象来构建更大的对象,而这些小的子对象本身也许是由更小的“孙对象”构成的.本文将详细介绍组合模式 宏命令 宏 ...
- springMVC使用jsp:include嵌入页面的两种方式
1.静态嵌入子页面 <%@ include file="header.jsp" %> 静态嵌入支持 jsp . html . xml 以及纯文本. 静态嵌入在编译时 ...
- 从Windows迁移SQL Server到Linux
前一篇博客关于SQL Server on Linux的安装,地址:http://www.cnblogs.com/fishparadise/p/8057650.html,现在测试把Windows平台下的 ...
- 逐步搭建Lamp环境之vim的三种模式以及基本命令
在Linux中vim的三种模式分别为:命令模式.末行模式.编辑模式.以下是三者的关系图: 三种模式的彼此切换: 命令模式是vim中的默认模式. 命令模式切换至末行模式: 使用英文冒号(:). 末行模式 ...
- Android从无知到有知——NO.1
如期而至的软件设计大赛吹响了重生的号角.正如同我们的指导老师所说,这个暑假会影响你近几年的发展,也可能会决定你以后所走的道路. 是的.我身边就有非常好的样例,有些师哥师姐们常常跟我们说.软件大赛不仅使 ...
- DDD理论学习系列——案例及目录
目录 DDD理论学习系列(1)-- 通用语言 DDD理论学习系列(2)-- 领域 DDD理论学习系列(3)-- 限界上下文 DDD理论学习系列(4)-- 领域模型 DDD理论学习系列(5)-- 统一建 ...
- 一位10年Java工作经验的架构师聊Java和工作经验
从事近十年的 JavaEE 应用开发工作,现任阿里巴巴公司系统架构师.对分布式服务架构与大数据技术有深入研究,具有丰富的 B/S 架构开发经验与项目实战经验,擅长敏捷开发模式.国内开源软件推动者之一, ...
- Learning to Rank算法介绍:GBRank
之前的博客:http://www.cnblogs.com/bentuwuying/p/6681943.html中简单介绍了Learning to Rank的基本原理,也讲到了Learning to R ...
- TextMesh Pro SpriteAsset Load From Assetbundle
遇到问题 我们项目分两个Unity的工程,Art(美术资源工程),Client(代码工程) 在Art工程中的TextMeshProUGUI Text中使用Emoji,打包成AB之后,在Client运行 ...
- JavaScript--AJAX页面传值
1.首先 闲话不说 直接代码走起,都是我工作闲事的积累干货 //重要 js 运行 $(function (){ 代码 }); 2.ajax 传值 //第一种 输入框 <input type=&q ...