众所周知程序当前所在目录在我们的编程中有着重要的作用,所以如何获取该变量显得很重要!
下面我说说我的思路,并保持更新!以便使用更优质的办法!
一、实现思路
1.1 相关函数介绍
假设我的 Demo.exe 现在的路径信息是: C:\Test\Room\Demo.exe
- 首先我们可以使用
GetModuleFileName
函数获取程序的绝对路径!也就是:C:\Test\Room\Demo.exe
! - 然后使用
_splitpath
函数分割该路径的个个部分,得到该路径的 “驱动器号(C:
)”、“目录路径(\Test\Room\
)”、“基文件名(Demo
)”和“文件扩展名(.exe
)”。 - 最后用
sprintf
函数组装出我们需要的信息。
下面讲解一下用到的这三个函数,方便大家在今后的编程中灵活运用它们!
二、函数讲解
详情请参考 Microsoft 的 MSDN 文档
2.1 GetModuleFileName
2.1.1 函数声明
DWORD GetModuleFileNameA(
[in, optional] HMODULE hModule,
[out] LPSTR lpFilename,
[in] DWORD nSize
);
- 参数
hModule
正在请求其路径程序的句柄。
如果此参数为 NULL
,则 GetModuleFileName
将检索当前进程的可执行文件的路径。
因为我们需要获取的是当前程序的路径,所以我们需要检索路径的程序就是当前进程对应的可执行文件。所以这一项我们可以填NULL
。
- 参数
lpFilename
指向接收模块的完全限定路径的缓冲区的指针。如果路径的长度小于 nSize
参数指定的大小,则函数将成功,并且路径将作为 NULL
终止的字符串返回。
如果路径的长度超过 nSize
参数指定的大小,则函数将成功,并且字符串将被截断为 nSize
字符,包括终止空字符。
- 参数
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 函数声明
void _splitpath(
const char *path,
char *drive,
char *dir,
char *fname,
char *ext
);
- 参数
path
此参数用于向函数传递需要分解的原路径。
- 参数
drive
接收分割得到的“驱动器号”。
- 参数
dir
接收分割得到的“目录路径”。
- 参数
fname
接收分割得到的“基文件名”。
- 参数
ext
接收分割得到的“文件扩展名”(带符号 ‘.’)。
2.2.2 返回值
该函数没有返回值,通过参数 drive
、dir
、name
和 ext
返回分解之后的各个部分。
2.2.3 要求
记得使用该函数需要头文件 <stdlib.h>
详情请参考 Microsoft 的 MSDN 文档: _splitpath
2.3 sprintf
2.3.1 函数声明
int sprintf(char *str, const char *format, ...);
- 参数
str
这是指向一个字符数组的指针,该数组存储了 C 字符串。
- 参数
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)。 |
- 附加参数
根据不同的 format
字符串,函数可能需要一系列的附加参数,每个参数包含了一个要被插入的值,替换了 format
参数中指定的每个 % 标签。参数的个数应与 % 标签的个数相同。
2.3.2 返回值
如果成功,则返回写入的字符总数,不包括字符串追加在字符串末尾的空字符。如果失败,则返回一个负数。
2.3.3 要求
记得使用该函数需要头文件 <stdio.h>
三、具体实例
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_DRIVE | 3 |
_MAX_DIR | 256 |
_MAX_FNAME | 256 |
_MAX_EXT | 256 |