linux系统编程:自己动手写一个ls命令
ls用于列举目录内容,要实现这个功能,毫无疑问,需要读取目录,涉及到两个api:
opendir:DIR *opendir(const char *name), 传文件名,返回一个指针,指向目录序列
readdir:struct dirent *readdir(DIR *dirp), 把opendir的返回值传过来, 返回值为一个结构体
struct dirent {
ino_t d_ino; /* inode number */
off_t d_off; /* not an offset; see NOTES */
unsigned short d_reclen; /* length of this record */
unsigned char d_type; /* type of file; not supported
by all filesystem types */
char d_name[]; /* filename */
};
有了这两个api,就可以实现一个简易的ls功能
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:myls.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月09日
* 描 述: ls命令
*
================================================================*/ #include <stdio.h>
#include <sys/types.h>
#include <dirent.h>
#include <stdlib.h> void do_ls( char [] ); int main(int argc, char *argv[])
{
if( argc == ) {
do_ls( "." );
}else {
while( --argc ) {
printf( "arg=%s\n", * ++argv );
do_ls( *argv );
}
}
return ;
} void do_ls( char dir_entry[] ) {
DIR* pDir;
struct dirent* pCurDir;
if( ( pDir = opendir( dir_entry ) ) == NULL ){
perror( "read dir" );
exit( - );
}else {
while( ( pCurDir = readdir( pDir ) ) != NULL ) {
printf( "%s\n", pCurDir->d_name );
}
}
}
这个简易的ls功能,列举出了所有的文件( 包括隐藏的 ), 但是很多的信息不全,如: 权限,用户和组,修改时间,文件大小,链接数目等,stat这个api可以获取文件的这些信息
stat:获取文件状态信息
原型:int stat(const char *pathname, struct stat *buf), 第一个参数:文件名, 第二个参数:保存文件状态信息的结构体( man 2 stat 有结构体相关说明 )
1、获取文件的大小
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:stat.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月09日
* 描 述:
*
================================================================*/ #include <stdio.h>
#include <sys/stat.h> #define FILENAME "/etc/passwd" int main(int argc, char *argv[])
{
struct stat filestat; if( - == stat( FILENAME, &filestat ) ) {
perror( "file stat" );
return -;
}else {
printf( "the size of %s is %ld\n",FILENAME, filestat.st_size );
} return ;
}
2、读取文件st_mode(权限位), 用户id, 组id, 修改时间,链接数目
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:stat2.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月09日
* 描 述:
*
================================================================*/ #include <stdio.h>
#include <sys/stat.h>
#include <string.h>
#include <time.h> void show_info( char *file, struct stat* statinfo );
void show_time( time_t filetime );
char* format_time( char* dsttime, const char* srctime ); int main(int argc, char *argv[])
{
struct stat fileinfo;
if( argc > ) {
/*调试信息
printf( "%s\n", argv[1] );
int res = stat( argv[1], &fileinfo );
printf( "%d\n", res );
*/
if( stat( argv[], &fileinfo ) != - ) {
show_info( argv[], &fileinfo );
}
}else {
perror( "get args from terminal" );
} return ;
} void show_info( char* file, struct stat* statinfo ){
printf( "%s文件信息如下:\n", file );
printf( "st_mode = %d\n", statinfo->st_mode );
printf( "links = %ld\n", statinfo->st_nlink );
printf( "uid = %d\n", statinfo->st_uid );
printf( "gid = %d\n", statinfo->st_gid );
printf( "file size = %ld\n", statinfo->st_size );
show_time( statinfo->st_mtime );
} void show_time( time_t filetime ) {
struct tm* ptm;
ptm = localtime( &filetime ); int month = ptm->tm_mon + ;
int day = ptm->tm_mday;
int hour = ptm->tm_hour;
int min = ptm->tm_min; char srchour[] = "";
char srcmin[] = "";
char dsthour[] = "";
char dstmin[] = "";
sprintf( srchour, "%d", hour );
sprintf( srcmin, "%d", min );
format_time( dsthour, srchour );
format_time( dstmin, srcmin ); printf( "文件最后修改时间: %d月\t%d\t%s:%s\n", month, day, dsthour, dstmin );
} char* format_time( char* dsttime, const char* srctime ) {
if( strlen( srctime ) < ) {
return strcat( dsttime, srctime );
}
return strcpy( dsttime, srctime );
}
3、权限st_mode转字符权限位( 如: -rwxrwxrwx ), 用户id和组id转用户名和组名称,判断文件类型
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:stat3.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月09日
* 描 述:文件类型与权限位
*
================================================================*/ #include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <string.h>
#include <sys/types.h>
#include <pwd.h>
#include <grp.h> void do_ls( char* filename );
void show_filetype( char* filename, int filemode );
void show_filetype2( char* filename, int filemode );
void mode_to_letters( int filemode, char str[] );
//用户id转名称
char* uid_to_name( uid_t uid );
//组id转名称
char* gid_to_name( gid_t gid ); int main(int argc, char *argv[])
{
if( argc < ) {
printf( "usage:%s file\n", argv[] );
return -;
}else {
do_ls( argv[] );
}
return ;
} char* uid_to_name( uid_t uid ){
return getpwuid( uid )->pw_name;
} char* gid_to_name( gid_t gid ){
return getgrgid( gid )->gr_name;
} void do_ls( char* filename ) {
struct stat fileinfo;
if( stat( filename, &fileinfo ) == - ) {
printf( "%s open failure\n", filename );
exit( - );
}
//printf( "st_mode = %d\n", fileinfo.st_mode );
show_filetype( filename, fileinfo.st_mode );
show_filetype2( filename, fileinfo.st_mode );
char file_permission[];
mode_to_letters( fileinfo.st_mode, file_permission );
printf( "%s\n", file_permission );
printf( "用户:%s\n", uid_to_name( fileinfo.st_uid ) );
printf( "组:%s\n", gid_to_name( fileinfo.st_gid ) );
} //掩码判断文件类型
void show_filetype( char* filename, int filemode ){
//用st_mode的值跟0170000这个掩码相位与的结果 判断文件类型
if ( ( filemode & ) == ){
printf( "%s是普通文件\n", filename );
}else if( ( filemode & ) == ){
printf( "%s是目录\n", filename );
}else if ( ( filemode & S_IFMT ) == S_IFLNK ){
printf( "%s是符号链接\n", filename );
}
} //用宏判断文件类型
void show_filetype2( char* filename, int filemode ){
if( S_ISREG( filemode ) ) {
printf( "%s是普通文件\n", filename );
}else if( S_ISDIR( filemode ) ) {
printf( "%s是目录\n", filename );
}else if( S_ISLNK( filemode ) ){
printf( "%s是符号链接\n", filename );
}
} //数字解码成字母权限位
void mode_to_letters( int filemode, char str[] ) {
strcpy( str, "----------" );
if( S_ISREG( filemode ) ) str[] = '-';
if( S_ISDIR( filemode ) ) str[] = 'd';
if( S_ISLNK( filemode ) ) str[] = 'l'; //用户权限位
if( filemode & S_IRUSR ) str[] = 'r';
if( filemode & S_IWUSR ) str[] = 'w';
if( filemode & S_IXUSR ) str[] = 'x'; //组权限位
if( filemode & S_IRGRP ) str[] = 'r';
if( filemode & S_IWGRP ) str[] = 'w';
if( filemode & S_IXGRP ) str[] = 'x'; //其他组权限位
if( filemode & S_IROTH ) str[] = 'r';
if( filemode & S_IWOTH ) str[] = 'w';
if( filemode & S_IXOTH ) str[] = 'x';
}
综合上面3个小实例,可以得到格式化比较好的ls命令版本:
/*================================================================
* Copyright (C) 2018 . All rights reserved.
*
* 文件名称:myls2.c
* 创 建 者:ghostwu(吴华)
* 创建日期:2018年01月09日
* 描 述:ls命令( version 1.2 )
*
================================================================*/ #include <stdio.h>
#include <sys/types.h>
#include <dirent.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <pwd.h>
#include <grp.h>
#include <time.h> void do_ls( char [] );
void do_stat( char* filename );
void show_list( char* filename, struct stat* statinfo );
void mode_to_letters( mode_t filemode, char str[] );
void show_time( time_t filetime );
char* format_time( char* dsttime, const char* srctime ); //用户id转名称
char* uid_to_name( uid_t uid );
//组id转名称
char* gid_to_name( gid_t gid ); int main(int argc, char *argv[])
{
if( argc == ) {
do_ls( "." );
}else {
while( --argc ) {
printf( "arg=%s\n", * ++argv );
do_ls( *argv );
}
}
return ;
} void do_ls( char dir_entry[] ) {
DIR* pDir;
struct dirent* pCurDir;
if( ( pDir = opendir( dir_entry ) ) == NULL ){
perror( "read dir" );
exit( - );
}else {
while( ( pCurDir = readdir( pDir ) ) != NULL ) {
do_stat( pCurDir->d_name );
}
closedir( pDir );
}
} //得到文件信息
void do_stat( char* filename ){
struct stat statinfo;
if ( stat( filename, &statinfo ) == - ) {
printf( "打开%s失败\n", filename );
exit( - );
}else {
show_list( filename, &statinfo );
}
} //显示文件列表
void show_list( char* filename, struct stat* statinfo ) {
mode_t st_mode = statinfo->st_mode; char str[];
mode_to_letters( st_mode, str );
printf( "%s\t", str ); printf( "%ld\t", statinfo->st_nlink ); //符号链接
printf( "%s\t\t", uid_to_name( statinfo->st_uid ) ); //用户名
printf( "%s\t", gid_to_name( statinfo->st_gid ) ); //组名
printf( "%10ld", statinfo->st_size ); //文件大小
show_time( statinfo->st_mtime ); //最后一次修改时间
printf( "\t%s", filename ); printf( "\n" );
} char* uid_to_name( uid_t uid ){
return getpwuid( uid )->pw_name;
} char* gid_to_name( gid_t gid ){
return getgrgid( gid )->gr_name;
} void mode_to_letters( mode_t filemode, char str[] ) { strcpy( str, "----------" );
if( S_ISREG( filemode ) ) str[] = '-';
if( S_ISDIR( filemode ) ) str[] = 'd';
if( S_ISLNK( filemode ) ) str[] = 'l'; //用户权限位
if( filemode & S_IRUSR ) str[] = 'r';
if( filemode & S_IWUSR ) str[] = 'w';
if( filemode & S_IXUSR ) str[] = 'x'; //组权限位
if( filemode & S_IRGRP ) str[] = 'r';
if( filemode & S_IWGRP ) str[] = 'w';
if( filemode & S_IXGRP ) str[] = 'x'; //其他组权限位
if( filemode & S_IROTH ) str[] = 'r';
if( filemode & S_IWOTH ) str[] = 'w';
if( filemode & S_IXOTH ) str[] = 'x';
} void show_time( time_t filetime ) {
struct tm* ptm;
ptm = localtime( &filetime ); int month = ptm->tm_mon + ;
int day = ptm->tm_mday;
int hour = ptm->tm_hour;
int min = ptm->tm_min; char srchour[] = "";
char srcmin[] = "";
char dsthour[] = "";
char dstmin[] = "";
sprintf( srchour, "%d", hour );
sprintf( srcmin, "%d", min );
format_time( dsthour, srchour );
format_time( dstmin, srcmin ); printf( "%4d月%4d%4s:%2s", month, day, dsthour, dstmin );
} char* format_time( char* dsttime, const char* srctime ) {
if( strlen( srctime ) < ) {
return strcat( dsttime, srctime );
}
return strcpy( dsttime, srctime );
}
总结:
1)opendir和readdir的用法
2)结构体struct dirent的应用
3)stat的用法
linux系统编程:自己动手写一个ls命令的更多相关文章
- Linux系统编程【3.2】——ls命令优化版和ls -l实现
前情提要 在笔者的上一篇博客Linux系统编程[3.1]--编写ls命令中,实现了初级版的ls命令,但是与原版ls命令相比,还存在着显示格式和无颜色标记的不同.经过笔者近两天的学习,基本解决了这两个问 ...
- Linux系统编程【2】——编写who命令
学到的知识点 通过实现who命令,学到了: 1.使用man命令寻找相关信息 2.基于文件编程 3.体会到c库函数与系统调用的不同 4.加深对缓冲技术的理解 who命令的作用 who命令的使用 在控制终 ...
- linux系统编程综合练习-实现一个小型的shell程序(一)
之前已经花了不少篇幅学习了linux系统编程的很多知识点:文件与io.进程.信号.管道,而零散的知识点,怎么能够综合的串接起来是学习的一个很重要的目的,当然最好的方式就是用所学的知识点做一个项目了,所 ...
- linux系统编程:自己动手写一个cp命令
cp命令的基本用法: cp 源文件 目标文件 如果目标文件不存在 就创建, 如果存在就覆盖 实现一个cp命令其实就是读写文件的操作: 对于源文件: 把内容全部读取到缓存中,用到的函数read 对于目标 ...
- linux系统编程综合练习-实现一个小型的shell程序(四)
上节中已经对后台作业进行了简单处理,基本上要实现的功能已经完了,下面回过头来,对代码进行一个调整,把写得不好的地方梳理一下,给代码加入适当的注释,这种习惯其实是比较好了,由于在开发的时候时间都比较紧, ...
- linux系统编程综合练习-实现一个小型的shell程序(三)
上节中已经实现了对普通命令的解析,包括输入重定向,输出重定向,管道,后台作业,这次就来执行已经解析好的命令,对应的函数为:execute_command(),首先对带有管道的命令进行执行: 比如:&q ...
- linux系统编程综合练习-实现一个小型的shell程序(二)
上节minishell当中,已经初步实现了一个简单命令的解析,这节来继续对更加复杂命令进行解析,包含:输入重定向的解析.管道行的解析.输出重定向的解析以及是否有后台作业的解析,如下: 下面对其进行实现 ...
- Linux系统编程(14)——shell常用命令
1. ls命令 ls命令是列出目录内容(ListDirectory Contents)的意思.运行它就是列出文件夹里的内容,可能是文件也可能是文件夹. "ls -l"命令已详情模式 ...
- Linux系统编程【1】——编写more命令
背景介绍 笔者知识背景 笔者接触Linux快一年了.理论知识方面:学习了操作系统基础知识,了解进程调度.内存分配.文件管理.磁盘I/O这些基本的概念. 实操方面:会使用Linux简单命令,在嵌入式系统 ...
随机推荐
- sublime text syntaxdef
http://sublimetext.info/docs/en/extensibility/syntaxdefs.html
- webstorm “Unterminated statement”
使用webstorm的时候,写console.log,或者一些其他语句的时候 偶尔会出现这种提示,不是报错,就是看着别扭,应该是写法规范问题. 解决办法: 在下面空一行就行了
- 【hyperscan】hyperscan开源了!
hyperscan开源了! 官网:https://01.org/zh/hyperscan 1. 新闻背景 当地时间10月19日,intel将它的高速正则表达式匹配引擎hyperscan开源了,版本4. ...
- Smart/400开发上手1:入门
1.介绍 Smart/400是在AS/400之上的开发平台,管理开发.运维的全生命周期. 2.设计基础 Introducing Fields Smart通过字段字典Field Dictionary来存 ...
- iOS UIFileSharingEnabled
一.让iOS App通过iTunes进行文件交换Documents 让iOS App通过iTunes进行文件交换 有一些App需要通过使用iTunes让用户上传和下载文档.要让iOS程序支持iTune ...
- POJ 2612
#include<iostream> #include<stdio.h> #include<algorithm> #define MAXN 11 using nam ...
- 写一个MySql存储过程实现房贷等额本息还款计算(另外附javascript代码)
写一个MySql存储过程实现房贷等额本息还款计算 MySql存储过程代码如下: DROP procedure IF EXISTS `calc_equal_interest_proc`; DELIMIT ...
- windows2016 安装mysql5.7
下载msi安装包,一路下一步. 安装好后,做下简单配置. 默认的my.ini和datadir在C:\ProgramData\MySQL Server 5.7下 更改默认my.ini的方法为修改注册表 ...
- (转)kafka实战教学
转载自:https://www.cnblogs.com/hei12138/p/7805475.html Apache kafka 工作原理介绍-----https://www.ibm.com/deve ...
- ES6常用语法总结
ECMAScript 6(以下简称ES6)是JavaScript语言的下一代标准.因为当前版本的ES6是在2015年发布的,所以又称ECMAScript 2015.也就是说,ES6就是ES2015.虽 ...