1,内存管理问题
下面的内存管理代码中,错误的是:
A:
int *a=new int(12);
//.....
free(a);
B:
int *ip=static_cast<int*>(malloc(sizeof(int)));
*ip=10;
//.....
delete ip;
C:
double *a=new double[1];
//....
delete a;
D:
int *ip=new int(12);
for(int i=0;i<12;++i){
ip[i]=i;
}
delete []ip;
对于上面的代码,首先:
- new和delete搭配,malloc和free搭配。
所以,A和B都是错误的。
而对于D,需要注意的是,new int(12)和new int[12]的区别。**new int(12)**是生成了一个值为12的int变量,**new int[12]**才是生成一个大小为12的数组。所以,delete []ip是错误的,D错。
而对于C选项,乍一看上去,a是一个数组,应该用delete []a。但是:
- 对于基本类型数组而言,delete a和delete []a的效果是一样的。
- 而如果a是一个用户自定义对象的数组,则只能用delete []a。
2,template问题
模板(Template)是一种强大的C++软件复用特性,通常有两种形式:
- 函数模板
- 类模板
模板是一个类家族的抽象, 它只是对类的一个描述。编译程序并不为类模板(包括成员函数定义)创建程序代码,但是通过对类模板的实例化,可以生成一个具体的类以及该具体类的对象。
- 函数模板的实例化是由编译程序在处理函数调用时自动完成的。
- 类模板的实例化必须由程序员在程序中显式地指定才行。
另外,
- 函数模板针对参数类型不同的函数,同时还可以将函数返回值类型作为模板参数。
- 类模板针对数据成员和成员函数类型不同的类,同时还针对继承的基类类型作为模板参数。
3,重载问题
重载的概念是:
- 方法名称相同,参数个数、次序、类型不同
因此重载对返回值没有要求,可以相同,也可以不同
但是如果参数的个数、类型、次序都相同,方法名也相同,仅返回值不同,则无法构成重载。
因此,重载的基本条件是:
- 参数的类型不同
- 参数的顺序不同
- 参数的个数不同
但是:
- 函数的返回值类型不同,不能作为基本条件
4,转义字符
对于问题:
下面程序的输出结果为:
char s[]="\\123456\123456\t";
printf("%d\n",strlen(s));
对于上面的问题,
如果不是转移字符:
char s[] = "//123456/123456/t";
则,这样strlen(s)输出17。
而对于原题:
char s[] = "\\123456\123456\t";
这就涉及到转义字符了。
在ascii码中:
- ‘\ddd’ 表示八进制
- ’\xdd’ 表示十六进制,(d是一位数字占位符,x是十六进制标志)
在题目中,因为‘\’后边没有‘x’,所以解释为三位八进制’\123’,其十进制是83,在ascii中是‘S’。
另外:
- \\ 表示字符 \
- \t表示制表符
因此,这道题的结果为12。
5,C语言变量声明内存分配
一个由C/C++编译的程序占用的内存分为以下几个部分
-
栈区(stack)— 程序运行时由编译器自动分配,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈。程序结束时由编译器自动释放。
-
堆区(heap) — 在内存开辟另一块存储区域。一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表。
-
全局区(静态区)(static)—编译器编译时即分配内存。全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。 - 程序结束后由系统释放
-
文字常量区 —常量字符串就是放在这里的。 程序结束后由系统释放
-
程序代码区—存放函数体的二进制代码。
程序举例:
//main.cpp
int a = 0; //全局初始化区
char *p1; //全局未初始化区
main()
{
int b; //栈
char s[] = "abc"; //栈
char *p2; //栈
char *p3 = "123456"; //"123456\0"在常量区,p3在栈上
static int c = 0l //全局(静态)初始化区
p1 = (char*)malloc(10);
p2 = (char*)malloc(20);
//上面分配得来的10和20字节的局域在堆区
strcpy(p1, "123456"); //123456\0放在常量区,编译器可能会将它与p3所指向的"123456"优化成一个地方。
}
有价值的博客:
C语言变量声明内存分配
6,递归调用
- 直接递归调用: 在函数a(或过程)中直接引用(调用)函数a本身。
- 间接递归调用: 在函数a(或过程)中调用另外一个函数b,而该函数b又引用(调用)了函数a。
7,友元函数的重载运算
C++规定:
- =
- [ ]
- ( )
- ->
上述四个运算符只能被重载为类的非静态成员函数,其他的则可以被友元函数重载。
原因在于其他的运算符重载函数都会根据参数类型或者数目进行精确匹配,而上述四个运算符不具有这种检查的功能,使用友元函数就会出错。
8,字符数组初始化的方法
字符数组初始化有两种方法:
- 逐个字符赋值
char b[10]={'H','E','L','L','O','!','\0'};
- 用字符常量对整个数组赋值
char b[10]="HELLO!";
而对于写法:
char b[10];b="HELLO!";
这样写是错误的,因为没有将字符串常量直接赋值给char 数组变量的方式,此时b是b[10]的首地址。
字符串拷贝函数:
- 格式:
char *strcpy (char *s1, const char *s2);
- 功能:将s2所指的字符串拷贝到s1所指的字符串中。需要说明的是:
(1)参数s1和s2都是指向字符串的指针,s1可以是字符数组名或者字符指针,但是不能使字符型常量,s2可以是字符串常量、字符数组或者字符指针。
(2)将s2所指的字符串拷贝到s1所指的字符串中,使用赋值语句s1 = s2是错误的。
(3)要保证s1的长度足够大,一般能过容纳下s2所指的字符串。
因此,对于写法:
char b[10];strcpy(b,"HELLO!");
这样写是正确的。
9,自加运算的特点
对于题目:
int i=5,k;
k=(++i)+(++i)+(i++);
printf("%d,%d",k,i);
输出应该是多少?
这道题需要注意的是,k并不是三个++i或者i++的结果的和,而是三个i的和。
同时,括号优先级最高,在经过前面两个++i的操作之后,i的值为7,而因为i++是当前语句执行完成之后才会执行的。
因此,最后的输出为:
21 8
10,字符串指针数组
对于问题:
已知char *a[]={ "fortran", " basic", "pascal", "java", "c++" };
,则cout<<a[3];
的显示结果应该是什么?
对于这道题,我们可以看出的是,a是一个数组,数组里面的内容是char类型的指针, 指针指向字符串。
因此,输出为java