C++面试问题

指针

参数传递

下列程序是否正确:

1
2
3
4
5
6
7
8
9
10
11
void GetMemory(char *p)
{
p = new char[100];
}
void Test(void)
{
char *str = NULL;
GetMemory(str);
strcpy(str, "hello world");
printf(str);
}

错误,程序崩溃。因为GetMemory 并不能传递动态内存, Test 函数中的 str 一直都是 NULL。strcpy(str, "hello world");将使程序崩溃。

1
2
3
4
5
6
7
8
9
10
11
char* GetMemory(void)
{
char p[] = "hello world";
return p;
}
void Test(void)
{
char *str = NULL;
str = GetMemory();
printf(str);
}

可能返回乱码。因为GetMemory 返回的是指向“栈内存”的指针,该指针的地址不是 NULL,但其原现的内容已经被清除,新内容不可知。

1
2
3
4
5
6
7
8
9
10
11
void GetMemory2(char **p, int num)
{
*p = new char[num];
}
void Test(void)
{
char *str = NULL;
GetMemory2(&str, 100);
strcpy(str, "hello");
printf(str);
}

能正常运行,但是内存泄漏。

1
2
3
4
5
6
7
8
9
10
11
oid Test(void)
{
char *str = new char[100];
strcpy(str, "hello");
delete[ ] str;
if (str != NULL)
{
strcpy(str,"world");
printf(str);
}
}

篡改动态内存区的内容,后果难以预料,非常危险。因为 delete[ ]str;之后,str成为野指针(需要str = NULL;)if(str != NULL)语句不起作用。

编译

编译选项MT和MD的区别

  在C++(特别是Windows平台下的MSVC编译器)中,/MT 和 /MD 是两种不同的运行时库链接方式,主要区别在于如何链接C/C++标准库(如 msvcrt.lib、libcmt.lib 等)。

  • /MT:静态链接运行时库

  全称是Multi-Threaded(静态多线程库)。特点是将C/C++标准库(如libcmt.lib)静态编译到你的程序中,生成的可执行文件不依赖外部的运行时库DLL,不过因为库代码被直接嵌入导致生成的文件体积较大。适用的场景有需要独立分发的程序(需要避免系统依赖),或兼容性要求高(避免用户缺少运行时库)。

  • /MD:动态链接运行时库

  全称是Multi-Threaded Dll(动态多线程库)。特点是程序动态链接到微软的运行时库DLL(如msvcrXX.DLL),生成的文件体积较小,但运行时需要系统中有对应的DLL,通常需要程序分发对应的vcredist(微软VisualC++运行时库)。适用的场景有需要多个程序共享同一份运行库(减少磁盘和内存占用),或依赖其他动态库时,保持运行时版本库一致。

需要注意的问题:

  1. 如果项目中的库(如第三方的DLL)和主程序使用不同的选项(一个用/MT,一个用/MD),会导致链接冲突或运行时崩溃。需要保证所有模块(包括exe和dll)使用相同版本的运行时库选项。
  2. /MTd和/MDd是对应的调试版本,分别链接到静态(如libcmtd.lib)或动态(如msvcrXXd.dll)的调试运行库。
  3. 现代大多数情况下推荐使用/MD动态链接,便于更新运行时库,只有特殊需求(如嵌入式环境)才使用/MT。

编程

不用C++库函数,自行实现strcpy

1
2
3
4
5
6
7
8
char *strcpy(char *strDest, const char *strSrc);
{
assert((strDest!=NULL) && (strSrc !=NULL));
char *address = strDest;
while( (*strDest++ = * strSrc++) != ‘\0’ )
;
return address ;
}