av手机免费在线观看,国产女人在线视频,国产xxxx免费,捆绑调教一二三区,97影院最新理论片,色之久久综合,国产精品日韩欧美一区二区三区

試題

計(jì)算機(jī)二級(jí)《C++》考點(diǎn)解析:堆和類(lèi)數(shù)組

時(shí)間:2025-05-09 11:42:35 試題 我要投稿
  • 相關(guān)推薦

計(jì)算機(jī)二級(jí)《C++》考點(diǎn)解析:堆和類(lèi)數(shù)組

  為幫助同學(xué)們更好更有準(zhǔn)備地復(fù)習(xí)計(jì)算機(jī)二級(jí)C++,以下是百分網(wǎng)小編搜索整理的計(jì)算機(jī)二級(jí)《C++》考點(diǎn)解析:堆和類(lèi)數(shù)組,供參考復(fù)習(xí),希望對(duì)大家有所幫助!想了解更多相關(guān)信息請(qǐng)持續(xù)關(guān)注我們應(yīng)屆畢業(yè)生考試網(wǎng)!

  一、構(gòu)造函數(shù)和析構(gòu)函數(shù)

  前面的例子已經(jīng)運(yùn)用了new和來(lái)為類(lèi)對(duì)象分配和釋放內(nèi)存。當(dāng)使用new為類(lèi)對(duì)象分配內(nèi)存時(shí),編譯器首先用new運(yùn)算符分配內(nèi)存,然后調(diào)用類(lèi)的構(gòu)造函數(shù);類(lèi)似的,當(dāng)使用來(lái)釋放內(nèi)存時(shí),編譯器會(huì)首先調(diào)用淚的析構(gòu)函數(shù),然后再調(diào)用運(yùn)算符。

  #include iostream.h

  class Date

  {

  int mo,da,yr;

  public:

  Date() { cout < ~Date() { cout< }

  int main()

  {

  Date* dt = new Date;

  cout < dt;

  return 0;

  }

  程序定義了一個(gè)有構(gòu)造函數(shù)和析構(gòu)函數(shù)的Date類(lèi),這兩個(gè)函數(shù)在執(zhí)行時(shí)會(huì)顯示一條信息。當(dāng)new運(yùn)算符初始化指針dt時(shí),執(zhí)行了構(gòu)造函數(shù),當(dāng)運(yùn)算符釋放內(nèi)存時(shí),又執(zhí)行了析構(gòu)函數(shù)。

  程序輸出如下:

  Date constructor

  Process the date

  Date destructor

  二、堆和類(lèi)數(shù)組

  前面提到,類(lèi)對(duì)象數(shù)組的每個(gè)元素都要調(diào)用構(gòu)造函數(shù)和析構(gòu)函數(shù)。下面的例子給出了一個(gè)錯(cuò)誤的釋放類(lèi)數(shù)組所占用的內(nèi)存的例子。

  #include iostream.h

  class Date

  {

  int mo, da, yr;

  public:

  Date() { cout < ~Date() { cout< }

  int main()

  {

  Date* dt = new Date[5];

  cout < dt; //這兒

  return 0;

  }

  指針dt指向一個(gè)有五個(gè)元素的數(shù)組。按照數(shù)組的定義,編譯器會(huì)讓new運(yùn)算符調(diào)用Date類(lèi)的構(gòu)造函數(shù)五次。但是被調(diào)用時(shí),并沒(méi)有明確告訴編譯器指針指向的Date對(duì)象有幾個(gè),所以編譯時(shí),只會(huì)調(diào)用析構(gòu)函數(shù)一次。下面是程序輸出;

  Date constructor

  Date constructor

  Date constructor

  Date constructor

  Date constructor

  Process the date

  Date destructor

  為了解決這個(gè)問(wèn)題,C++允許告訴運(yùn)算符,正在刪除的那個(gè)指針時(shí)指向數(shù)組的,程序修改如下:

  #include iostream.h

  class Date

  {

  int mo, da, yr;

  public:

  Date() { cout < ~Date() { cout< }

  int main()

  {

  Date* dt = new Date[5];

  cout < [] dt; //這兒

  return 0;

  }

  最終輸出為:

  Date constructor

  Date constructor

  Date constructor

  Date constructor

  Date constructor

  Process the date

  Date destructor

  Date destructor

  Date destructor

  Date destructor

  Date destructor

  三、重載new和運(yùn)算符

  前面已經(jīng)介紹了如何用new和運(yùn)算符函數(shù)來(lái)動(dòng)態(tài)第管理內(nèi)存,在那些例子中使用的都是全局的new和運(yùn)算符。我們可以重載全局的new和運(yùn)算符,但這不是好的想法,除非在進(jìn)行低級(jí)的系統(tǒng)上或者嵌入式的編程。

  但是,在某個(gè)類(lèi)的內(nèi)部重載new和運(yùn)算符時(shí)可以的。這允許一個(gè)類(lèi)有它自己的new和運(yùn)算符。當(dāng)一個(gè)類(lèi)需要和內(nèi)存打交道時(shí),采用這種方法來(lái)處理其中的細(xì)節(jié),可以獲得很搞的效率,同時(shí)避免了使用全局new和運(yùn)算符帶來(lái)的額外開(kāi)銷(xiāo)。因?yàn)槿侄巡僮鲿r(shí)調(diào)用操作系統(tǒng)函數(shù)來(lái)分配和釋放內(nèi)存,這樣效率很低。

  如果確定某個(gè)類(lèi)在任何時(shí)候,其實(shí)例都不會(huì)超過(guò)一個(gè)確定的值,那么就可以一次性為類(lèi)的所有實(shí)例分配足夠的內(nèi)存,然后用該類(lèi)的new和運(yùn)算符來(lái)管理這些內(nèi)存。下面的程序說(shuō)明了如何對(duì)new和進(jìn)行重載。

  #include iostream.h

  #include string.h

  #include stddef.h

  #include new.h

  const int maxnames = 5;

  class Names

  {

  char name[25];

  static char Names::pool[];

  static bool Names::inuse[maxnames];

  public:

  Names(char* s) { strncpy(name,s,sizeof(name)); }

  void* operator new(size_t) throw(bad_alloc);

  void operator (void*) throw();

  void display() const { cout < };

  char Names::pool[maxnames * sizeof(Names)];

  bool Names::inuse[maxnames];

  void* Names::operator new(size_t) throw(bad_alloc)

  {

  for(int p=0; p {

  if(!inuse[p])

  {

  inuse[p] = true;

  return pool+p*sizeof(Names);

  }

  }

  throw bad_alloc();

  }

  void Names::operator (void* p) throw()

  {

  if(p!=0)

  inuse[((char*)p - pool)/sizeof(Names)] = false;

  }

  int main()

  {

  Names* nm[maxnames];

  int i;

  for(i=0; i {

  cout < char name[25];

  cin >> name;

  nm[i] = new Names(name);

  }

  for(i=0; i {

  nm[i]- >display();

   nm[i];

  }

  return 0;

  }

  上面的程序提示輸入5個(gè)姓名,然后顯示它們。程序中定義了名為Names的類(lèi),它的構(gòu)造函數(shù)初始化對(duì)象的name值。這個(gè)類(lèi)定義了自己的new和運(yùn)算符。這是因?yàn)槌绦蚰鼙WC不會(huì)一次使用超過(guò)maxnames個(gè)姓名,所以可以通過(guò)重載默認(rèn)的new和運(yùn)算符來(lái)提高運(yùn)行速度。

  Names類(lèi)中的內(nèi)存池是一個(gè)字符數(shù)組,可以同時(shí)容納程序需要的所有姓名。與之相關(guān)的布爾型數(shù)組inuse為每個(gè)姓名記錄了一個(gè)true和false值,指出內(nèi)存中的對(duì)應(yīng)的項(xiàng)是否正在使用。

  重載的new運(yùn)算符在內(nèi)存池中尋找一個(gè)沒(méi)有被使用的項(xiàng),然后返回它的地址。重載的運(yùn)算符則標(biāo)記那些沒(méi)有被使用的項(xiàng)。

  在類(lèi)定義中重載的new和運(yùn)算符函數(shù)始終是靜態(tài)的,并且沒(méi)有和對(duì)象相關(guān)的this指針。這是因?yàn)榫幾g器會(huì)在調(diào)用構(gòu)造函數(shù)之前調(diào)用new函數(shù),在調(diào)用析構(gòu)函數(shù)后調(diào)用函數(shù)。

  new函數(shù)是在類(lèi)的構(gòu)造函數(shù)之前被調(diào)用的。因?yàn)檫@時(shí)內(nèi)存中還不存在類(lèi)的對(duì)象而且構(gòu)造函數(shù)也沒(méi)有提供任何初始化值,所以它不可以訪(fǎng)問(wèn)類(lèi)的任何成員。同理,運(yùn)算符是在析構(gòu)函數(shù)之后被調(diào)用的,所以它也不可以訪(fǎng)問(wèn)類(lèi)的成員。

  四、異常監(jiān)測(cè)和異常處理

  1.檢測(cè)異常

  上面的例子還缺少必要的保護(hù)機(jī)制。比如,重載的運(yùn)算符函數(shù)并沒(méi)有檢查它的參數(shù),確認(rèn)其是否落在內(nèi)存池內(nèi)部。如果你絕對(duì)相信自己編的程序中不會(huì)傳遞錯(cuò)誤的指針值給運(yùn)算符,那么可以省掉合法性檢查以提高效率,特別是在優(yōu)先考慮效率的程序中。否則應(yīng)該使用預(yù)編譯的條件語(yǔ)句。在軟件的測(cè)試版本中加入這些檢測(cè),在正式的發(fā)行版本中去掉這些檢查。

  2.重載new和中的異常處理

  上面的兩個(gè)重載運(yùn)算符函數(shù)都是用了異常處理。異常處理是C++的新內(nèi)容之一,目前還沒(méi)有講到。在這里不必關(guān)心它是如何工作的。上面程序中,當(dāng)試圖分配超過(guò)內(nèi)存池容量的Names緩沖區(qū),重載的new運(yùn)算符函數(shù)就會(huì)拋出異常,終止程序。

  --------------------------------------------------------------------------------

  五、重載new[]和[]

  對(duì)于上面的程序,假如有下面的語(yǔ)句:

  Names *nms=new Names[10]

  ...

   [] nms;

  那么,這些語(yǔ)句會(huì)調(diào)用全局new和運(yùn)算符,而不是重載過(guò)的new和。為了重載能為對(duì)象數(shù)組分配內(nèi)存的new和運(yùn)算符,必須像下面的程序一樣,對(duì)new[]和[]也進(jìn)行重載。

  #include iostream.h

  #include string.h

  #include stddef.h

  #include new.h

  const int maxnames = 5;

  class Names

  {

  char name[25];

  static char Names::pool[];

  static bool Names::inuse[maxnames];

  public:

  Names(char* s) { strncpy(name,s,sizeof(name)); }

  void* operator new(size_t) throw(bad_alloc);

  void operator (void*) throw();

  void display() const { cout < };

  char Names::pool[maxnames * sizeof(Names)];

  bool Names::inuse[maxnames];

  void* Names::operator new[](size_t size) throw(bad_alloc)

  {

  int elements=size/sizeof(Names);

  int p=-1;

  int i=0;

  while((i {

  if(!inuse[i]) p=i;

  ++i;

  }

  // Not enough room.

  if ((p==-1) || ((maxnames-p) for(int x=0; x return pool+p*sizeof(Names);

  }

  void Names::operator [](void* b) throw()

  {

  if(b!=0)

  {

  int p=((char*)b- pool)/sizeof(Names);

  int elements=inuse[p];

  for (int i=0; i }

  }

  int main()

  {

  Names* np = new Names[maxnames];

  int i;

  for(i=0; i {

  cout < char name[25];

  cin >> name;

  *(np + i) = name;

  }

  for(i=0; idisplay();

   [] np;

  return 0;

  }

  重載new[]和[]要比重載new和考慮更多的問(wèn)題。這是因?yàn)閚ew[]運(yùn)算符時(shí)為數(shù)組分配內(nèi)存,所以它必須記住數(shù)組的大小,重載的[]運(yùn)算符才能正確地把緩沖區(qū)釋放回內(nèi)存池。

  上面的程序采用的方法比較簡(jiǎn)單,吧原來(lái)存放緩沖區(qū)使用標(biāo)志的布爾型數(shù)組換成一個(gè)整型數(shù)組,該數(shù)組的每個(gè)元素記錄new[]運(yùn)算符分配的緩沖區(qū)個(gè)數(shù),而不再是一個(gè)簡(jiǎn)單的true。

  當(dāng)[]運(yùn)算符函數(shù)需要把緩沖區(qū)釋放回內(nèi)存池時(shí),它就會(huì)用該數(shù)組來(lái)確認(rèn)釋放的緩沖區(qū)個(gè)數(shù)


【計(jì)算機(jī)二級(jí)《C++》考點(diǎn)解析:堆和類(lèi)數(shù)組】相關(guān)文章:

2017計(jì)算機(jī)二級(jí)C++考點(diǎn)類(lèi)的轉(zhuǎn)換05-20

2016最新計(jì)算機(jī)二級(jí)C++考點(diǎn)解析「模板」03-21

計(jì)算機(jī)二級(jí)C++考點(diǎn):C++語(yǔ)言概述07-17

計(jì)算機(jī)二級(jí)C++函數(shù)考點(diǎn)07-22

2017計(jì)算機(jī)二級(jí)C++考點(diǎn)習(xí)題及答案07-07

計(jì)算機(jī)二級(jí)C++考點(diǎn):基本控制結(jié)構(gòu)01-25

計(jì)算機(jī)二級(jí)C++程序題及答案解析12-29

2016年計(jì)算機(jī)二級(jí)C++復(fù)習(xí)資料:類(lèi)對(duì)象數(shù)組和靜態(tài)成員08-04

計(jì)算機(jī)二級(jí)C++模擬題及解析201603-10