C++的一些细节
1. 类的初始化
几种不同初始化的区别:
A a;和A *a=new A()的区别,示例程序如下:
#include <stdio.h>
#include <string>
#include <string.h>
#include <iostream>
using namespace std;
class A{
public:
A(string name){
cout<<"a instance of A : "<<name<<endl;
this->name=name;
}
~A(){
cout<<"destory instance of A : "<<this->name <<endl;
}
void print(){
cout<<this->name<<" is printing..."<<endl;
}
private:
string name;
};
int main(){
A a1("gqx");
A* a2 = new A("test");
a1.print();
a2->print();
return 0;
}
执行结果如下,通过A a1("gqx")形式初始化的对象,在程序结束的时候会自动回收内存,调用析构函数,而通过new方式初始化的对象需要手动释放内存。
a instance of A : gqx a instance of A : test gqx is printing... test is printing... destory instance of A : gqx 按 <RETURN> 来关闭窗口...
修改主程序如下后:
int main(){
A a1("gqx");
A* a2 = new A("test");
a1.print();
a2->print();
delete a2;
return 0;
}
执行结果如下:
a instance of A : gqx a instance of A : test gqx is printing... test is printing... destory instance of A : test destory instance of A : gqx 按 <RETURN> 来关闭窗口...
由此可知:
A a1("gqx");实例化对象在栈内存,出了作用域自动释放空间。(当然如果是全局变量,存在全局区,程序结束后由系统自动释放。)
A* a2 = new A("test");实例化对象在堆内存,需要手动释放内存空间
而A* a=new A和A* a=new A()的区别则是:
A* a=new A分配一块内存空间,并将其地址传给指针a
A* a=new A()分配一块内存空间,并将其地址传给指针a。且后面的()表示的意思是用()中的内容对指针*a中的内容进行初始化,此处默认是空字符串。
2. memset函数
memset函数是内存赋值函数,用来给某一块内存空间进行赋值的,将每个字节初始化为同一值,即达到对对象的初始化,一般是对结构体的初始化,比如:
struct sockaddr_in addr; memset(&addr, 0, sizeof(addr));
其原型是:
void* memset(void *_Dst, int _Val, size_t _Size) //_Dst是目标起始地址,_Val是要赋的值,_Size是要赋值的字节数。
注意:memset()是逐字节拷贝的。例如int num[8];。我们用memset给num初始化为{1,1,1,1,1,1,1,1},一个int是4个字节的,8个int是32个字节,所以首先要赋值的长度就不应该为8而是32。 因为memset是逐字节拷贝,以num为首地址的8字节空间都被赋值为1, 即一个int变为0X00000001 00000001 00000001 00000001,显然,把这个数化为十进制不会等于1的。
所以,在memset使用时要千万小心,在给char以外的数组赋值时,只能初始化为0或者-1。-1和0的补码各个位都是1和0。
3.字符串拷贝函数
strcpy的函数原型:
char *strcpy(char* dest, const char *src);
作用:把从src地址开始且含有'\0'结束符的字符串复制到以dest开始的地址空间。
这里需要注意的是:dest应该是一片内存的首地址,不能是没有分配内存的指针,且该对象有足够的存储空间用于存储源字符串。
strncpy相对于strcpy函数多了一个表示可拷贝的最大字符数的参数。这样在很大程度上就能避免strcpy函数的不足(程序员在用strncpy时,需要填写拷贝个数,这时程序员就很肯能会检查一下dest是否有足够的内存)。
char *strncpy(char *dest, const char *src, int n);
要注意:n表示可拷贝的最大字符数,如果提前遇到'\0',拷贝的长度会小于n。另外,如果仅拷贝源字符串的一部分,可能会造成目标串没有字符串结束标志'\0';
将string类型转化为char*类型 ——C++中,c_str()的用法,就是把 string 转成 char*
string str="yoooo"; char *a=str.c_str();
4.
举例如下,为一个结构体数组申请内存空间的时候,常常会使用如下语句:
ep_events = (struct epoll_event*)malloc(sizeof(struct epoll_event)*EPOLL_SIZE);
ep_events = malloc(sizeof(struct epoll_event)*EPOLL_SIZE);
编译器会报错:
epolltest.cpp:46:23: error: invalid conversion from ‘void*’ to ‘epoll_event*’ [-fpermissive] ep_events = malloc(sizeof(struct epoll_event)*EPOLL_SIZE);
大概的意思是,malloc返回的是一个void*类型指针,编译器在编译的时候无法将其自动转换成struct epoll_event*类型的指针,需要我们手动强制转换。
5. printf
printf是一个行缓冲函数,先写到缓冲区,满足条件后,才将缓冲区刷到对应文件中,刷缓冲区的条件如下:
-
缓冲区填满
-
写入的字符中有‘\n’ '\r'
-
调用fflush手动刷新缓冲区
-
调用scanf要从缓冲区中读取数据时,也会将缓冲区内的数据刷新。
做如下实验:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include<unistd.h>
using namespace std;
int main(int argc, char *argv[])
{
printf("test");
sleep(10);
return 0;
}
10秒之后才打印test,中途Ctrl+C则不会打印test。如果打印语句换成以下形式:
printf("test\n");
程序运行时刻打印test。

浙公网安备 33010602011771号