百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

C语言fopen函数的文件相对路径到底相对哪?

liebian365 2024-11-21 17:34 24 浏览 0 评论

C语言怎么打开文件?

C语言要打开一个文件可以使用标准库的fopen函数,来看下fopen函数的声明:

FILE *fopen(const char *filename, const char *mode)

其中,filename参数的类型是字符串,用于指定要打开文件的路径,可以是绝对路径,也可以是相对路径。mode参数的类型也是字符串,用于指定打开文件的操作模式,比如"r"(只读模式)等等。

那么,这里要讨论的是filename参数使用相对路径的时候,相对位置到底是哪里呢?
1、源代码文件所在路径?
2、源代码编译后的可执行文件所在路径?
3、我们自己当前所在的路径?

你觉得是哪里?

答案是3,我们自己当前所在的路径。Show me the code,写个代码证明一下。

证明fopen函数相对路径的位置

创建个新目录CFileTest,在目录下面创建main.c源文件,hello.txt文本文件,以及build子目录(存放编译后的可执行文件)。目录结构如下:

CFileTest
  - build
  - hello.txt
  - main.c

hello.txt只是一个普通的文本文件,里面只有Hello World字符串。
main.c源代码的功能也很简单,就是使用fopen函数打开相对路径下的hello.txt文本文件,并打印出文件内的字符串:

#include <errno.h>
#include <stdio.h>
#include <string.h>

int main() {
  FILE* pFile;
  //以只读模式打开相对路径下的hello.txt文件
  pFile = fopen("hello.txt", "r");


  //如果文件打开失败,打印错误码和错误描述
  if (pFile == NULL) {
    int errorCode = errno;
    char* errorMsg = strerror(errorCode);
    printf("Open file fail, errorCode:%d, errorMsg:%s\n", errorCode, errorMsg);
    return -1;
  }

  //如果文件打开成功,打印出文件内容
  int c;
  while (1) {
    c = fgetc(pFile);
    if (feof(pFile)) {
      printf("\n");
      break;
    }
    printf("%c", c);
  }

  fclose(pFile);
  return 0;
}

1、证明相对路径不是相对于源文件

现在进入build子目录,在build子目录下编译main.c,并把可执行文件指定到build目录:

nan@HWin-Jianan:~/CFileTest$ cd build
nan@HWin-Jianan:~/CFileTest/build$ gcc ../main.c -o main
nan@HWin-Jianan:~/CFileTest/build$ ls -l
total 20
-rwxr-xr-x 1 nan nan 17016 Feb  4 00:14 main

可以看到在build子目录下已经生成main可执行文件,当前的目录结构如下:

CFileTest
  - build
    - main
  - hello.txt
  - main.c

在build目录执行下main可执行文件:

nan@HWin-Jianan:~/CFileTest/build$ ./main
Open file fail, errorCode:2, errorMsg:No such file or directory

可以看到hello.txt文件和main.c源文件同在一个目录,但结果却报错了,找不到hello.txt这个文件。说明"hello.txt"这个相对路径不是相对于源文件的路径。

2、证明相对路径不是相对于可执行文件所在路径

把hello.txt文件移动到build目录下:

nan@HWin-Jianan:~/CFileTest/build$ mv ../hello.txt ./

当前目录结构如下:

CFileTest
  - build
    - hello.txt
    - main
  - main.c

在build目录下继续执行main可执行文件:

nan@HWin-Jianan:~/CFileTest/build$ ./main
Hello World!

可以看到成功打印出了hello.txt文件内容"Hello World",这能确定相对路径是相对于可执行文件所在路径了吗?别急,别忘了我们自己当前所在路径也在build目录下,也有可能是因为相对于我们自己当前所在路径才成功找到文件的。那么,切换下我们自己当前所在的路径看看,切换到上一级CFileTest目录再执行看看:

nan@HWin-Jianan:~/CFileTest/build$ cd ..
nan@HWin-Jianan:~/CFileTest$ ./build/main
Open file fail, errorCode:2, errorMsg:No such file or directory

又找不到文件了,可以确定相对路径也不是相对于可执行文件所在的路径。

说明:可执行文件所在路径不一定就是我们自己当前所在的路径。比如上面这次实验,可执行文件在build目录下,我们自己当前所在的路径在CFileTest目录下。

3、证明相对路径相对于我们自己当前所在路径

再次把hello.txt文件移动到CFileTest目录下:

nan@HWin-Jianan:~/CFileTest$ mv ./build/hello.txt ./

移动后目录结果恢复为:

CFileTest
  - build
    - main
  - hello.txt
  - main.c

在CFileTest目录下执行可执行文件:

nan@HWin-Jianan:~/CFileTest$ ./build/main
Hello World!

成功找到hello.txt文件,说明相对路径是相对于我们自己当前所在路径的。
就这?有更直接的证据吗?那就gdb debug模式打个断点,打印下执行文件时当前的路径。
首先重新gcc编译下源文件,增加-g参数保留下调试信息:

nan@HWin-Jianan:~/CFileTest$ gcc -g main.c -o ./build/main

使用gdb调试main可执行文件,在源代码第8行"pFile = fopen("hello.txt", "r");"处打个断点,然后运行程序到这个断点:

nan@HWin-Jianan:~/CFileTest$ gdb ./build/main
GNU gdb (Ubuntu 9.2-0ubuntu1~20.04.1) 9.2
...
Reading symbols from ./build/main...
(gdb) b 8
Breakpoint 1 at 0x1235: file main.c, line 8.
(gdb) r
Starting program: /home/nan/CFileTest/build/main


Breakpoint 1, main () at main.c:8
8         pFile = fopen("hello.txt", "r");

输入pwd命令,打印可执行文件时当前路径:

(gdb) pwd
Working directory /home/nan/CFileTest.

可以看到,文件执行时的工作目录在CFileTest,与我们自己当前所在的目录一致。

总结

C语言的fopen函数,如果指定相对路径的文件需要注意一下,相对路径是相对于我们自己当前所在的路径,而不是源文件或者可执行文件的路径。

相关推荐

go语言也可以做gui,go-fltk让你做出c++级别的桌面应用

大家都知道go语言生态并没有什么好的gui开发框架,“能用”的一个手就能数的清,好用的就更是少之又少。今天为大家推荐一个go的gui库go-fltk。它是通过cgo调用了c++的fltk库,性能非常高...

旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗

这几天老毛桃整理了几个微型Linux发行版,准备分享给大家。要知道可供我们日常使用的Linux发行版有很多,但其中的一些发行版经常会被大家忽视。其实这些微型Linux发行版是一种非常强大的创新:在一台...

codeblocks和VS2019下的fltk使用中文

在fltk中用中文有点问题。英文是这样。中文就成这个样子了。我查了查资料,说用UTF-8编码就行了。edit->Fileencoding->UTF-8然后保存文件。看下下边的编码指示确...

FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库

FLTK(FastLightToolkit)是一个轻量级的跨平台GUI库,特别适用于开发需要快速、高效且简单界面的应用程序。本文将介绍Python中的FLTK库,包括其特性、应用场景以及如何通过代...

中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux

IT之家1月29日消息,去年6月份,中科院大学教授、中科院计算所研究员包云岗,发布了开源高性能RISC-V处理器核心——香山。近日,包云岗在社交平台晒出图片,香山芯片已流片,回片后...

Linux 5.13内核有望合并对苹果M1处理器支持的初步代码

预计Linux5.13将初步支持苹果SiliconM1处理器,不过完整的支持工作可能还需要几年时间才能完全完成。虽然Linux已经可以在苹果SiliconM1上运行,但这需要通过一系列的补丁才能...

Ubuntu系统下COM口测试教程(ubuntu port)

1、在待测试的板上下载minicom,下载minicom有两种方法:方法一:在Ubuntu软件中心里面搜索下载方法二:按“Ctrl+Alt+T”打开终端,打开终端后输入“sudosu”回车;在下...

湖北嵌入式软件工程师培训怎么选,让自己脱颖而出

很多年轻人毕业即失业、面试总是不如意、薪酬不满意、在家躺平。“就业难”该如何应对,参加培训是否能改变自己的职业走向,在湖北,有哪些嵌入式软件工程师培训怎么选值得推荐?粤嵌科技在嵌入式培训领域有十几年经...

新阁上位机开发---10年工程师的Modbus总结

前言我算了一下,今年是我跟Modbus相识的第10年,从最开始的简单应用到协议了解,从协议开发到协议讲解,这个陪伴了10年的协议,它一直没变,变的只是我对它的理解和认识。我一直认为Modbus协议的存...

创建你的第一个可运行的嵌入式Linux系统-5

@ZHangZMo在MicrochipBuildroot中配置QT5选择Graphic配置文件增加QT5的配置修改根文件系统支持QT5修改output/target/etc/profile配置文件...

如何在Linux下给zigbee CC2530实现上位机

0、前言网友提问如下:粉丝提问项目框架汇总下这个网友的问题,其实就是实现一个网关程序,内容分为几块:下位机,通过串口与上位机相连;下位机要能够接收上位机下发的命令,并解析这些命令;下位机能够根据这些命...

Python实现串口助手 - 03串口功能实现

 串口调试助手是最核心的当然是串口数据收发与显示的功能,pzh-py-com借助的是pySerial库实现串口收发功能,今天痞子衡为大家介绍pySerial是如何在pzh-py-com发挥功能的。一、...

为什么选择UART(串口)作为调试接口,而不是I2C、SPI等其他接口

UART(通用异步收发传输器)通常被选作调试接口有以下几个原因:简单性:协议简单:UART的协议非常简单,只需设置波特率、数据位、停止位和校验位就可以进行通信。相比之下,I2C和SPI需要处理更多的通...

同一个类,不同代码,Qt 串口类QSerialPort 与各种外设通讯处理

串口通讯在各种外设通讯中是常见接口,因为各种嵌入式CPU中串口标配,工业控制中如果不够还通过各种串口芯片进行扩展。比如spi接口的W25Q128FV.对于软件而言,因为驱动接口固定,软件也相对好写,因...

嵌入式linux为什么可以通过PC上的串口去执行命令?

1、uboot(负责初始化基本硬bai件,如串口,网卡,usb口等,然du后引导系统zhi运行)2、linux系统(真正的操作系统)3、你的应用程序(基于操作系统的软件应用)当你开发板上电时,u...

取消回复欢迎 发表评论: