C++new的三種用法
1. new() 分配這種型別的一個大小的記憶體空間,並以括號中的值來初始化這個變數;
2. new[] 分配這種型別的n個大小的記憶體空間,並用預設建構函式來初始化這些變數;
#include<iostream>
#include<cstring>
using namespace std;
int main(){
//char* p=new char("Hello");
//error分配一個char(1位元組)的空間,
//用"Hello"來初始化,這明顯不對
char* p=new char[6];
//p="Hello";
//不能將字串直接賦值給該字元指標p,原因是:
//指標p指向的是字串的第一個字元,只能用下面的
//strcpy
strcpy(p,"Hello");
cout<<*p<<endl; //只是輸出p指向的字串的第一個字元!
cout<<p<<endl; //輸出p指向的字串!
delete[] p;
return 0;}
輸出結果:
H
Hello
3. 當使用new運算子定義一個多維陣列變數或陣列物件時,它產生一個指向陣列第一個元素的指標,返回的型別保持了除最左邊維數外的所有維數。例如:
int *p1 = new int[10];
返回的是一個指向int的指標int*
int (*p2)[10] = new int[2][10];
new了一個二維陣列, 去掉最左邊那一維[2], 剩下int[10], 所以返回的是一個指向int[10]這種一維陣列的指標int (*)[10].
int (*p3)[2][10] = new int[5][2][10]; new了一個三維陣列, 去掉最左邊那一維[5], 還有int[2][10], 所以返回的是一個指向二維陣列int[2][10]這種型別的指標int (*)[2][10].
#include<iostream>
#include <typeinfo>
using namespace std;
int main() {
int *a = new int[34];
int *b = new int[];
int (*c)[2] = new
int[34][2];
int (*d)[2] = new int[][2];
int (*e)[2][3] = new int[34][2][3];
int (*f)[2][3] = new int[][2][3];
a[0] = 1;
b[0] = 1; //執行時錯誤,無分配的記憶體,b只起指標的作用,用來指向相應的資料
c[0][0] = 1;
d[0][0] = 1;//執行時錯誤,無分配的記憶體,d只起指標的作用,用來指向相應的資料
e[0][0][0] = 1;
f[0][0][0] = 1;//執行時錯誤,無分配的記憶體,f只起指標的作用,用來指向相應的資料
cout<<typeid(a).name()<<endl;
cout<<typeid(b).name()<<endl;
cout<<typeid(c).name()<<endl;
cout<<typeid(d).name()<<endl;
cout<<typeid(e).name()<<endl;
cout<<typeid(f).name()<<endl;
delete[] a; delete[] b; delete[] c;
delete[] d; delete[] e; delete[] f;
}
輸出結果:
int *
int *
int (*)[2]
int (*)[2]
int (*)[2][3]
int (*)[2][3]
雖然有三種new的用法,但是分為兩大類也未嘗不可,那麼是哪兩類呢?其一是new operator,也叫new表示式;其二是operator new,也叫new操作符。這兩個英文名稱起的也太絕了,很容易搞混,那就記中文名稱吧。new表示式比較常見,也最常用,例如:
string* ps = new string("abc");
上面這個new表示式完成了兩件事情:申請記憶體和初始化物件。
new操作符類似於C語言中的malloc,只是負責申請記憶體,例如:
void* buffer = operator new(sizeof(string));
注意這裡多了一個operator。這是new的第二個用法,也算比較常見吧。
那麼第三個用法就不很常見了,官方的說法是placement new,它用於在給定的記憶體中初始化物件,也就是說你手中已有一塊閒置的記憶體,例如:
void* buffer = operator new(sizeof(string));
//那麼現在buffer是你所擁有閒置記憶體的指標
buffer = new(buffer) string("abc"); //呼叫了placement new,在buffer所指向的記憶體中初始化string型別的物件,初始值是"abc"
事實上,placement new也是new表示式的一種,但是比普通的new表示式多了一個引數,當然完成的操作和返回值也不同。
因此上面new的第一種用法可以分解兩個動作,分別為後面的兩種用法。