编译时打印宏变量内容

1
2
3
4
5
6
7
8
9
#define PRINT_MACRO_HELPER(_x) #_x
#define PRINT_MACRO(_x) #_x " = " PRINT_MACRO_HELPER(_x)
 
#define DEFINED_PI 3.14
#define DEFINED_NULL
#pragma message(PRINT_MACRO(DEFINED_PI))
#pragma message(PRINT_MACRO(DEFINED_NULL))
#pragma message(PRINT_MACRO(UNDEFINED))
//#error print stop here.

refer to: https://blog.csdn.net/xshbx/article/details/7981564

晋朝帝王年表

西晋

庙号 帝号 姓名 在位时期 上位 在位 年号
高祖 宣帝 司马懿 249-251 70岁 3年
世宗 景帝 司马师 252-254 44岁 3年
太祖 文帝 司马昭 255-265 44岁 11年
世祖 武帝 司马炎 266-290 30岁 25年 泰始/咸宁/太康/太熙
孝惠帝 司马衷 291-306 35岁 16年 永平/元康/永康/永宁/太安/永安/建武/永兴/光熙
孝怀帝 司马炽 307-313 23岁 7年 永嘉
孝愍帝 司马邺 313-316 13岁 4年 建兴

东晋

庙号 帝号 姓名 在位时期 上位 在位 年号
中宗 元帝 司马睿 317-323 41岁 7年 建武/太兴/永昌/太宁
肃宗 明帝 司马绍 323-326 24岁 4年 太宁
显宗 成帝 司马衍 326-342 5岁 15年 咸和/咸康
康帝 司马岳 343-344 21岁 2年 建元
孝宗 穆帝 司马聃 345-361 2岁 17年 永和/升平
哀帝 司马丕 362-365 21岁 4年 隆和/兴宁
司马奕 366-371 24岁 6年 太和
太宗 简文帝 司马昱 371-372 51岁 2年 咸安
烈宗 孝武帝 司马曜 373-396 11岁 28年 宁康/太元
安帝 司马德宗 397-402 15岁 6年 隆安/元兴/大亨
恒楚 桓玄 403-404 34岁 2年 永始
恒楚 恒谦 404-410 7年 天康
安帝 司马德宗 405-418 23岁 14年 义熙
恭帝 司马德文 419-420 33岁 2年 元熙

refer to: https://baike.baidu.com/item/%E6%99%8B%E6%9C%9D/195770?fr=aladdin

从cmdline解析出参数token

将命令行字符串解析成argv中的一个个token,上源码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#include <string>
#include <vector>
#include <sstream>
 
using namespace std;
 
wstring copySpanStr(const wchar_t* start, const wchar_t* end)
{
	wchar_t* buf = (wchar_t*)malloc((end - start + 1) * sizeof(wchar_t));
	wchar_t* pTo = buf;
	for (const wchar_t* p = start; p != end; p++)
	{
		if (*p == L'\\' && p + 1 != end)
		{
			p++;
			*pTo++ = *p;
		}
		else
			*pTo++ = *p;
	}
	*pTo = 0;
	wstring out = buf;
	free(buf);
	return out;
}
 
int getParams(const wchar_t* str, vector<wstring>& out)
{
	int isInQMark = 0;
	int isInSpace = 1;
	const wchar_t* tokenStart = str;
	for (const wchar_t* p = str; *p != 0; p++)
	{
		if (*p == L'"')
		{
			if (!isInQMark)
			{
				isInQMark = 1;
				tokenStart = p + 1;
			}
			else
			{
				isInQMark = 0;
				out.push_back(copySpanStr(tokenStart, p));
			}
		}
		else if (*p == L'\\')
		{
			if (*(p + 1) != 0)
				p++;
		}
		else if (*p == L' ')
		{
			if (!isInQMark)
			{
				if (!isInSpace)
				{
					isInSpace = 1;
					if (tokenStart != p)
					{
						out.push_back(copySpanStr(tokenStart, p));
						tokenStart = p;
					}
				}
				else
				{
 
				}
			}
		}
		else
		{
			if (!isInQMark)
			{
				if (!isInSpace)
				{
 
				}
				else
				{
					isInSpace = 0;
					tokenStart = p;
				}
			}
		}
	}
	return out.size();
}
 
wstring strReplaceCh2Str(const wchar_t* in, wchar_t fromCh, const wchar_t* toStr)
{
	const wchar_t* p = in;
	wstringstream ss;
	const wchar_t* pToStr;
	for (; *p != 0; p++)
	{
		if (*p == fromCh)
		{
			for (pToStr = toStr; *pToStr != 0; pToStr++)
			{
				ss << *pToStr;
			}
		}
		else
			ss << *p;
	}
	return ss.str();
}
 
int main()
{
	wstring path = strReplaceCh2Str(L"c:\\aa.txt", L'\\', L"\\\\");
	wstring msg = strReplaceCh2Str(L"\"优孩\"是EuhatExpert的中文名。", L'\"', L"\\\"");
 
	wstring cmd = L"EuhatExample.exe subCmd \"" + path + L"\" \"" + msg + L"\"";
 
	vector<wstring> params;
	getParams(cmd.c_str(), params);
	return 0;
}

dll中的全局类实例什么时候初始化

如果外部工程ExeB调了动态库DllA中的函数,当ExeB运行时,DllA中的全局类实例一定会初始化,但不一定非得在ExeB中调DllA中的函数后DllA中的全局类实例才会初始化,比如在DllA中代码如下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class DllA01()
{
public:
	DllA01()
	{
		MessageBoxA(NULL, "init", NULL, 0);
	}
	~DllA01()
	{
		MessageBoxA(NULL, "fini", NULL, 0);
	}
	void test(void)
	{
		MessageBoxA(NULL, "test", NULL, 0);
	}
};
__declspec(dllexport) DllA01 gA01;
extern "C" __declspec(dllexport) void dllA01Test(void)
{
	MessageBoxA(NULL, "dllA01Test", NULL, 0);
}

在ExeB中加入

1
2
3
4
#pragma comment(lib, "DllA.lib")
#pragma comment(linker, "/include:__imp__dllA01Test")
//#pragma comment(linker, "/include:__imp_?gA01@@3VDllA01@@A")
//#pragma comment(linker, "/ENTRY:foo")

休眠后启动移动硬盘不重新初始化连接

比如有这样的需求,VMWare的虚拟机放在移动硬盘中,虚拟机一直开着,有时需要离开一会,就让机器休眠了,结果回来唤醒机器后,发现由于插在USB口上的移动硬盘重新初始化了,VMWare一直连接着的文件句柄也就失效了,结果就是运行中的这个移动硬盘上的虚拟机崩了。

这是个很严重的问题,这让人无法将虚拟机放在移动硬盘中运行。

一直都在找如何让Windows唤醒后不初始化移动硬盘而应该一直保持休眠前的状态的方法,没有如果。

这几天买了一个新的移动硬盘,一直插在机器上用,因有事去办就休眠了机器,回来唤醒机器,发现并没有像以前那个移动硬盘那样新弹出移动硬盘的根目录文件夹窗口。

哦,明白了,以前的移动硬盘是机械硬盘,而新买的移动硬盘是固态的,唤醒机器时固态的可以直接进入工作状态,不需要重新初始化,也就说明VMWare虚拟机可以安全地在固态移动硬盘上运行,不怕机器休眠!

GetHDSerial编译不通过

源码参见https://www.codeleading.com/article/62852336906/
提示

1
2
3
4
error C2065: “STORAGE_PROPERTY_QUERY”: 未声明的标识符
error C2065: “StorageDeviceProperty”: 未声明的标识符
error C2065: “PropertyStandardQuery”: 未声明的标识符
error C2065: “IOCTL_STORAGE_QUERY_PROPERTY”: 未声明的标识符

按F1在msdn上查到的可能结果为

1
#include <Ntddstor.h>

但加了还是编译不通过,其实正确的是只要加入

1
#include <WinIoCtl.h>

refer to: https://stackoverflow.com/questions/4420632/i-dont-know-what-problem-it-is-error-error-c2065-storage-property-query