Linux 编写ls -l 命令

ls -l 命令的输出

每行都包含7个字段:


          1.模式(mode)                       每一行的第一个字符表示文件类型。“-”代表普通文件,“d”代表目录。接下来的9个字符表示文件访问权限,

                                                              分为读、写和执行,又分别针对三种对象:用户、同组用户和其他用户,所以一共需要9位来表示。从前面

                                                               的ls -l 的输出可以看到,所有的文件和目录对所有用户都是可读的,只有文件的所有者才可以对文件进行

                                                               修改,所有用户都有taill的权限。

           2.链接数(links)                    该文件被引用的次数。

           3.文件所有者(owner)         指出文件所有者的用户名。

           4.组(group)                          指文件所有者所在的组。有些版本的ls显示组名。

           5.大小(size)                         文件大小

           6.最后修改时间(last-modified)

           7.文件名(name)


ls01.c代码

/**
* ls01.c
* purpose list contents of directory or directories
* action if no args, use. else list files in args
* note uses stat and pwd.h and grp.h
* BUG : try ./ls01 /tmp
*/
#include<stdio.h>
#include<sys/types.h>
#include<dirent.h>
#include<sys/stat.h>
#include<grp.h>
#include<pwd.h>

void do_ls(char *dirname);
void dostat(char *filename);
void show_file_info(char *filename, struct stat *info_p);
void mode_to_letters(int mode, char *str);
char *uid_to_name(uid_t uid);
char *gid_to_name(gid_t uid);

int main(int ac, char **av)
{
	if(ac == 1)
	{
		do_ls(".");
	}
	else
	{
		while(--ac)
		{
			printf("%s:\n", *++av);
			do_ls( *av );
		}
	}
}
/**
* list files in directory called dirname
*/
void do_ls(char *dirname)
{
	DIR *dir_ptr;           // the directory
	struct dirent *direntp; // each entry
	
	if( (dir_ptr = opendir(dirname)) == NULL )
	{
		fprintf(stderr, "ls01: cannot %s\n", dirname);
	}
	else
	{
		while( (direntp = readdir(dir_ptr)) != NULL )
		{
			dostat(direntp->d_name);
		}
		closedir(dir_ptr);
	}
}

void dostat(char *filename)
{
	struct stat info;

	if(stat(filename, &info) == -1) // cannot stat
	{
		perror(filename);       // say why;
	}
	else  // else show information
	{
		show_file_info(filename, &info);
	}
}

/**
* display the information. The information is stored in stuct at *info_p
*/
void show_file_info(char *filename, struct stat *info_p)
{
	char *uid_to_name();
	char *ctime();
	char *gid_to_name();
	char *filemode();
	char modestr[11];

	mode_to_letters(info_p->st_mode, modestr);

	printf("%s", modestr);
	printf("%4d ", (int)info_p->st_nlink);
	printf("%-8s", uid_to_name(info_p->st_uid) );
	printf("%-8s", gid_to_name(info_p->st_gid) );
	printf("%8ld ", (long)info_p->st_size);
	printf("%.12s ", 4 + ctime(&info_p->st_mtime));
	printf("%s\n", filename);
}

/**
* utility function
*/

/**
* This function takes a mode value and a char array
* and puts into the char array the file type and the
* nine letters that correspond to the bits in mode.
* Note: It dose not code setuid, setgid, and sticky
* codes
*/
void mode_to_letters(int mode, char *str)
{
	strcpy(str, "----------");         // defult = no perms
	
	if( S_ISDIR(mode) ) str[0] = 'd'; // directory
	if( S_ISCHR(mode) ) str[0] = 'c'; // char devices
	if( S_ISBLK(mode) ) str[0] = 'b'; // block device

	if( mode & S_IRUSR) str[1] = 'r'; // 3 bits for user
	if( mode & S_IWUSR) str[2] = 'w';
    if( mode & S_IXUSR) str[3] = 'x';

    if( mode & S_IRGRP) str[4] = 'r'; // 3 bits for group
	if( mode & S_IWGRP) str[5] = 'w';
	if( mode & S_IXGRP) str[6] = 'x';


    if( mode & S_IROTH) str[7] = 'r'; // 3 bits for other
	if( mode & S_IWOTH) str[8] = 'w';
	if( mode & S_IXOTH) str[9] = 'x';
}
/**
* return pointer to username asscociated with uid, uses getpw()
*/
char *uid_to_name(uid_t uid)
{
	struct passwd *getpwuid();
	struct passwd *pw_ptr;
	static char numstr[10];

	if( (pw_ptr = getpwuid(uid)) == NULL )
	{
		sprintf(numstr, "%d", uid);
		return numstr;
	}
	else
	{
		return pw_ptr->pw_name;
	}
}

/**
* return pointer to group number gid. used getgrgid(3)
*/
char *gid_to_name(gid_t gid)
{
	struct group *getgrgid();
	struct group *grp_ptr;
	static char numstr[10];

	if( (grp_ptr = getgrgid(gid)) == NULL )
	{
		sprintf(numstr, "%d", gid);
		return numstr;
	}
	else
	{
		return grp_ptr->gr_name;
	}
}

编译。执行。

$ ./ls01


OK

不过,它不能指定参数 -a   -l 等,不能指定目录。

参考《Unix/linux编程实践指南》


  • 0
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要在Linux下用Java编写ls -l命令的话,需要先熟悉Java的文件操作和系统命令执行相关的API调用,然后实现类似于执行 ls -l 命令并输出结果的逻辑代码。虽然实现起来并不简单,但是有了Java强大的功能库,还是可以实现的。 ### 回答2: 在仿Linux下的ls-l命令Java时,我们需要利用Java的文件IO操作和格式化输出功能。 首先,我们需要获取指定目录下的所有文件,并且获取每个文件的属性信息。Java提供了File类来实现这一功能,我们可以使用File类的listFiles()方法来获取目录下所有的文件,然后遍历每个文件,获取文件的属性信息。 获取文件属性信息有很多方式,常见的有使用Java的NIO包中的Files类或者使用Java的File类。在这里,我们可以使用File类提供的lastModified()方法获取文件的最后修改时间,使用length()方法获取文件的大小(以字节为单位),以及使用isDirectory()方法获取文件是否是目录。 获取完文件的属性信息后,我们需要将它们格式化输出。Java提供了很多实用的格式化输出方法,其中最常用的是String.format()方法。使用String.format()方法,我们可以指定输出的格式,将文件的属性信息输出成一行,并且和Linux下的ls-l命令输出的格式一致。 下面是一个示例代码: import java.io.File; import java.text.SimpleDateFormat; import java.util.Date; public class LsLCommand { public static void main(String[] args) { File dir = new File(args[0]); File[] files = dir.listFiles(); for (File file : files) { String name = file.getName(); String permissions = ""; String owner = System.getProperty("user.name"); long size = file.length(); Date modified = new Date(file.lastModified()); if (file.isDirectory()) { permissions += "d"; } else { permissions += "-"; } permissions += file.canRead() ? "r" : "-"; permissions += file.canWrite() ? "w" : "-"; permissions += file.canExecute() ? "x" : "-"; String pattern = "MMM dd HH:mm"; SimpleDateFormat format = new SimpleDateFormat(pattern); String output = String.format("%s %3d %s %s %6d %s %s", permissions, 1, owner, owner, size, format.format(modified), name); System.out.println(output); } } } 这里使用了SimpleDateFormat类来格式化输出文件的最后修改时间,输出格式为“MMM dd HH:mm”。同时,为了和Linux下的ls-l命令输出的格式一致,这里的代码输出了每个文件的权限、硬链接数、拥有者、分组、大小、最后修改时间和文件名。 最后,我们可以通过命令行参数传入要遍历的目录路径,运行该程序,便可以得到该目录下所有文件的权限、大小、最后修改时间及文件名等信息,效果和Linux下的ls-l命令非常相似。 ### 回答3: 1. 了解需求:我们要仿的是Linux下的ls-l命令,即以长格式显示目录下的文件和目录信息,并包括文件的权限、所有者、组、大小和修改时间等信息。 2. 分析实现方式:我们可以使用Java的File类,来获取目录下的全部子文件和子目录信息,再使用Java的DateFormat类,来格式化文件的修改时间信息。 3. 实现步骤: (1)使用File类获取目录下的全部子文件和子目录信息,方法为:File[] files = new File(directory).listFiles(); (2)使用DateFormat类来格式化文件的修改时间信息,方法为:DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); (3)使用for循环遍历目录下的全部子文件和子目录信息,并输出每个文件和目录的信息: ``` for(File file : files){ String name = file.getName(); //获取文件名 String size = file.length() + "B"; //获取文件大小 String owner = Files.getOwner(file.toPath()).getName(); //获取文件所有者 String group = Files.readAttributes(file.toPath(), PosixFileAttributes.class).group().getName(); //获取文件所属组 String permission = ""; if(file.isDirectory()){ permission += "d"; }else{ permission += "-"; } if(file.canRead()){ permission += "r"; }else{ permission += "-"; } if(file.canWrite()){ permission += "w"; }else{ permission += "-"; } if(file.canExecute()){ permission += "x"; }else{ permission += "-"; } String time = df.format(new Date(file.lastModified())); //格式化文件修改时间 String info = permission + " " + owner + " " + group + " " + size + " " + time + " " + name; //输出文件信息 System.out.println(info); } ``` 4. 测试与优化:我们运行程序,并对输出的文件信息进行检查,测试是否符合ls-l命令的要求。如果发现有错误或不符合要求的地方,我们可以根据具体的情况进行代码的优化和修改。 最终,我们就可以仿一个类似于Linux下的ls-l命令的Java程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值