Linux李哥私房菜——open、close和fd

open()

头文件:#include<fcntl.h>//在centos6.0中只要此头文件就可以
       #include<sys/types.h>
       #incldue<sys/stat.h>
功能:打开和创建文件(建立一个文件描述符,其他的函数可以通过文
     件描述符对指定文件进行读取与写入的操作。)

文件描述符号fd详解:

  1. 概述
    在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件、目录文件、链接文件和设备文件。文件描述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符。程序刚刚启动的时候,0是标准输入,1是标准输出,2是标准错误,POSIX标准要求每次打开文件时(含socket)必须使用当前进程中最小可用的文件描述符号码,因此,在网络通信过程中稍不注意就有可能造成串话,所以,如果此时去打开一个新的文件,它的文件描述符会是3(标准输入输出错误也是文件,提前把前三个坑抢了)。标准文件描述符图如下:
    在这里插入图片描述

总结:

fd定义:每一个进程在内核中,会有一个“打开文件”表数组,每一个元素都指向一个内核的打开文件对象,而fd就是数组的下标,我们对文件进行操作时,系统调用,将fd传入内核,内核通过fd找到文件,对文件进行操作,
虽然只是下标,但是内核根据下标操作文件,所以fd用于指代被打开的文件,由于这个表处于内核,并且用户无法访问到,因此用户即使拥有fd,也无法得到打开文件对象的地址,只能够通过系统提供的函数来操作(比如write和read)。

既然是数组下标,fd的类型为int, < 0 为非法值, >=0 为合法值。在linux中,一个进程默认可以打开的文件数为1024个,fd的范围为0~1023。可以通过设置,改变最大值。

通常在程序中打开的fd,是从3开始的。我们在判断一个fd是否合法时,依然要使用>=0的判断标准

内核对所有打开的文件的文件维护有一个系统级的描述符表格(open file description table)。有时,也称之为打开文件表(open file table),并将表格中各条目称为打开文件句柄(open file handle)。一个打开文件句柄存储了与一个打开文件相关的全部信息,如下所示:
1. 当前文件偏移量(调用read()和write()时更新,或使用lseek()直接修改)
2. 打开文件时所使用的状态标识(即,open()的flags参数)
3. 文件访问模式(如调用open()时所设置的只读模式、只写模式或读写模式)
4. 与信号驱动相关的设置
5. 对该文件i-node对象的引用
6. 文件类型(例如:常规文件、套接字或FIFO)和访问权限
7. 一个指针,指向该文件所持有的锁列表
8. 文件的各种属性,包括文件大小以及与不同类型操作相关的时间戳
下图展示了文件描述符、打开的文件句柄以及i-node之间的关系,图中,两个进程拥有诸多打开的文件描述符。

这里是引用

int open(const char*pathname,int flags);
int open(const char*pathname,int flags,mode_t mode);
参数说明:
1.pathname
  要打开或创建的目标文件
2.flags
  打开文件时,可以传入多个参数选项,用下面的
  一个或者多个常量进行“或”运算,构成falgs
  参数:
  O_RDONLY:   只读打开
  O_WRONLY:   只写打开
  O_RDWR:     读,写打开
这三个常量,必须制定一个且只能指定一个
  O_CREAT:    若文件不存在,则创建它,需要使
              用mode选项。来指明新文件的访问权限
  O_APPEND:   追加写,如果文件已经有内容,这次打开文件所
              写的数据附加到文件的末尾而不覆盖原来的内容


返回值

成功:新打开的文件描述符
失败:-1
open返回的文件描述符一定是最小的而且没有被使用的

fopen与open的区别

以可写的方式fopen一个文件时,如果文件不存在则会自动创建,而open一个文件时必须明确O_CREAT才会创建文件,否则文件不存在就出错返回

close

头文件:#include<unistd.h>
功能:关闭一个已经打开的文件

原型

int close(int fd)
参数说明:
fd:是需要关闭的文件描述符

返回值

成功:返回0;
失败:返回-1,并设置errno

打开的文件描述符一定要记得关闭,否则资源会被大量的占用,导致内存不够

open打开的文件存在时:
示例:
1.
代码:

#include<stdio.h>
#include<string.h>
#include<fcntl.h>
#include<unistd.h>
#include<stdlib.h>
int main()
{
 int fd=open("myfile",O_WRONLY);
 if(fd<0)
 {
  perror("open");
  exit(1);
 }
 const char*msg="hello open\n";
 int count = 6;
 while(count--)
 {
  write(fd,msg,strlen(msg));
 }
 close(fd);
 return 0;
}

运行结果:
在这里插入图片描述
2.
代码:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<fcntl.h>
#include<unistd.h>
int main()
{
 int fd=open("myfile",O_RDWR);
 if(fd<0)
 {
  perror("open");
  exit(1);
 }
 const char*msg="hello  hahaha\n";
 int count= 10;
 while(count--)
 {
 write(fd,msg,strlen(msg));
 }
 char buf[1024]={0};
 int num=10;
 while(num--)
 {
 read(fd,buf,strlen(msg));
 }
 close(fd);
 return 0;
}

运行结果:
这里写图片描述
open打开的文件不存在时:

代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<fcntl.h>
int main()
{
 int fd=open("file",O_WRONLY|O_CREAT,0644);
 //file文件不存在,所以在书写第二个参数时,记得把O_CREAT加上,
 //如果不加O_CREAT的话,程序就会报此文件不存在
 if(fd<0)
 {
  perror("open");
  exit(1);
 }
 const char*msg="hello file\n";
 int count=10;
 while(count--)
 {
  write(fd,msg,strlen(msg));
 }
 close(fd);
 return 0;
}

运行结果:
在这里插入图片描述

总结:

open函数会返回一个fd,fd是一个整数类型的数组下标,,系统函数可以通过fd可以对文件进行操作。这个数组中的指针指向打开文件表,表中的每一行都描述了一个文件的属性,包括文件偏移量,i-node指针和打开模式。文件偏移量会根据write和read时更新(read和write函数成功调用时会返回这一次调用所读写的字节长度)。

参考:
https://blog.csdn.net/sybnfkn040601/article/details/73718332?utm_medium=distribute.pc_relevant.none-task-blog-baidujs_title-0&spm=1001.2101.3001.4242

https://blog.csdn.net/dangzhangjing97/article/details/79631173

他们都告诉你,必须要做什么,却没告诉你为什么。
##《 欢迎访问我的网站,ai工具箱https://4398ai.com里面有免费的chatgpt网站,和很多免费的编程资源的干货》
在这里插入图片描述

### 关于深度学习图片分类的教程资源 #### 神经网络基础与应用实例 对于初学者来说,了解神经网络的基础知识至关重要。专项课程中的第一门课涵盖了神经网络的核心概念以及构建方法,包括如何创建并训练一个用于识别猫的小型深度神经网络案例[^1]。 #### 传统机器学习对比 在探讨深度学习之前,理解其相对于传统机器学习的优势也很重要。传统上,为了使计算机能够识别特定对象(如狗),开发者需手动定义一系列复杂的特征提取过程;而在现代深度学习框架下,则无需如此繁琐的人工干预,模型可以直接从原始像素值中自动学到有用的表示形式[^2]。 #### 实际操作指南 当准备动手实践时,可以从加载预训练好的GoogLeNet模型入手。这不仅提供了快速入门的机会,而且有助于熟悉常用的工具技术栈。具体而言,在MATLAB环境中启动“深度网络设计器”,并通过该平台导入所需的架构文件[^3]。 #### 数据预处理的重要性 值得注意的是,在正式开始训练前还应重视数据的质量控制环节。特别是针对图像类任务,建议利用Python库PIL来进行初步筛选——任何无法正常打开或是存在明显缺陷的照片都应当被排除在外,以免影响最终效果[^4]。 ```python from PIL import Image def check_image_quality(image_path): try: img = Image.open(image_path) # 尝试打开图片 img.verify() # 验证图片完整性 return True # 如果成功则返回True except (IOError, SyntaxError) as e: print(f'Bad file: {image_path}') # 打印错误信息 return False # 否则返回False ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值