字符串的连接的另一种常见的情况是字符串和数字的组合。有时数字需要被转换为字符串,有时需要把若干数字和字符串混合组合起来。这往往用于打印日志的时候,日志中可能含有文件名,时间和行号,以及其他的信息。
熟悉C远的读者会使用sprintf,这个函数的宽字符版本为swprintf。该函数在驱动开发中依然可以使用,但是不安全,微软建议使用RtlStringCbPrintfW 来带替它。RtlStringCbPrintfW需要包含头文件,如下:
#include <ntstrsafe.h>在链接的时候,还需要连接库 : ntsafestr.lib 。
下面的代码生成一个字符串,字符串中包含文件的路径和这个文件的大小。
#include <ntstrsafe.h>RtlStringCbPrintfW在目标缓冲区内存不足的时候依然可以打印,但是多余的部分被截去了,返回 status值为 STATUS_BUFFER_OVERFLOW。调用这个函数之前很难知道究竟需要多长的缓冲区,一般都采取倍增尝试。每次都传入一个前次尝试长度为2倍的新缓冲区,直到这个函数返回 STATUS_SUCCESS为止。
//--任何时候,假设文件路径的长度为有限的都是不对的。应该动态的分配内存。
//--但是动态分配内存还没有讲述,所以,这里再次把内存空间定义在局部变量中,也就是所谓的“在栈中”
WCHAR buf[512] = {0};
UNICODE_STRING dst;
NTSTATUS status;
........
//---字符串初始化为空串,缓冲区长度为 512 * sizeof(WCHAR)
RtlInitEmptyString(dst, dst_buf, 512 * sizeof(WCHAR));
//---调用RtlStringCbPrintfW 函数进行打印
status = RtlStringCbPrintfW(
dst->Buffer, 512 * sizeof(WCHAR),
L"filepath = %wz file size = %d \r\n",
&file_path,
file_szie
);
//--这里调用wcslen没问题,这是因为RtlStringCbInitPrintfW打印的
//--字符串是空结束的
dst->Length = wcslen(dst->Buffer) * sizeof(WCHAR);
值得注意的是,UNICODE_STRING 类型的指针,用%wz打印可以打印出字符串。在不能保证字符串为空结束的时候,必须避免使用%ws或者 %s。其他的打印格式字符串与传统C语言中的printf函数完全相同,可以尽情使用。
另外就是常见的输出打印。printf函数只有在控制台输出的情况下才有意义,在驱动中没有控制台,但是Windows内核中拥有调试信息输出机制,可以使用WinDbg查看打印的调试信息。
驱动中可以调用 DbgPrint()函数来打印调试信息。这个函数的使用和printf基本相同,但是格式字符串要使用宽字符。DbgPrint()的一个缺点在于,发行的驱动程序往往不希望附带任何输出信息,只有调试版本才需要调试信息。但是,DbgPrint()无论是发行版本还是调试版本编译都会无效,为此可以自己定义一个宏:
#if DBG不过这样做的后果是,由于 KdPrint(a)只支持1个参数,因此必须把 DbgPrint()所在参数都括起来当作一个参数传入。导致 KdPrint()看起来很奇特的是用了双重括弧:
KdPrint(a) DbgPrint##a
#else
KdPrint(a)
#endif
//--调用KdPrint((这个宏没有必要自己定义,WDK包中已有,所以可以直接用 KdPrint来代替 DbgPrint取得更方便的效果。
L"file path = %wz file size = %d \r\n",
&file_path,
file_size));
------------------摘自<[天书夜读-从汇编到Windows内核编程]>
----------------------------------------- 这个字儿写的漂亮吧------------------------------------
我模仿了好久呢,都还不如这个的百分之40。。 哈哈,,承认,我是个菜鸟。