linux C/C++ struct stat示例

前言

struct stat这个结构体是用来描述一个linux系统文件系统中的文件属性的结构。stat函数获取文件的所有相关信息,一般情况下,我们关心文件大小和创建时间、访问时间、修改时间。

struct stat 结构体介绍

首先还是先来所用到的struct stat结构体函数原型:

int stat(const char *path, struct stat *buf);
int lstat(const char *path, struct stat *buf);

这些两个函数返回关于文件的信息。两个函数的第一个参数都是文件的路径,第二个参数是struct stat的指针。返回值为0,表示成功执行。
如果这些失败,则下面的信息设置为:

EBADF: 文件描述词无效

EFAULT: 地址空间不可访问

ELOOP: 遍历路径时遇到太多的符号连接

ENAMETOOLONG:文件路径名太长

ENOENT:路径名的部分组件不存在,或路径名是空字串

ENOMEM:内存不足

ENOTDIR:路径名的部分组件不是目录 

这两个方法区别在于stat没有处理字符链接(软链接)的能力,如果一个文件是符号链接,stat会直接返回它所指向的文件的属性;而lstat返回的就是这个符号链接的内容。

struct stat 信息如下:

struct stat {

        mode_t     st_mode;       //文件对应的模式,文件,目录等
        ino_t      st_ino;       //inode节点号
        dev_t      st_dev;        //设备号码
        dev_t      st_rdev;       //特殊设备号码
        nlink_t    st_nlink;      //文件的连接数
        uid_t      st_uid;        //文件所有者
        gid_t      st_gid;        //文件所有者对应的组
        off_t      st_size;       //普通文件,对应的文件字节数
        time_t     st_atime;      //文件最后被访问的时间
        time_t     st_mtime;      //文件内容最后被修改的时间
        time_t     st_ctime;      //文件状态改变时间
        blksize_t st_blksize;    //文件内容对应的块大小
        blkcnt_t   st_blocks;     //伟建内容对应的块数量
      };

定义以下POSIX宏来使用st_mode字段检查文件类型:

S_ISREG(m) 是常规文件吗?
S_ISDIR (m) 目录吗?
S_ISCHR (m) 字符设备?
S_ISBLK (m) 块设备?
S_ISFIFO(m)  FIFO(命名管道)?
S_ISLNK (m) 符号链接呢? (不是posix . 1的授权- 1996)。
S_ISSOCK (m) 套接字? (不是posix . 1的授权- 1996)

st_mode字段定义了以下标志:

文件类型位字段的S_IFMT 0170000 位掩码
S_IFSOCK 0140000 套接字
S_IFLNK 0120000 符号链接
S_IFREG 0100000 常规文件
S_IFBLK 0060000 块设备
S_IFDIR 0040000 目录
S_IFCHR 0020000 字符设备
S_IFIFO 0010000 先进先出
设置UID位S_ISUID 0004000
S_ISGID 0002000 集组id位(见下)
S_ISVTX 0001000 粘性位(见下)
S_IRWXU 00700 文件所有者权限掩码

struct stat 示例

下面的程序调用stat()并在返回的stat结构中显示所选字段。

stat.c

#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>
#include <stdio.h>
#include <stdlib.h>

//int stat(const char *path, struct stat *buf);


int main(int argc, char *argv[])
{

	   struct stat sb;

	   if (argc != 2) 
	   {
	       fprintf(stderr, "Usage: %s <pathname>\n", argv[0]);
	       exit(EXIT_FAILURE);
	   }

	   if (stat(argv[1], &sb) == -1) 
	   {
	       perror("stat");
	       exit(EXIT_SUCCESS);
	   }

	   printf("文件类型:                ");

	   switch (sb.st_mode & S_IFMT) //st_mode进行”&”操作,从而就可以得到某些特定的信息。
	   {
		   case S_IFBLK:  printf("块设备驱动\n");    
		   		break;
		   case S_IFCHR:  printf("字符设备\n");      
		   		break;
		   case S_IFDIR:  printf("目录\n");          
		   		break;
		   case S_IFIFO:  printf("先入先出/管道\n");  
		   		break;
		   case S_IFLNK:  printf("创建符号链接\n");                 
		   		break;
		   case S_IFREG:  printf("普通文件\n");            
		   		break;
		   case S_IFSOCK: printf("套接字\n");                  
		   		break;
		   default:       printf("未知数?\n");                
		   		break;
	   }

	   printf("索引节点号:            %ld\n", (long) sb.st_ino);

	   printf("Mode:                     %lo (octal)\n",(unsigned long) sb.st_mode);

	   printf("链接数:               %ld\n", (long) sb.st_nlink);
	   printf("所有权:                UID=%ld   GID=%ld\n",(long) sb.st_uid, (long) sb.st_gid);

	   printf("首选I/O块大小: %ld bytes\n",(long) sb.st_blksize);
	   printf("文件大小:                %lld bytes\n",(long long) sb.st_size);
	   printf("块分配:         %lld\n",(long long) sb.st_blocks);

	   printf("最后状态更改:       %s", ctime(&sb.st_ctime));
	   printf("最后的文件访问:         %s", ctime(&sb.st_atime));
	   printf("最后的文件修改:   %s", ctime(&sb.st_mtime));

	   exit(EXIT_SUCCESS);

}

编译结果:
在这里插入图片描述

接着咱在看看shell stat 命令打印的信息。跟stat.c函数做对比
在这里插入图片描述

可以看出stat.c打印的信息跟shell stat命令打印的信息都是一样的。

在这里插入图片描述

扫二维码关注微信公众号,获取技术干货

  • 7
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个使用UDT(UDP数据报传输)进行文件传输的完整示例: 发送端(client)代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <arpa/inet.h> #include <fcntl.h> #include <sys/stat.h> #define BUF_SIZE 1460 typedef struct { int type; int seq; char data[BUF_SIZE]; } packet; int main(int argc, char* argv[]) { if (argc != 4) { printf("Usage: %s <server_ip> <server_port> <file_name>\n", argv[0]); return 1; } // 创建UDP套接字 int sock = socket(AF_INET, SOCK_DGRAM, 0); if (sock == -1) { perror("socket"); return 1; } // 设置套接字为非阻塞模式 int flags = fcntl(sock, F_GETFL, 0); fcntl(sock, F_SETFL, flags | O_NONBLOCK); // 构造服务器地址结构体 struct sockaddr_in serv_addr; memset(&serv_addr, 0, sizeof(serv_addr)); serv_addr.sin_family = AF_INET; serv_addr.sin_addr.s_addr = inet_addr(argv[1]); serv_addr.sin_port = htons(atoi(argv[2])); // 打开文件并获取文件大小 int fd = open(argv[3], O_RDONLY); if (fd == -1) { perror("open"); return 1; } struct stat st; if (fstat(fd, &st) == -1) { perror("fstat"); return 1; } int file_size = st.st_size; // 发送文件大小 packet pkt; pkt.type = 0; pkt.seq = 0; memcpy(pkt.data, &file_size, sizeof(file_size)); sendto(sock, &pkt, sizeof(pkt), 0, (struct sockaddr*)&serv_addr, sizeof(serv_addr)); // 发送文件数据 int seq = 1; while (1) { // 读取数据 int nread = read(fd, pkt.data, BUF_SIZE); if (nread == -1) { if (errno == EAGAIN) { // 暂时没有数据可读,等待下一次读取 continue; } else { perror("read"); return 1; } } else if (nread == 0) { // 文件读取完毕,退出循环 break; } // 发送数据 pkt.type = 1; pkt.seq = seq++; sendto(sock, &pkt, sizeof(pkt), 0, (struct sockaddr*)&serv_addr, sizeof(serv_addr)); } // 关闭套接字和文件 close(sock); close(fd); return 0; } ``` 接收端(server)代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <arpa/inet.h> #include <fcntl.h> #include <sys/stat.h> #define BUF_SIZE 1460 typedef struct { int type; int seq; char data[BUF_SIZE]; } packet; int main(int argc, char* argv[]) { if (argc != 2) { printf("Usage: %s <port>\n", argv[0]); return 1; } // 创建UDP套接字 int sock = socket(AF_INET, SOCK_DGRAM, 0); if (sock == -1) { perror("socket"); return 1; } // 绑定端口 struct sockaddr_in addr; memset(&addr, 0, sizeof(addr)); addr.sin_family = AF_INET; addr.sin_addr.s_addr = htonl(INADDR_ANY); addr.sin_port = htons(atoi(argv[1])); if (bind(sock, (struct sockaddr*)&addr, sizeof(addr)) == -1) { perror("bind"); return 1; } // 设置套接字为非阻塞模式 int flags = fcntl(sock, F_GETFL, 0); fcntl(sock, F_SETFL, flags | O_NONBLOCK); // 接收文件大小 int file_size = 0; packet pkt; while (1) { // 接收数据 int nread = recv(sock, &pkt, sizeof(pkt), 0); if (nread == -1) { if (errno == EAGAIN) { // 暂时没有数据可读,等待下一次读取 continue; } else { perror("recv"); return 1; } } // 处理数据 if (pkt.type == 0) { // 接收到文件大小 memcpy(&file_size, pkt.data, sizeof(file_size)); break; } } // 打开文件并准备写入数据 int fd = open("output.dat", O_WRONLY | O_CREAT | O_TRUNC, 0644); if (fd == -1) { perror("open"); return 1; } int bytes_written = 0; char* write_ptr = (char*)&file_size; // 接收文件数据 int expected_seq = 1; while (bytes_written < file_size) { // 接收数据 int nread = recv(sock, &pkt, sizeof(pkt), 0); if (nread == -1) { if (errno == EAGAIN) { // 暂时没有数据可读,等待下一次读取 continue; } else { perror("recv"); return 1; } } // 处理数据 if (pkt.type == 1 && pkt.seq == expected_seq) { // 接收到正确的数据包,写入数据 write(fd, pkt.data, nread - sizeof(pkt.seq) - sizeof(pkt.type)); bytes_written += nread - sizeof(pkt.seq) - sizeof(pkt.type); expected_seq++; } } // 关闭套接字和文件 close(sock); close(fd); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值