程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> C++獲取private的變量-偷走private

C++獲取private的變量-偷走private

編輯:C++入門知識

private提供了對數據的封裝,使得private成員只能被類自身的成員函數以及類的友元訪問,其他的函數或者類想要訪問private成員只能通過該類所提供的set和get的方法進行訪問,

或者返回其指針或引用(effective C++中提到過要避免返回對象內部構件的引用,指針,或迭代器。這樣會提高封裝性,幫助 const 成員函數產生 const 效果,並將懸空句柄產生的可能性降到最低,所以但這個方法並不是特別的好)

 

但如果你想獲得一個類的private成員,但是該類的已經在項目被大量的使用,或者是因為其他的原因,你沒有辦法添加get和set方法時,又應該如何獲得該類的private成員呢?

我總結出了以下幾種方法

 

方法一:重定義

#define  private public

示例

A.h定義

 once
), j(2< Type>
    <<  <<

main.cpp

#include <iostream>
 private public
 << a.i <<
    system( 

該方法的優點的是簡單但也有不少的缺點

1.如果在類的定義時不指定訪問標號關鍵字(public,protected,private),使用默認的private訪問限制,那麼該方法就無法達到目的了,比如這裡的j就無法獲得

2.降低代碼的可讀性,改變的一個關鍵字的意義,會沒有注意到這一點的程序員照成困擾

3.將所有使用了private訪問的標號的成員的訪問等級都變成了public,降低了數據的封裝性

 

 

方法二:模擬內存法

A.h定義

#include <iostream>
  *p = &<<  << *(*)p <<<<  << *((*)p+)<< endl;
    system( 

C++標准中要求,在同一個訪問區域中,成員的排列只需符合較晚出現的成員在類的成員中有較高的地址即可,成員之間可能會因為數據對齊所需,添加一些字節

目前各編譯器都是吧一個以上的訪問區域連在一起,安裝聲明的順序成為一個連續的區域

所以類A的一個對象的內存布局類似於這樣:

1、數據對齊

A.h定義

 once
), j(< Type>
    <<  <<

char j占用了一個byte,而i為了數據對齊,在內存布局上並不是與j緊挨著的,而是隔了3個byte,

所以獲得i和j的間隔與上一個一樣,只是j的類型變了

#include <iostream>
  *p = &<<  << *((*)p) <<<<  << *((*)p+)<< 

 

2.加入虛函數

A.h定義

 once
), j( < Type>
    <<  <<

編譯器為了支持虛函數,會在類的每一個對象中,產生一個額外的虛函數指針指向相應的虛函數表,不同的編譯器對這個指針處理不同,有點將它放在了類對象的尾端,有的將它放在了類對象的開始處

vs2013將它放在了類的開頭處

所以類A的一個對象的內存布局應該類似於這樣:

所以如果虛函數過多,又加入了虛繼承, 類裡面又有大量程序員自己定義的類型,那麼該方法就會很麻煩了。

 

 

方法三:李代桃僵

A.h的定義

 once
), j( < Type>
    <<  <<

模擬法應該是模擬內存布局的另一個實現方式

我們看到現在A裡有一個虛函數,一個j和一個i

如果直接使用模擬內存法的話會很麻煩

 

所以我們可以另聲明一個對象B,它的內存布局和A的一樣,只是i和j的訪問限制變成了public

這樣我們可以把一個指向A的對象的指針當做一個指向B的對象指針來使用

#include <iostream>
 ), j( 
*b = (B*)&<< << b->j <<<< << b->i << 

非虛成員函數show放在函數段中,並不在類對象的布局中占用空間,所以有沒有show函數都可以

因為B的對象的內存布局與A一樣,只是訪問限制不同,所以可以利用對B對象的規則去訪問A的對象

一個指向B對象的指針實際指向了一個A對象,對B中j和i的訪問實際上是對A對象中i和j的訪問

 

該方法模擬內存法容易了很多,但你需要額外聲明一個B對象的定義,而且必須要確保B對象的內存布局要與A對象的一致

 

 

方法四 特化函數模板法

a.h的定義

 once
), j( < Type>
    <<  <<

 

這裡我們發現A有個函數模板show,所以我們可以利用對函數模板show進行特化的方式合法的獲得i和j的public訪問權限

#include <iostream>
 <>
<< <<->j <<<<  << ->i << 

該方法合理,簡單,但也有缺點就是相應的類必須要有成員模板,並且該模板的訪問限制為public才可以

 

 

總結

 

方法 優點 缺點 可移植性 重定義 簡單

1.如果在類的定義時不指定訪問標號關鍵字(public,protected,private),使用默認的private訪問限制,那麼該方法就無法達到目的了,比如這裡的j就無法獲得

2.降低代碼的可讀性,改變的一個關鍵字的意義,會沒有注意到這一點的程序員照成困擾

3.將所有使用了private訪問的標號的成員的訪問等級都變成了public,降低了數據的封裝性

中 模擬內存法 無 虛函數過多,又加入了虛繼承, 類裡面又有大量程序員自己定義的類型時,那麼該方法就會很麻煩了。需要程序員對內存布局有較深的認識 低 李代桃僵 簡單,可能在有些人看來比較清楚 需要額外聲明一個B對象的定義,而且必須要確保B對象的內存布局要與想要訪問的A對象的一致 中 特化函數模板法 合理,簡單 相應的類必須要有成員模板,並且該模板的訪問限制為public才可以 高

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved