C++是基于C語言擴展發(fā)展而來的面向?qū)ο蟮某绦蛟O(shè)計語言,本文將主要討論C++語言基于C語言擴展的方面。
創(chuàng)新互聯(lián)建站專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于網(wǎng)站建設(shè)、網(wǎng)站制作、垣曲網(wǎng)絡(luò)推廣、重慶小程序開發(fā)、垣曲網(wǎng)絡(luò)營銷、垣曲企業(yè)策劃、垣曲品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運營等,從售前售中售后,我們都將竭誠為您服務(wù),您的肯定,是我們大的嘉獎;創(chuàng)新互聯(lián)建站為所有大學生創(chuàng)業(yè)者提供垣曲建站搭建服務(wù),24小時服務(wù)熱線:18980820575,官方網(wǎng)址:www.cdcxhl.comC語言中變量的定義必須在作用域開始的位置進行定義。
#include
int main(int argc, char *argv[])
{
int i;//定義變量
int j;
//使用變量
for(i = 0; i < 10; i++)
{
for(j = 0; j < 10; j++)
{
}
}
//error: 'for' loop initial declarations are only allowed in C99 mode
for(int k = 0; k < 10; k++)
{
}
return 0;
}
C++更強調(diào)語言的實用性,C++中所有的變量可以在需要使用時再定義。
#include
using namespace std;
int main(int argc, char *argv[])
{
int a = 7;
//使用時定義變量
for(int i = 0; i < 10; i++)
{
for(int j = 0; j < 10; j++)
{
}
}
return 0;
}
在C語言中:
const int a = 100;
int *p = &a;
在C++語言中:
const int a = 100;//必須在定義的時候初始化
const int *p = &a;//類型必須嚴格匹配
在C++語言中不能隱式轉(zhuǎn)換數(shù)據(jù)類型。error: invalid conversion from 'const int*' to 'int*' [-fpermissive]
C語言中,沒有定義bool類型,表示真用非0,假用0。
C++語言中,定義了自己的bool類型,真為true,假為false,是基本類型。sizeof(bool) = 1
C++編譯器會將非0值轉(zhuǎn)換為true,0轉(zhuǎn)換為false
#include
using namespace std;
int main(int argc, char *argv[])
{
bool a = 0;
printf("a = %d\n", a);//0
bool b = -1;
printf("b = %d\n", b);//1
b = b + 1;//1 + 1 = 2 => 1
printf("b = %d\n", b);//1
printf("sizeof(bool) = %d\n", sizeof(bool));//1
return 0;
}
C語言中枚舉本質(zhì)就是整型,枚舉變量可以用任意整型賦值。而c++中枚舉變量,只能用被枚舉出來的元素初始化。
在C語言中,枚舉的使用
#include
enum weekday
{
monday,tuesday,wednesday,thursday,friday,saturday,sunday
};
int main(int argc, char **argv)
{
enum weekday day = monday;
enum weekday a = sunday;
enum weekday b = 100;
//weekday c = sunday;錯誤用法,需要使用enum聲明
printf("%d %d %d\n", day, a, b);
return 0;
}
在C++語言中枚舉的使用
enum weekday
{
monday,tuesday,wednesday,thursday,friday,saturday,sunday
};
int main()
{
weekday day = sunday;
enum weekday a = monday;
cout<
C語言中表達式不可以做左值,C++中某些表達式可以做左值。(a = b) = 6;
register關(guān)鍵字請求編譯器將局部變量存儲到寄存器中。
C語言中,無法獲取register關(guān)鍵字修飾的局部變量的地址。
C++語言中,C++依然支持register關(guān)鍵字,但C++編譯器對register關(guān)鍵字進行了優(yōu)化,C++語言中可以獲取register變量的地址。對于早期的C++編譯器,C++編譯器發(fā)現(xiàn)程序中需要獲取register關(guān)鍵字修飾的變量地址時,register關(guān)鍵字對變量的聲明失效;對于現(xiàn)代C++編譯器,register關(guān)鍵字的存在只是為了兼容C語言,register關(guān)鍵字本身已經(jīng)無任何意義。
C語言中可以重復(fù)定義多個重名的全局變量,同名的全局變量被鏈接到全局數(shù)據(jù)區(qū)的一個地址空間。
#include
//定義重名的全局變量合法
int a;
int a;
int main(int argc, char *argv[])
{
return 0;
}
C++語言不允許定義多個同名的全局變量。
#include
using namespace std;
//定義重名的全局變量是非法的
int a;
int a;
//error: redefinition of 'int a'
int main(int argc, char *argv[])
{
return 0;
}
C語言中,struct定義了一種變量的集合,struct定義的標識符不是一種新類型。C語言中的struct內(nèi)部不可以定義函數(shù)。
#include
struct tag_student
{
const char* name;
int age;
};
typedef struct tag_student Student;
int main(int argc, char *argv[])
{
//合法定義
Student s;
s.name = "lee";
s.age = 30;
//非法定義
tag_student ts;//error: unknown type name 'tag_student'
//合法定義
struct tag_student sts;
sts.name = "bauer";
sts.age = 23;
return 0;
}
C語言中只有使用typedef關(guān)鍵字重命名struct后才可以使用Student定義變量。
C++語言中struct用于定義一種全新的類型,可以使用struct定義的標識符直接定義變量。
#include
using namespace std;
struct Student
{
const char* name;
int age;
};
int main(int argc, char *argv[])
{
Student s;
s.name = "bauer";
s.age = 20;
return 0;
}
在C語言中,函數(shù)在定義時沒有給出參數(shù)、返回值的類型,默認為int。
int f()表示返回值為int,接受任意參數(shù)的函數(shù)
f(void)表示返回值為int的無參數(shù)函數(shù)
#include
//接受任意個參數(shù),返回int類型
func1()
{
return 30;
}
//無參函數(shù),返回int類型
func2(void)
{
return 20;
}
//參數(shù)i默認為int類型
void func3(i)
{
printf("i = %d\n", i);
}
int main(int argc, char *argv[])
{
int a = func1(1,2,3);
printf("a = %d\n", a);//30
int b = func2();
printf("b = %d\n", b);//20
func3(10);//10
return 0;
}
C++語言中,所有的標識符都必須顯示的聲明類型。C語言中的默認類型在C++中是不合法的。
#include
using namespace std;
//error: ISO C++ forbids declaration of 'func1' with no type [-fpermissive]
func1()
{
return 30;
}
//無參數(shù),返回int
int func2()
{
return 20;
}
int func3(void)
{
return 10;
}
int main(int argc, char *argv[])
{
//error: too many arguments to function 'int func2()'
int a = func2(20);
int b = func2();
int c = func3();
return 0;
}
C語言中,const修飾的變量是只讀的,本質(zhì)還是變量,可以借助指針修改變量空間的值;const修飾的變量會分配存儲空間,const修飾的局部變量分配在棧上,const修飾的全局變量分配在只讀存儲區(qū),修改const修飾的全局變量的值將會導致異常錯誤;const只在編譯期有效,在運行期無效;const關(guān)鍵詞用于向編譯器表明const修飾的變量不能做左值。
#include
//const全局變量,分配在只讀存儲區(qū)
const int number = 10;
int main(int argc, char *argv[])
{
const int c = 0;
int* p = (int*)&c;//編譯器會為c分配空間
printf("Begin...\n");
*p = 5;
printf("c = %d\n", c);//5
c = 10;//error: assignment of read-only variable 'c'
int* cp = &number;
*cp = 100;//程序異常
const int number = 10;
int array[number] = {0};//只讀變量,編譯時無法確定其值
//error: variable-sized object may not be initialized
printf("End...\n");
return 0;
}
C++語言中,對C語言基礎(chǔ)的const進行了優(yōu)化處理。編譯器編譯過程中遇到const修飾的標識符時,會將const修飾的標識符放入符號表中。如果后續(xù)編譯過程中發(fā)現(xiàn)const修飾的標識符時,直接使用符號表中const修飾的標識符對應(yīng)的值直接替換。但在以下情況下C++編譯器會給const聲明的常量分配空間:
A、const修飾的常量為全局(extern修飾),并且需要在其它文件中使用
B、使用&操作符對cosnt常量取地址
C++編譯器雖然會對const常量分配空間,但不會使用其存儲空間的值。
const常量的判別:
A、只有用字面量初始化的const常量才會進入符號表
B、使用其他變量初始化的const常量仍然是只讀變量
C、被volatile修飾的const常量不會進入符號表
const引用類型與初始化變量的類型相同時,初始化變量為只讀變量;不同時,生成一個新的只讀變量。
#include
using namespace std;
int main(int argc, char *argv[])
{
const int a = 10;
//error: invalid conversion from 'const int*' to 'int*'
int* p = &a;
int* cp = (int*)&a;
*cp = 100;
printf("a = %d\n", a);//10
printf("*cp = %d\n", *cp);//100
int b = 3;
//使用其它變量初始化的const常量是只讀變量
const int c = b;
//error: variable-sized object 'array' may not be initialized
int array[c] = {0};
//使用volatile修飾的const常量不會進入符號表
volatile const int d = 10;
//error: variable-sized object 'varray' may not be initialized
int varray[d] = {0};
return 0;
}
C++語言中const與宏定義的不同在于,const常量由編譯器處理,編譯器會對const常量進行類型檢查和作用域檢查,而宏定義由預(yù)處理器進行處理,是單純的文本替換。
#include
using namespace std;
void func1()
{
#define NUMBER 100
const int number = 10;
printf("NUMBER = %d\n",NUMBER);
printf("number = %d\n",number);
}
void func2()
{
//宏定義沒有作用域概念,預(yù)處理時直接替換
printf("NUMBER = %d\n",NUMBER);
printf("number = %d\n",number);
//'number' was not declared in this scope
}
int main(int argc, char *argv[])
{
func1();
func2();
const int number = 10;
//編譯時使用符號表的值替換
int array[number] = {0};
return 0;
}
C語言中,三目運算符返回變量的值,三目運算符表達式不能做左值使用。
C++語言中,三目運算符可直接返回變量本身,三目運算符表達式可以作為左值使用。但是當三目運算符表達式可能返回的值中有一個是常量值,則三目運算符表達式不能作為左值使用。
C++中,當三目運算符表達式可能返回的都是變量時,返回的是變量的引用;當三目運算符表達式可能返回的有常量值時,返回的是值。
#include
using namespace std;
int main(int argc, char *argv[])
{
int a = 3;
int b = 2;
//返回變量本身,可以做左值
(a>b?a:b) = 10;
printf("a>b?a:b = %d\n",a>b?a:b);
//返回變量的值,不能做左值
(a
cin和cout是C++的標準輸入流和輸出流,在頭文件 iostream 中定義。
流名 含義 隱含設(shè)備 流名 含義 隱含設(shè)備
cin 標準輸入 鍵盤 cerr 標準錯誤輸出 屏幕
cout 標準輸出 屏幕 clog cerr 的緩沖輸出 屏幕
int main()
{
char name[30];
int age;
cout<<"pls input name and age:"<>name;
cin>>age;
cout<<"your name is: "<
A、按進制輸出數(shù)據(jù)類型
cout<
B、設(shè)置域?qū)?,設(shè)置左右對齊及填充字符
int main()
{
cout<
C、實型數(shù)據(jù)的設(shè)置
cout<
C語言中不允許重名函數(shù)的存在。
C++語言中為了簡化編程允許重名函數(shù)的存在,即使用同一個函數(shù)名定義不同的函數(shù),重名函數(shù)稱為函數(shù)重載。
重載函數(shù)本質(zhì)是定義的相互獨立的不同函數(shù)。當函數(shù)名和不同的參數(shù)搭配時函數(shù)的含義不同。
int abs(int a)
{
return a>0? a:-a;
}
double abs(double a)
{
return a>0? a:-a;
}
函數(shù)重載的規(guī)則如下:
A、函數(shù)名相同。
B、參數(shù)個數(shù)不同,參數(shù)的類型不同,參數(shù)順序不同,均可構(gòu)成重載。
C、返回值類型不同則不可以構(gòu)成重載。
#include
using namespace std;
int func(int a, int b, int c = 0)
{
return a + b + c;
}
int func(int a, int b)
{
return a + b;
}
int main(int argc, char *argv[])
{
//函數(shù)調(diào)用時出現(xiàn)二義性
int x = func(1,2);
//error: call of overloaded 'func(int, int)' is ambiguous
return 0;
}
編譯器調(diào)用重載函數(shù)的匹配規(guī)則如下:
A、將所有同名函數(shù)作為候選者
B、尋找可行的候選參數(shù)
C、匹配成功或失敗
函數(shù)重載的匹配規(guī)則如下:
A、精確匹配實參,找到則調(diào)用。
B、通過默認參數(shù)能夠匹配實參
C、通過默認類型轉(zhuǎn)換匹配實參
通過默認類型轉(zhuǎn)換匹配實參時,通過隱式轉(zhuǎn)換尋求一個匹配,找到則調(diào)用。
C++允許int到long和double的隱式類型轉(zhuǎn)換,因此在函數(shù)重載時會引起二義性,解決方法是在調(diào)用時強轉(zhuǎn)類型。
#include
using namespace std;
int func(int a, int b)
{
return a + b;
}
double func(double a, double b,double c)
{
cout << "func(double a, double b,double c)"<
編譯器調(diào)用重載函數(shù)匹配失敗的規(guī)則:
A、如果最終找到的候選函數(shù)不唯一,則出現(xiàn)二義性,編譯報錯。
B、如果無法匹配所有候選者,函數(shù)未定義,編譯報錯。
重載函數(shù)使用默認參數(shù)可能會造成二義性。
#include
using namespace std;
int func(int a, int b, int c = 0)
{
return a + b + c;
}
int func(int a, int b)
{
return a + b;
}
int main(int argc, char *argv[])
{
//函數(shù)調(diào)用時出現(xiàn)二義性
int x = func(1,2);
//error: call of overloaded 'func(int, int)' is ambiguous
return 0;
}
C++利用name mangling(傾軋)技術(shù),來改名函數(shù)名,區(qū)分參數(shù)不同的同名函數(shù)。
C++的name mangling實現(xiàn)使用 v c i f l d表示void char int float long double及其引用。
void func(char a); // func_c(char a)
void func(char a, int b, double c);//func_cid(char a, int b, double c);
name mangling發(fā)生在兩個階段,.cpp編譯階段和.h的聲明階段。只有兩個階段同時進行,才能匹配調(diào)用。
#include
using namespace std;
//函數(shù)類型:int(int,int)
int func(int a, int b)
{
return a + b;
}
//函數(shù)類型:double(double, double, double)
double func(double a, double b,double c)
{
cout << "func(double a, double b,double c)"<
使用nm工具查看main.o文件中符號表信息的命令如下:nm.exe -a main.o
func重載函數(shù)的符號表信息如下:
00000036 T __Z4funcddd
00000029 T __Z4funcii
00000090 T __Z4funclll
上述的信息表示代碼中的三個重載函數(shù)。
將重載函數(shù)名賦值給函數(shù)指針時,根據(jù)重載規(guī)則選擇與函數(shù)指針參數(shù)列表一致的函數(shù)。重載函數(shù)的函數(shù)類型與函數(shù)指針類型必須嚴格匹配(不能有任何類型的隱式轉(zhuǎn)換),此時函數(shù)返回類型將參與函數(shù)類型匹配。
函數(shù)重載必須發(fā)生在同一個作用域,無法通過函數(shù)名得到重載函數(shù)的入口地址。
重載函數(shù)的函數(shù)類型不同。
#include
using namespace std;
//函數(shù)類型:int(int,int)
typedef int(*pFunc1)(int,int);
int func(int a, int b)
{
cout << "func(int a, int b)"<
重載函數(shù)的調(diào)用可能會存在隱式類型轉(zhuǎn)換,比如int到long、double類型的轉(zhuǎn)換,但是要函數(shù)指針調(diào)用重載函數(shù)時,函數(shù)指針的類型必須與重載函數(shù)的類型嚴格匹配。
函數(shù)重載的注意事項如下:
A、函數(shù)重載必然發(fā)生在同一個作用域中。
B、編譯器需要使用參數(shù)列表或函數(shù)類型進行函數(shù)的選擇。
C、不能直接通過函數(shù)名得到重載函數(shù)的入口地址。
C++完全兼容C語言,因此必須完全兼容C的類庫。由于.c文件的類庫文件中函數(shù)名并沒有發(fā)生name manling行為,而在包含.c文件所對應(yīng)的.h文件時,.h 文件要發(fā)生name manling行為,因而會在編譯鏈接時候發(fā)生錯誤。
C++為了避免上述錯誤的發(fā)生,重載了關(guān)鍵字extern。只需要要避免name manling的函數(shù)前,加extern "C"如有多個,則extern "C"{}。
C語言標準庫中實際上對C++語言程序引用時做了特殊處理,在C++語言編譯器編譯時使用extern "C"將C語言的標準庫函數(shù)排除了命名傾軋。
為了確保無論在C、C++編譯器中C代碼以C語言方式編譯:
#ifdef __cplusplus
extern "C"{
#endif
//c-style code
#ifdef __cplusplus
}
#endif
C++編譯器不能以C語言方式編譯重載函數(shù),C++編譯器將函數(shù)名和參數(shù)列表編譯為目標名,C語言編譯方式只將函數(shù)名作為目標名進行編譯。
C++調(diào)用C語言編碼的.dll時,當包含.dll的頭文件或聲明接口函數(shù)時需要加extern “C”。
add.h源碼:
#ifndef ADD_H
#define ADD_H
extern int add(int a, int b);
#endif
add.c源碼:
#include "add.h"
int add(int a, int b)
{
return a + b;
}
main.cpp源碼:
#include
extern "C"
{
#include "add.h"
}
int main()
{
int c = add(10, 100);
printf("%d\n", c);
return 0;
}
gcc add.c -o add.o
g++ add.o main.cpp
C代碼中引用C++的函數(shù)和變量時,C++頭文件需要添加extern “C”,但在C代碼中不能直接引用聲明了extern “C”的C++頭文件,C代碼中只需要將C++中定義的extern “C”函數(shù)聲明為extern類型即可。
add.h源碼:
#ifndef ADD_H
#define ADD_H
extern "C" int add(int a, int b);
#endif
add.cpp源碼:
#include "add.h"
int add(int a, int b)
{
return a + b;
}
main.c源碼:
#include
//#include "add.h" 錯誤
extern int add(int a, int b);
int main()
{
int x = add(1, 2);
printf("x = %d\n", x);
return 0;
}
編譯:
g++ -c add.cpp -o add.o
gcc main.c add.o -lstdc++
或是gcc add.cpp main.c -lstdc++
g++會自動進行C++標準庫的連接;用gcc連接C++程序也可以,但需要人為指定連接C++標準庫(-lstdc++),否則就會出現(xiàn)undefined reference to __gxx_personality_v/0
之類的錯誤。
C++提供了運算符重載機制??梢詾樽远x數(shù)據(jù)類型重載運算符。實現(xiàn)構(gòu)造數(shù)據(jù)類型也可以像基本數(shù)據(jù)類型一樣的運算特性。
struct COMP
{
float real;
float image;
};
COMP operator+(COMP one, COMP another)
{
one.real += another.real;
one.image += another.image;
return one;
}
int main()
{
COMP c1 = {1,2};
COMP c2 = {3,4};
COMP sum = operator+(c1,c2); //c1+c2;
cout<
實例代碼重載了一個全局的操作符+號用于實現(xiàn)將兩個自定義結(jié)構(gòu)體類型相加。本質(zhì)是函數(shù)的調(diào)用。
C++語言中,可以在函數(shù)聲明時為參數(shù)提供一個默認值。當函數(shù)調(diào)用沒有提供參數(shù)的值時,使用默認值。
函數(shù)默認參數(shù)的規(guī)則如下:
A、默認參數(shù)的順序,是從右向左,不能跳躍。
B、定義在前,調(diào)用在后(此時定義和聲明為一體),默認參數(shù)在定義處;聲明在前,調(diào)用在后,默認參數(shù)在聲明處。
C、一個函數(shù),不能既作重載,又作默認參數(shù)的函數(shù)。當你少寫一個參數(shù)時,系統(tǒng)無法確認是重載還是默認參數(shù)。
函數(shù)調(diào)用時參數(shù)從左到右匹配,如果一個參數(shù)使用了默認值,則后續(xù)參數(shù)必須使用默認值。
A、單個參數(shù)
#include
#include
using namespace std;
void weatherForcast(char * w="sunny")
{
time_t t = time(0);
char tmp[64];
strftime( tmp, sizeof(tmp), "%Y/%m/%d %X %A ",localtime(&t) );
cout<
B、多個參數(shù)
#include
using namespace std;
float volume(float length, float weight = 4,float high = 5)
{
return length*weight*high;
}
int main(int argc, char *argv[])
{
float v = volume(10);
float v1 = volume(10,20);
float v2 = volume(10,20,30);
cout<
C++語言中可以為函數(shù)提供占位參數(shù),占位參數(shù)只有類型聲明,沒有參數(shù)名聲明。由于C++類型檢查較為嚴格,為兼容C語言,可以將函數(shù)參數(shù)默認值和占位參數(shù)結(jié)合使用。
C語言中func函數(shù)如下:
#include
void func()
{
}
int main(int argc, char *argv[])
{
func(5,10);
return 0;
}
C語言中func函數(shù)接收任意個數(shù)的參數(shù)。
C++語言中對func函數(shù)增加占位參數(shù)可以使C語言中的func函數(shù)快速地滿足C++語言的語法要求,代碼如下:
#include
using namespace std;
void func(int x, int = 0)
{
}
int main(int argc, char *argv[])
{
func(5,10);
return 0;
}
變量名,本身是一段內(nèi)存的引用,即別名(alias)。引用,是為己有變量起一個別名。Type& name = var;
int a;
int &b = a;
普通引用在定義時必須使用同類型的變量進行初始化。
A、引用沒有定義,是一種關(guān)系型聲明。聲明它和原有某一變量(實體)的關(guān)系。故 而類型與原類型保持一致,且不分配內(nèi)存,與被引用的變量有相同的地址。
B、聲明的時候必須初始化,一經(jīng)聲明,不可變更。
C、可對引用再次引用。多次引用的結(jié)果,是某一變量具有多個別名。
D、&符號前有數(shù)據(jù)類型時,是引用。其它皆為取地址。
#include
using namespace std;
int main(int argc, char *argv[])
{
int a = 3;
int c = 6;
float f = 3.14;
int& b = a;
printf("&a = 0x%X\n", &a);
printf("&b = 0x%X\n", &b);
//error: redeclaration of 'int& b'
int& b = c;//error
//給b賦值
b = c;
//引用的類型必須與變量類型相同
int& d = f;//error
//error: invalid initialization of reference of type 'int&' from expression of type 'float'
//引用在定義時必須初始化
int& rd;//error
//error: 'rd' declared as reference but not initialized
//引用不可以使用字面值初始化
int& r = 10;//error
//error: invalid initialization of non-const reference of type 'int&' from an rvalue of type 'int'
return 0;
}
函數(shù)中的引用形參不需要進行初始化,函數(shù)調(diào)用時進行初始化。
#include
using namespace std;
//引用
void swap(int &a, int &b)
{
int tmp;
tmp = a;
a = b;
b = tmp;
printf("swap(int &a, int &b)\n");
}
//指針
void swap(int* a, int* b)
{
int tmp;
tmp = *a;
*a = *b;
*b = tmp;
printf("swap(int* a, int* b)\n");
}
int main(int argc, char *argv[])
{
int a = 3;
int b = 6;
swap(a,b);
printf("a = %d\n",a);//6
printf("b = %d\n",b);//3
swap(&a,&b);
printf("a = %d\n",a);//3
printf("b = %d\n",b);//6
return 0;
}
4、引用的提高
A、可以定義指針的引用,但不能定義引用的引用。
int a;
int* p = &a;
int*& rp = p; // ok
int& r = a;
int&& rr = r; // error
B、可以定義指針的指針(二級指針),但不能定義引用的指針。
int a;
int* p = &a;
int** pp = &p; // ok
int& r = a;
int&* pr = &r; // error
C、可以定義指針數(shù)組,但不能定義引用數(shù)組,可以定義數(shù)組引用。
int a, b, c;
int* parr[] = {&a, &b, &c}; // ok
int& rarr[] = {a, b, c}; // error
int arr[] = {1, 2, 3};
int (&rarr)[3] = arr; // ok 的
數(shù)組是連續(xù)的存儲空間,數(shù)組中的元素如果是引用,會導致數(shù)組的元素存儲不連續(xù)。引用數(shù)組會破壞數(shù)組存儲空間的連續(xù)性。
const引用所引用的對象必須是const的,將普通引用綁定到const引用對象是不合法的。const type& name = var;
const引用可使用相關(guān)類型的對象(常量,非同類型的變量或表達式)初始化,const引用讓變量具有只讀屬性,是const引用與普通引用大的區(qū)別。
非const引用只能綁定到與該引用同類型的對象。
當const引用使用字面常量值初始化時,C++編譯器會為常量值分配空間,使用字面常量對const引用初始化將生成一個只讀變量。
#include
using namespace std;
int main(int argc, char *argv[])
{
int a = 10;
const int& ra = a;//const引用,為只讀變量
//只讀變量不能作為左值
ra = 100;//error: assignment of read-only reference 'c'
int* p = (int*)&ra;
*p = 5;
printf("ra = %d\n", ra);//5
const int& rb = 10;//rb為只讀變量,占用內(nèi)存空間
//只讀變量不能作為左值
rb = 100;//error
int arraya[ra] = {0};//error
//error: variable-sized object 'arraya' may not be initialized
int arrayb[rb] = {0};//error
//error: variable-sized object 'arrayb' may not be initialized
double pi = 3.14;
int& rpi = pi;//非法
//error: invalid initialization of reference of type 'int&' from expression of type 'double'
const int& crpi = pi;//合法
printf("rpi= %d\n",crpi);//3
return 0;
}
C++編譯器在編譯過程中使用指針常量作為引用的內(nèi)部實現(xiàn),因此引用所占用空間大小與指針相同。引用的本質(zhì)是一個指針常量。type & name<====> type * const name;
使用引用時不能返回局部變量的引用
#include
using namespace std;
int main(int argc, char *argv[])
{
printf("sizeof(char&) = %d\n", sizeof(char&));//1
char c = 'a';
char& rc = c;
printf("sizeof(char&) = %d\n", sizeof(rc));//1
return 0;
}
上述代碼的匯編代碼如下:
將字符’a’(0x61)存儲到指針寄存器0x1b,將0x1b放入eax數(shù)據(jù)寄存器中,再將eax數(shù)據(jù)寄存器存儲內(nèi)容放入指針寄存器0x1c中。
C語言中提供了malloc和free兩個系統(tǒng)函數(shù),完成對堆內(nèi)存的申請和釋放。而C++則提供了兩關(guān)鍵字new和delete。
new分配內(nèi)存空間時,不能保證按需分配,分配內(nèi)存空間大小可能會大于所需空間大小。因此,new會分配至少申請大小的內(nèi)存空間。
A、開辟單變量地址空間
int *p = new int; //開辟大小至少為sizeof(int)空間
int *a = new int(5); //開辟大小至少為sizeof(int)空間,并初始化為 5
B、開辟數(shù)組空間
一維: int a = new int[100];//開辟一個大小不少于400字節(jié)的整型數(shù)組空間
二維: int (a)[6] = new int[5][6]
三維: int (*a)[5][6] = new int[3][5][6]
A、釋放單變量空間
int *a = new int;
delete a; //釋放單個 int 的空間
B、釋放數(shù)組空間
int *a = new int[5];
delete []a; //釋放 int 數(shù)組空間
int *p = new int(5);
cout<<*p<
//C++ 內(nèi)存申請失敗會拋出異常
try{
int *p = new int[10];
}
catch(const std::bad_alloc e)
{
return -1;
}
//C++ 內(nèi)存申請失敗不拋出異常版本
int *q = new (std::nothrow)int[10];
if(q == NULL)
return -1;
C++中堆空間的分配和釋放注意事項如下:
A、new/delete 是關(guān)鍵字,效率高于 malloc 和 free.
B、配對使用,避免內(nèi)存泄漏和多重釋放。
C、避免交叉使用。比如 malloc 申請的空間去 delete,new 出的空間被 free;
D、重點用在類對像的申請與釋放。申請的時候會調(diào)用構(gòu)造器完成初始化,
釋放的時候,會調(diào)用析構(gòu)器完成內(nèi)存的清理。
malloc與new的區(qū)別如下:
A、new是C++關(guān)鍵字,malloc是C語言庫函數(shù)
B、new以具體類型為單位進行內(nèi)存分配,malloc以字節(jié)位單位分配內(nèi)存
C、new在申請單個類型變量時可以進行初始化,malloc不具備
D、new在所有C++編譯器中都支持,malloc在某些系統(tǒng)開發(fā)中不可調(diào)用
E、new能夠觸發(fā)構(gòu)造函數(shù)的調(diào)用,malloc僅分配需要的內(nèi)存空間
F、對象的創(chuàng)建只能使用new,malloc不適合面向?qū)ο箝_發(fā)
free與delete的區(qū)別如下:
A、delete是C++關(guān)鍵字,free是庫函數(shù)
B、delete在所有C++編譯器中都支持,free在某些系統(tǒng)開發(fā)中不可調(diào)用
C、delete能夠觸發(fā)析構(gòu)函數(shù)的調(diào)用,free僅歸還分配的內(nèi)存空間
D、對象的銷毀只能使用delete,free不適合面向?qū)ο箝_發(fā)
E、free可以歸還new申請的內(nèi)存空間,但不會調(diào)用析構(gòu)函數(shù),可能會造成內(nèi)存泄漏
F、delete可以釋放malloc分配的內(nèi)存空間,但會調(diào)用析構(gòu)函數(shù),會造成其他問題。
C語言中有宏函數(shù)的概念。宏函數(shù)的特點是內(nèi)嵌到調(diào)用代碼中去,避免了函數(shù)調(diào)用的開銷。但宏函數(shù)的處理發(fā)生在預(yù)處理階段,缺少作用域檢查和類型檢查。
C++提供了inline關(guān)鍵字,請求C++編譯器將一個函數(shù)進行內(nèi)聯(lián)編譯(C++編譯器可以拒絕),C++編譯器會直接將內(nèi)聯(lián)的函數(shù)體代碼插入函數(shù)調(diào)用的地方。
內(nèi)聯(lián)函數(shù)聲明時inline關(guān)鍵字必須和函數(shù)定義結(jié)合在一起,否則編譯器會直接忽略內(nèi)聯(lián)請求。
inline int sqr(int x)
{
return x*x;
}
內(nèi)聯(lián)函數(shù)的優(yōu)點:避免調(diào)用時的額外開銷(入棧與出棧操作)
內(nèi)聯(lián)函數(shù)的缺點:內(nèi)聯(lián)函數(shù)的函數(shù)體在代碼段中會出現(xiàn)多個“副本”,因此會增加代碼段的空間。
內(nèi)聯(lián)函數(shù)的本質(zhì):以犧牲代碼段空間為代價,提高程序的運行時間的效率。
內(nèi)聯(lián)函數(shù)的適用場景:函數(shù)體很“小”,且被“頻繁”調(diào)用。
Inline關(guān)鍵字是對編譯器的建議,如果編譯器認為inline聲明的函數(shù)可以內(nèi)聯(lián),則編譯器會將函數(shù)內(nèi)聯(lián),如果編譯器認為inline聲明的函數(shù)的函數(shù)體太長,則不會內(nèi)聯(lián),按普通函數(shù)處理。
using namespace std;
inline int func(int a, int b)
{
return a*a + b*b;
}
int main(int argc, char *argv[])
{
int a = 3;
int b = 4;
int c = func(a,b);
return 0;
}
上述代碼在QtCreator+MinGW編譯器下調(diào)試時,查看匯編代碼如下:
調(diào)用func函數(shù)時,C++編譯器沒有將func函數(shù)內(nèi)聯(lián),仍然使用函數(shù)調(diào)用。
inline聲明的函數(shù),內(nèi)聯(lián)請求可能被C++編譯器拒絕。
現(xiàn)代C++編譯器能夠進行編譯優(yōu)化,一些函數(shù)即使沒有inline關(guān)鍵字聲明也能夠內(nèi)聯(lián)編譯,同時現(xiàn)代C++編譯器提供了擴展的語法,能夠?qū)瘮?shù)進行強制內(nèi)聯(lián)。如現(xiàn)代G++編譯器使用__attribute__((always_inline))
聲明強制內(nèi)聯(lián),MSVC編譯器使用__forceinline
聲明強制內(nèi)聯(lián),不再使用inline關(guān)鍵字。
#include
using namespace std;
__attribute__((always_inline))
int func(int a, int b);
int main(int argc, char *argv[])
{
int a = 3;
int b = 4;
int c = func(a,b);
return 0;
}
int func(int a, int b)
{
return a*a + b*b;
}
上述代碼在QtCreator+MinGW編譯器中進行調(diào)試時,main函數(shù)中func函數(shù)調(diào)用代碼的匯編代碼如下:
MinGW編譯器已經(jīng)對func函數(shù)進行了內(nèi)聯(lián)。
C++中使用inline關(guān)鍵字內(nèi)聯(lián)編譯函數(shù)的限制:
A、不能存在任何形式的循環(huán)語句
B、不能存在過多的條件判斷語句
C、函數(shù)體不能過于龐大
D、不能對函數(shù)進行取地址操作
E、函數(shù)內(nèi)聯(lián)聲明必須在調(diào)用語句前
對于現(xiàn)代C++編譯器的擴展語法提供的強制內(nèi)聯(lián)不受上述條件限制。
C語言中的類型轉(zhuǎn)換是強制轉(zhuǎn)換,任何類型間都可以轉(zhuǎn)換,過于粗暴。
C++語言引入了static_cast、dynamic_cast、const_cast、reinterpret_cast四個關(guān)鍵字處理不同類型間的轉(zhuǎn)換。
靜態(tài)類型轉(zhuǎn)換是在編譯期內(nèi)即可決定其類型的轉(zhuǎn)換。
靜態(tài)類型轉(zhuǎn)換的使用場合:
A、用于基本類型間的轉(zhuǎn)換
B、不能用于基本類型指針間的轉(zhuǎn)換
C、用于有繼承關(guān)系類對象間的轉(zhuǎn)換和類指針間的轉(zhuǎn)換(轉(zhuǎn)換一般從子對象向父對象轉(zhuǎn)換)
語法格式:
static_cast<目標類型> (標識符)
應(yīng)用實例:
#include
class A
{
private:
int a;
int b;
public:
A()
{
a = 0;
b = 0;
}
void print()
{
printf("a = %d, b = %d\n", a, b);
}
};
class B : public A
{
private:
int c;
int d;
public:
void display()
{
printf("c = %d, d = %d\n", c, d);
}
};
class C
{
public:
void print()
{
printf("hello\n");
}
};
int main()
{
float f = static_cast(9)/10;//基本類型的轉(zhuǎn)換
printf("f = %f\n", f);
A a;
B b;
A aa = static_cast(b);//將子類對象轉(zhuǎn)換為父類對象
aa.print();
//B bb = static_cast(a);//不能將父對象轉(zhuǎn)換為子對象
A* pa = static_cast(&a);//在同類型對象指針間轉(zhuǎn)換
pa->print();
pa = static_cast(&b);//將子類對象指針轉(zhuǎn)換為父類對象指針
pa->print();
B* pb = static_cast(&b);
pb->display();
pb = static_cast(&a);
pb->display();
//C c = static_cast(a);//沒有轉(zhuǎn)換構(gòu)函數(shù),不能將A類型轉(zhuǎn)換為C類型
//c.print();
return 0;
}
動態(tài)類型轉(zhuǎn)換的使用場合:
A、用于有繼承關(guān)系的類指針間的轉(zhuǎn)換
B、有交叉關(guān)系的類指針間的轉(zhuǎn)換
C、具有類型檢查
D、必須有虛函數(shù)支持
語法格式:
dynamic_cast<目標類型> (標識符)
用于有直接或間接繼承關(guān)系的指針(引用)的強制轉(zhuǎn)換
轉(zhuǎn)換指針成功將會得到目標類型的指針,轉(zhuǎn)換失敗將得到一個空指針;
轉(zhuǎn)換引用成功將得到目標類型的引用,轉(zhuǎn)換失敗將得到一個異常操作信息。
使用實例:
#include
#include
using namespace std;
class Base
{
public:
Base()
{
cout << "Base::Base()" << endl;
}
virtual ~Base()
{
cout << "Base::~Base()" << endl;
}
};
class Derived : public Base
{
};
int main()
{
Base* p = new Derived;
Derived* pd = dynamic_cast(p);//將指向子類對象的父類指針轉(zhuǎn)換為子類指針,轉(zhuǎn)換成功
if( pd != NULL )
{
cout << "pd = " << pd << endl;
}
else
{
cout << "Cast error!" << endl;
}
delete p;
cout << endl;
p = new Base;
pd = dynamic_cast(p);//將指向父類對象的父類指針轉(zhuǎn)換為子類指針,轉(zhuǎn)換失敗
if( pd != NULL )
{
cout << "pd = " << pd << endl;
}
else
{
cout << "Cast error!" << endl;
}
delete p;
return 0;
}
常量類型轉(zhuǎn)換的使用場合:
A、用于去除變量的只讀屬性
B、目標類類型只能是指針或引用
語法格式:
const_cast<目標類型> (標識符) //目標類類型只能是指針或引用。
const_cast將轉(zhuǎn)換掉表達式的const屬性
應(yīng)用實例:
#include
using namespace std;
int main()
{
const int& a = 10;
int& b = const_cast(a);//將a的只讀屬性去除,并初始化b
b = 0;
cout << "a = " << a << endl;//0
cout << "b = " << b << endl;//0
const int c = 100;
int& d = const_cast(c);//為c分配一個只讀空間
//int x = const_cast(c);//error,目標類型只能為指針和引用
d = 1000;
cout << "c = " << c << endl;//100
cout << "d = " << d << endl;//1000
return 0;
}
重解釋類型轉(zhuǎn)換使用場合:
A、用于指針類型間的強制轉(zhuǎn)換
B、用于整數(shù)和指針類型間的強制轉(zhuǎn)換
語法格式:
reinterpret_cast<目標類型> (標識符)
為數(shù)據(jù)的二進制形式重新解釋,但是不改變其值。
使用實例:
#include
using namespace std;
int main(int argc, char *argv[])
{
int a = 100;
char c = 'a';
int* pa = reinterpret_cast(&c);
printf("*pa = %d\n", *pa);//687781729
printf("*pa = %c\n", *pa);//'a'
//int x = reinterpret_cast(c);//error
//int y = reinterpret_cast(1.1);//error
return 0;
}
C語言中,只有一個全局作用域,所有的全局標識符共享一個作用域,因此標識符之間可能存在沖突。
C++語言中,提出了命名空間的概念。命名空間將全局作用域分為不同的部分,不同命令空間中的標識符可以重名而不會發(fā)生沖突,命名空間可以嵌套。全局作用域即默認命名空間。
global scope是一個程序中大的scope,是引起命名沖突的根源。C語言沒有從語言層面提供命名空間機制來解決。global scope是無名的命名空間。
NameSpace是對全局區(qū)域的再次劃分。
命名空間的聲明如下:
namespace NAMESPACE
{
全局變量 int a;
數(shù)據(jù)類型 struct Stu{};
函數(shù) void func();
}
直接指定命名空間: NameSpace::a = 5;
使用using+命名空間+空間元素:using NameSpace::a; a = 2000;
使用using +namespace+命名空間;
#include
using namespace std;
namespace MySpace
{
int x = 1;
int y = 2;
}
namespace Other {
int x = 3;
int y = 4;
}
int main()
{
{
using namespace MySpace;
cout<
可以使用塊語句將命名空間限定在塊語句內(nèi)部。
namespace MySpace
{
int x = 1;
int y = 2;
namespace Other {
int m = 3;
int n = 4;
}
}
在實際項目開發(fā)中,可以將一個類或者具有相同屬性的多個類聲明在一個命名空間內(nèi),在使用時只需要聲明命名空間即可。
#ifndef A_H
#define A_H
namespace XX {
class A
{
public:
A();
~A();
};
}
#endif // A_H
#include "a.h"
using namespace XXX
{
A::A()
{}
A::~A()
{}
}
除了使用字符數(shù)組來處理字符串以外,C++引入了字符串類型??梢远x字符串變量。
string str;
str = "china";
string str2 = " is great ";
string str3 = str2;
string str = "china";
cout << sizeof(str) << " " << str.max_size() << " " << str.size()<
4 1073741820 5
A、賦值string str3 = str2;
B、加法string combine = str + str2;
C、關(guān)系
string s1 = "abcdeg";
string s2 = "12345";
if(s1>s2)
cout<<"s1>s2"<
string數(shù)組是高效的,如果用二維數(shù)組來存入字符串數(shù)組的話,則容易浪費空間,此時列數(shù)是由最長的字符串決定。如果用二級指針申請堆空間,依據(jù)大小申請相應(yīng)的空間,雖然解決了內(nèi)存浪費的問題,但是操作麻煩。用 string 數(shù)組存儲,字符串數(shù)組的話,效率即高又靈活。
string sArray[10] = {
"0",
"1",
"22",
"333",
"4444",
"55555",
"666666",
"7777777",
"88888888",
"999999999",
};
for(int i=0; i<10; i++)
{
cout<
int capacity()const;??//返回當前容量(即string中不必增加內(nèi)存即可存放的元素個數(shù))
int max_size()const;??? //返回string對象中可存放的大字符串的長度
int size()const;??????? //返回當前字符串的大小
int length()const;?????? //返回當前字符串的長度
bool empty()const;??????? //當前字符串是否為空
void resize(int len,char c);//把字符串當前大小置為len,并用字符c填充不足的部分
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機、免備案服務(wù)器”等云主機租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。