【嵌入式软件工程师面经】Linux文件IO

简介: 【嵌入式软件工程师面经】Linux文件IO

什么是IO操作:

       IO操作,即输入/输出操作,是指在计算机系统中数据在内存和外围设备之间传输的行为。在操作系统的上下文中,IO通常关联于文件系统的操作,以及与外设如硬盘、显示器、网络接口和键盘鼠标等的交互。

  1. 文件IO(File I/O)
  • 读取(Read): 从文件系统中的文件读取数据到内存中。
  • 写入(Write): 将数据从内存写入到文件系统的文件中。
  • 打开(Open): 打开文件以便读取或写入数据。
  • 关闭(Close): 完成文件操作后关闭文件,释放资源。
  • 定位(Seek): 移动文件内的指针到特定位置,用于读写操作。
  • 同步(Sync): 确保内存中的数据与存储设备中的数据同步。
  1. 设备IO(Device I/O)
  • 直接IO: 与具体的硬件设备进行数据传输,如磁盘、GPU等。
  • 缓冲IO: 数据首先被读取到一个内存缓冲区,然后再从缓冲区移动到最终的目的地。
  • 同步与异步IO: 同步IO要求操作完成后程序才继续运行,而异步IO允许程序在操作还在进行时继续其他任务。

一、 Linux的根目录有哪些?各有什么作用?

       在Unix和Unix-like操作系统(比如Linux)中,根目录(表示为/)是文件系统结构的最高级别。根目录下有一系列的标准子目录,虽然这些可能会根据不同的发行版有所差异,但大多数系统都遵循以下的结构和用途:

目录 用途
/bin 存放基本命令和程序,系统启动和修复时需要
/boot 启动相关文件,如内核vmlinuz,启动加载器GRUB配置文件等
/dev 包含硬件设备文件,如/dev/sda/dev/tty
/etc 系统配置文件,如启动、服务配置、用户账户设置等
/home 用户主目录,每个用户都有自己的子目录(除了root用户)
/lib 系统核心操作和文件操作所需的库文件,如C标准库、设备驱动等
/media 用于挂载可移动媒体,如CD-ROMs、USB驱动器等
/mnt 用于临时挂载文件系统,如网络文件系统NFS
/opt 第三方应用程序或额外软件包的安装位置
/proc 虚拟文件系统,映射运行时系统信息,如内存、硬件配置等
/root root用户的主目录
/run 系统运行时信息,如当前登录用户和运行中的服务
/sbin 系统管理命令,通常由系统管理员使用
/srv 服务提供的数据
/sys 虚拟文件系统,包含设备和驱动信息,内核设备树的一部分
/tmp 存储临时文件,系统重启时通常会被清理
/usr 用户安装的应用程序和文件,如/usr/bin/usr/sbin/usr/local
/var 存放经常变化的文件,如日志文件(/var/log),包和数据库(/var/lib)

二、linux下有哪几种文件

       使用ls -l命令时,列出的文件和目录会在第一列显示这些符号,从而可以快速识别它们的类型。而file命令则用于获取文件的具体类型信息,帮助用户更精确地了解文件的性质。

文件类型 表示符号 描述
普通文件 - 包括文本文件、二进制文件和数据文件。
目录文件 d 用来存储其他文件或目录的容器。
链接文件 l 硬链接和符号链接(软链接),类似快捷方式,指向其他文件。
设备文件 c, b c表示字符设备文件,b表示块设备文件。
管道文件 p 命名管道(FIFO),用于进程间通信。
套接字文件 s 用于进程间的网络通信。
特殊文件   例如/dev/null,用于废弃输出或产生空输入的特殊目的文件。

三、 系统IO和标准IO操作文件的区别?

3.1 文件类型

       标准IO打开的是一个文件指针;

       系统IO打开的是一个文件描述符。

3.1.1 标准I/O:

  • 概述:标准I/O是C语言标准库提供的一组函数,它提供了缓冲的I/O操作。这意味着当你使用标准I/O函数(如fopen, fprintf, fread, fwrite等)时,数据可能首先被存储在一个中间缓冲区,然后才实际写入磁盘或从磁盘读取。这种缓冲可以提高I/O操作的效率。
  • 文件指针:使用标准I/O打开文件时,会返回一个文件指针(通常是FILE*类型)。这个文件指针是对打开文件的抽象表示,可以用来访问和操作文件。

3.1.2 系统I/O:

  • 概述:系统I/O通常指的是操作系统提供的系统调用(如UNIX/Linux环境中的open, read, write, close等)。与标准I/O不同,系统I/O通常是非缓冲的,这意味着每次read或write调用都直接与磁盘进行交互,没有中间缓冲区。
  • 文件描述符:当使用系统I/O函数打开文件时,会得到一个文件描述符,这是一个整数值,用于标识和访问打开的文件。文件描述符是对打开文件的底层表示,可以用于后续的系统I/O操作。

3.2 系统 IO与标准IO各自优势

3.2.1 系统IO的特点:

  1. 通用性:系统IO提供了一组非常基础的系统调用,如open, read, write, close等,这些调用可以用于普通文件、设备文件(如字符设备和块设备)、管道、套接字等。这意味着系统IO对于UNIX和类UNIX操作系统上所有类型的文件描述符都是通用的。
  2. 简约性:系统IO调用通常不会在用户空间添加额外的缓冲层。这意味着当你使用read或write系统调用时,数据直接从内核缓冲区传输到你的程序缓冲区,或者反过来,没有中间缓冲处理。这个过程是透明的,没有格式化操作,确保了数据的读写是原始和未处理的。
  3. 非格式化IO:使用系统IO,数据的读写是非格式化的,即数据被视为原始字节流。程序员负责解释数据的结构和意义。

3.2.2 标准IO的特点:

  1. 缓冲处理:标准IO提供了缓冲机制,这可以提高IO操作的效率,因为它减少了程序与内核之间的系统调用次数。然而,这种缓冲处理也增加了复杂性,尤其是在需要处理同步和数据一致性时。
  2. 格式化IO:标准IO提供了格式化的输入输出函数,如printf和scanf,这些函数可以让你以一定格式读写数据,而无需手动处理数据的序列化和反序列化。
  3. 抽象层:标准IO通过FILE结构体提供了一个文件的高级抽象,这使得读写文件更加简单,但也隐藏了底层细节。

3.3 系统IO和标准IO缺省打开的三个标准文件的方式?

       在Unix和类Unix系统中,当一个新的程序开始执行时,系统IO和标准IO都会默认打开三个特殊的文件,这些文件是标准输入(stdin)、标准输出(stdout)和标准错误(stderr)。这三个文件的行为在系统IO和标准IO中是一致的,但它们的表示形式不同。

3.3.1 系统IO (底层IO):

在系统IO中,这三个文件分别与三个特定的文件描述符相关联:

  • 标准输入:文件描述符0
  • 标准输出:文件描述符1
  • 标准错误:文件描述符2

这些文件描述符是在程序启动时由操作系统自动打开的,并且它们通常连接到终端(也就是命令行界面)。

3.3.2 标准IO (高级IO):

在使用标准IO时,这三个标准流以流(FILE*)的形式存在:

  • 标准输入:stdin
  • 标准输出:stdout
  • 标准错误:stderr

这些流是标准C库自动提供的,并且已经在程序启动时预定义好了。

3.3.3 打开方式:

  • 标准输入(stdin):默认情况下,标准输入是为读操作打开的,它读取来自用户的输入,通常是来自键盘。
  • 标准输出(stdout):默认情况下,标准输出是为写操作打开的,用于向用户显示文本,通常是发送到屏幕。
  • 标准错误(stderr):标准错误同样是为写操作打开的,并且它用于输出错误信息或诊断信息,通常错误信息也会被发送到屏幕。

       对于标准IO,不需要打开这些流;它们已经是打开状态,并且准备好供您的程序使用。对于系统IO,即使这些文件描述符也是预先打开的,仍然可以使用相关的系统调用,如read和write,通过这些文件描述符与这些文件进行交云。

标准流 系统IO描述符 标准IO流 默认打开方式 默认设备
标准输入(stdin) 0 stdin 只读 键盘
标准输出(stdout) 1 stdout 只写 屏幕
标准错误(stderr) 2 stderr 只写 屏幕
  • 系统IO中的文件描述符是小的非负整数。
  • 标准IO流是FILE*类型的指针,其具体类型在C语言的标准库中定义。
  • 默认打开方式列出了这些文件的访问模式,标准输入是读,标准输出和错误是写。
  • 默认设备指的是这些流通常所连接的设备,通常是键盘和屏幕(终端或控制台)。

四、 什么是空洞文件?谈一谈对空洞文件的理解。

       空洞文件(Sparse file)是文件系统中的一种文件,其中包含了一定量的未使用的空间,这部分空间在磁盘上不占用物理位置,即使文件的大小看起来很大。这些未使用的空间被称作“空洞”。在大多数文件系统中,当读取文件中的这些空洞时,它们表现得就像读取了一串零。

       空洞通常在文件中出现,因为数据被写入到文件的不连续区域。例如,有些程序可能会根据需要将信息设置或写入文件中特定的位置,并跳过文件的其他部分。如果程序在文件的开始处写入数据,然后跳到接近文件末尾的位置继续写入,它之间的部分就形成了空洞。

4.1 空洞文件的特点:

       4.1.1 物理和逻辑大小:空洞文件的逻辑大小(文件的大小,如用ls -l查看)可能会很大,但它的物理大小(即实际占用的磁盘空间,可以用du命令查看)可能相当小。

       4.1.2 节省空间:空洞文件的优点在于它可以在不实际使用相同数量磁盘空间的情况下创建大文件。这对于某些特定的应用程序来说是非常有用的,例如数据库和虚拟机映像,它们可能需要创建大型文件,但实际上只使用了文件的一小部分。

       4.1.3 透明使用:对于操作系统和应用程序,空洞文件的使用是透明的。当程序读取文件中的空洞部分时,文件系统会返回一个数据全零的缓冲区,程序通常无需关心这些零是实际存储在磁盘上还是由空洞生成的。

4.2 创建空洞文件的方法:

       在Unix和类Unix系统中,可以通过使用lseek系统调用在文件中创建空洞。例如,一个程序可以打开一个文件,写入一些数据,然后使用lseek跳过一段距离,再写入更多数据。跳过的部分就会形成一个空洞。

4.3 空洞文件的局限性:

尽管空洞文件节省了空间,但它们也有一些潜在的局限性和缺点:

  • 某些文件系统或存储设备可能不支持空洞文件。
  • 当空洞文件被复制或转移到不支持空洞的文件系统时,所有的空洞都将被零填充,这将导致文件占用完整的物理空间。
  • 在涉及大量空洞的文件中进行随机读写可能会影响性能。

五、其他常见问题

5.1 解释Linux文件描述符。

       在Linux中,文件描述符是一个非常小的正整数,用于表示一个打开的文件。每个进程都有一个文件描述符表,用于跟踪打开的文件。

5.2 Linux系统调用和库函数之间有什么区别?

       系统调用(如open, read, write, close)提供了操作系统级别的文件IO操作,直接与内核交云,性能较高但使用复杂。

       库函数(如fopen, fread, fwrite, fclose)是建立在系统调用之上的封装,提供了更加用户友好的接口,自动处理错误和缓冲。

5.3 什么是标准输入、标准输出和标准错误?

       这些是三个预定义的文件描述符,分别是标准输入(stdin, 0)、标准输出(stdout, 1)和标准错误(stderr, 2),它们默认分别连接到键盘和终端。

5.4 解释open系统调用的作用。

       open系统调用用于打开一个文件,如果需要也可以创建一个文件,它返回一个文件描述符,用于后续的读写操作。

5.5 read和write系统调用如何使用?

       read用于从一个已打开的文件描述符中读取数据,而write用于向一个已打开的文件描述符写入数据。

5.6 什么是文件权限和所有权?

       文件权限决定了用户对文件的读、写、执行权限,而所有权则指文件属于哪个用户和用户组。

5.7 解释Linux中的软链接和硬链接。

       软链接(或称符号链接)类似于Windows的快捷方式,是指向另一个文件的特殊文件。

       硬链接是文件系统中另一个指向相同inode的名称,与原始文件无法区分。

5.8 什么是inode?

       inode是文件系统中的一个数据结构,包含了关于文件的元数据,如文件大小、权限、所有权和数据位置等,但不包含文件名。

5.9 解释lseek的功能。

       lseek可以重新定位文件描述符的读写文件偏移,允许随机访问文件中的任何位置。

5.10 什么是文件锁定?

       文件锁定是一种机制,可以防止多个进程同时写入或读写同一个文件,从而避免数据损坏或不一致。

5.11 如何检测和处理文件IO错误?

       文件IO操作通常在错误时返回特定的值(如-1),并设置errno变量以指示具体错误。程序应该检查IO操作的返回值,并适当地处理错误情况。

5.12 解释fsync和fdatasync。

       这两个调用用于将缓冲的数据写入硬盘,以确保数据的持久化。fsync同时刷新文件的元数据,而fdatasync只刷新文件数据,不包括元数据。

5.13 什么是直接IO?

       直接IO绕过操作系统的缓冲,直接在应用程序和硬盘之间传输数据,这可以减少延迟并提高大型连续数据传输的性能。

5.14 如何处理大文件(大于4GB)?

       为了处理大文件,应该使用支持64位偏移量的函数(如lseek64),并确保文件系统和操作系统都支持大文件处理。

5.15 解释阻塞与非阻塞IO的区别。

       阻塞IO指的是IO操作会使得进行该操作的进程挂起,直到操作完成。而非阻塞IO则是IO操作会立即返回,不会等待操作完成。

5.16 什么是多路复用IO?

       多路复用IO允许一个线程同时监视多个文件描述符,以检查一个或多个文件描述符是否就绪(可读、可写、有错误等)。常见的多路复用IO实现有select, poll, epoll。

5.17 解释mmap系统调用的用途。

       mmap允许程序将一个文件或其他对象映射到内存地址空间,这可以提供一种访问文件的高效方法,特别是对于频繁的随机访问。

5.18 如何确定文件的大小?

       可以使用stat或fstat系统调用获取文件的状态信息,其中包括文件大小。

5.19 讨论同步IO与异步IO的区别。

       同步IO要求发起IO操作的进程等待IO操作完成才能继续执行。异步IO则允许进程继续执行,并在IO操作完成时通过信号、回调或其他机制通知进程。

5.20 ioctl系统调用是用来做什么的?

       ioctl是设备独立的系统调用,用来与设备驱动程序进行交互,可以用来执行设备特定的操作,如格式化、设置硬件参数等。

5.21 什么是umask?

       umask是一个进程属性,用于设置默认的文件权限。当创建新文件时,umask会从完全权限中减去,以得到新文件的权限。

5.22 解释O_SYNC和O_ASYNC标志。

       这些是open系统调用的标志。O_SYNC用于同步IO,使得每次write都等待数据物理写入磁盘。O_ASYNC使得在文件可读或可写时,操作系统会向进程发送一个信号。

5.23 Linux中有哪些IO模型?

       Linux支持多种IO模型,包括阻塞IO、非阻塞IO、IO多路复用、信号驱动IO和异步IO。

5.24 如何控制文件的缓存行为?

       可以通过调用posix_fadvise和madvise函数来给操作系统建议如何处理文件或内存映射文件的缓存。

5.25 如何检测文件类型?

       可以使用stat或fstat系统调用并检查返回的结构中的st_mode字段,以确定文件是普通文件、目录、字符设备、块设备还是管道。

目录
相关文章
|
13天前
|
Ubuntu Linux Shell
(已解决)Linux环境—bash: wget: command not found; Docker pull报错Error response from daemon: Get https://registry-1.docker.io/v2/: net/http: request canceled
(已成功解决)Linux环境报错—bash: wget: command not found;常见Linux发行版本,Linux中yum、rpm、apt-get、wget的区别;Docker pull报错Error response from daemon: Get https://registry-1.docker.io/v2/: net/http: request canceled
178 68
(已解决)Linux环境—bash: wget: command not found; Docker pull报错Error response from daemon: Get https://registry-1.docker.io/v2/: net/http: request canceled
|
3月前
|
存储 Linux C语言
Linux C/C++之IO多路复用(aio)
这篇文章介绍了Linux中IO多路复用技术epoll和异步IO技术aio的区别、执行过程、编程模型以及具体的编程实现方式。
122 1
Linux C/C++之IO多路复用(aio)
|
2天前
|
Ubuntu Linux 开发者
Ubuntu20.04搭建嵌入式linux网络加载内核、设备树和根文件系统
使用上述U-Boot命令配置并启动嵌入式设备。如果配置正确,设备将通过TFTP加载内核和设备树,并通过NFS挂载根文件系统。
31 15
|
5天前
|
Linux API C语言
Linux基础IO
Linux基础IO操作是系统管理和开发的基本技能。通过掌握文件描述符、重定向与管道、性能分析工具、文件系统操作以及网络IO命令等内容,可以更高效地进行系统操作和脚本编写。希望本文提供的知识和示例能帮助读者更深入地理解和运用Linux IO操作。
34 14
|
16天前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
79 13
|
2月前
|
域名解析 网络协议 安全
|
2月前
|
运维 监控 网络协议
运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面
本文介绍了运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面,旨在帮助读者提高工作效率。从基本的文件查看与编辑,到高级的网络配置与安全管理,这些命令是运维工作中的必备工具。
161 3
|
3月前
|
运维 监控 网络协议
|
3月前
|
存储 安全 Unix
2024 年 16 个适用于 Linux 的开源云存储软件 (下)
2024 年 16 个适用于 Linux 的开源云存储软件 (下)
67 0
2024 年 16 个适用于 Linux 的开源云存储软件 (下)
|
2月前
|
Linux
Linux - 如何编译源码安装软件
源码编译安装通常包括三个步骤:1) `./configure` 检测平台特征和依赖项,生成 Makefile;2) `make` 编译源码,生成可执行文件;3) `make install` 将可执行文件安装到指定目录并配置环境变量。
64 0