众所周知程序当前所在目录在我们的编程中有着重要的作用,所以如何获取该变量显得很重要!

下面我说说我的思路,并保持更新!以便使用更优质的办法!

一、实现思路

1.1 相关函数介绍

假设我的 Demo.exe 现在的路径信息是: C:\Test\Room\Demo.exe

  1. 首先我们可以使用 GetModuleFileName 函数获取程序的绝对路径!也就是: C:\Test\Room\Demo.exe
  2. 然后使用 _splitpath 函数分割该路径的个个部分,得到该路径的 “驱动器号(C:)”、“目录路径(\Test\Room\)”、“基文件名(Demo)”和“文件扩展名(.exe)”。
  3. 最后用 sprintf 函数组装出我们需要的信息。

下面讲解一下用到的这三个函数,方便大家在今后的编程中灵活运用它们!

二、函数讲解

详情请参考 Microsoft 的 MSDN 文档

2.1 GetModuleFileName

2.1.1 函数声明

1
2
3
4
5
DWORD GetModuleFileNameA(
[in, optional] HMODULE hModule,
[out] LPSTR lpFilename,
[in] DWORD nSize
);
  1. 参数 hModule

正在请求其路径程序的句柄。

如果此参数为 NULL,则 GetModuleFileName 将检索当前进程的可执行文件的路径。

因为我们需要获取的是当前程序的路径,所以我们需要检索路径的程序就是当前进程对应的可执行文件。所以这一项我们可以填NULL

  1. 参数 lpFilename

指向接收模块的完全限定路径的缓冲区的指针。如果路径的长度小于 nSize 参数指定的大小,则函数将成功,并且路径将作为 NULL 终止的字符串返回。

如果路径的长度超过 nSize 参数指定的大小,则函数将成功,并且字符串将被截断为 nSize 字符,包括终止空字符。

  1. 参数 nSize

lpFilename 缓冲区的大小。

2.1.2 返回值

如果函数成功,则返回值是复制到缓冲区的字符串的长度(以字符为单位),不包括终止空字符。如果缓冲区太小而无法容纳模块名称,则字符串将被截断为 nSize 字符(包括终止空字符),函数返回 nSize,并且函数将最后一个错误设置为ERROR_INSUFFICIENT_BUFFER

Windows XP: 如果缓冲区太小而无法保存模块名称,则该函数将返回 nSize。最后一个错误代码将保留ERROR_SUCCESS。如果 nSize 为零,则返回值为零,最后一个错误代码ERROR_SUCCESS

如果函数失败,则返回值为 0(零)。

2.1.3 要求

记得使用该函数需要头文件 <Windows.h>

详情请参考 Microsoft 的 MSDN 文档: GetModuleFileName

2.2 _splitpath

2.2.1 函数声明

1
2
3
4
5
6
7
void _splitpath(
const char *path,
char *drive,
char *dir,
char *fname,
char *ext
);
  1. 参数 path

此参数用于向函数传递需要分解的原路径。

  1. 参数 drive

接收分割得到的“驱动器号”。

  1. 参数 dir

接收分割得到的“目录路径”。

  1. 参数 fname

接收分割得到的“基文件名”。

  1. 参数 ext

接收分割得到的“文件扩展名”(带符号 ‘.’)。

2.2.2 返回值

该函数没有返回值,通过参数 drivedirnameext 返回分解之后的各个部分。

2.2.3 要求

记得使用该函数需要头文件 <stdlib.h>

详情请参考 Microsoft 的 MSDN 文档: _splitpath

2.3 sprintf

2.3.1 函数声明

1
int sprintf(char *str, const char *format, ...);
  1. 参数 str

这是指向一个字符数组的指针,该数组存储了 C 字符串。

  1. 参数 format

这是字符串,包含了要被写入到字符串 str 的文本。它可以包含嵌入的 format 标签,format 标签可被随后的附加参数中指定的值替换,并按需求进行格式化。format 标签属性是 %[flags][width][.precision][length]specifier,具体讲解如下:

specifier(说明符)输出
c字符
d 或 i有符号十进制整数
e使用 e 字符的科学科学记数法(尾数和指数)
E使用 E 字符的科学科学记数法(尾数和指数)
f十进制浮点数
g自动选择 %e 或 %f 中合适的表示法
G自动选择 %E 或 %f 中合适的表示法
o有符号八进制
s字符的字符串
u无符号十进制整数
x无符号十六进制整数
X无符号十六进制整数(大写字母)
p指针地址
n无输出
%字符
flags(标识)描述
-在给定的字段宽度内左对齐,默认是右对齐(参见 width 子说明符)。
+强制在结果之前显示加号或减号(+ 或 -),即正数前面会显示 + 号。默认情况下,只有负数前面会显示一个 - 号。
(space)如果没有写入任何符号,则在该值前面插入一个空格。
#与 o、x 或 X 说明符一起使用时,非零值前面会分别显示 0、0x 或 0X。
与 e、E 和 f 一起使用时,会强制输出包含一个小数点,即使后边没有数字时也会显示小数点。默认情况下,如果后边没有数字时候,不会显示显示小数点。
与 g 或 G 一起使用时,结果与使用 e 或 E 时相同,但是尾部的零不会被移除。
0在指定填充 padding 的数字左边放置零(0),而不是空格(参见 width 子说明符)。
width(宽度)描述
(number)要输出的字符的最小数目。如果输出的值短于该数,结果会用空格填充。如果输出的值长于该数,结果不会被截断。
*宽度在 format 字符串中未指定,但是会作为附加整数值参数放置于要被格式化的参数之前。
.precision(精度)描述
.number对于整数说明符(d、i、o、u、x、X):precision 指定了要写入的数字的最小位数。如果写入的值短于该数,结果会用前导零来填充。如果写入的值长于该数,结果不会被截断。精度为 0 意味着不写入任何字符。
对于 e、E 和 f 说明符:要在小数点后输出的小数位数。
对于 g 和 G 说明符:要输出的最大有效位数。
对于 s: 要输出的最大字符数。默认情况下,所有字符都会被输出,直到遇到末尾的空字符。
对于 c 类型:没有任何影响。
当未指定任何精度时,默认为 1。如果指定时不带有一个显式值,则假定为 0。
.*精度在 format 字符串中未指定,但是会作为附加整数值参数放置于要被格式化的参数之前。
length(长度)描述
h参数被解释为短整型或无符号短整型(仅适用于整数说明符:i、d、o、u、x 和 X)。
l参数被解释为长整型或无符号长整型,适用于整数说明符(i、d、o、u、x 和 X)及说明符 c(表示一个宽字符)和 s(表示宽字符字符串)。
L参数被解释为长双精度型(仅适用于浮点数说明符:e、E、f、g 和 G)。
  1. 附加参数

根据不同的 format 字符串,函数可能需要一系列的附加参数,每个参数包含了一个要被插入的值,替换了 format 参数中指定的每个 % 标签。参数的个数应与 % 标签的个数相同。

2.3.2 返回值

如果成功,则返回写入的字符总数,不包括字符串追加在字符串末尾的空字符。如果失败,则返回一个负数。

2.3.3 要求

记得使用该函数需要头文件 <stdio.h>

三、具体实例

1
2
3
4
5
6
7
8
9
TCHAR proPath_m[_MAX_PATH], 
Drive_m[_MAX_DRIVE],
Dir_m[_MAX_DIR],
FName_m[_MAX_FNAME],
Ext_m[_MAX_EXT];

GetModuleFileName(NULL, (LPSTR)proPath_m, MAX_PATH);
_splitpath(proPath_m, Drive_m, Dir_m, FName_m, Ext_m);
sprintf(workDir, __TEXT("%s%s"), Drive_m, Dir_m);// 得到 C:\Test\Room\

补充

属性
_MAX_DRIVE3
_MAX_DIR256
_MAX_FNAME256
_MAX_EXT256