鍍金池/ 問答/Python  C  C++/ C++ 如何獲取類的屬性和方法?

C++ 如何獲取類的屬性和方法?

在.h文件中一個(gè)類的定義,該類擁有多個(gè)方法和屬性:

class Test1
{
    public:
        Test1(int n);
        void set(int, int, int);  
        int isLeapYear();
        void print();
    
    private:
        int num;
        int month;
        int day;
        int year;
        double length;
};

就是我在main函數(shù)中獲取到了類Test1實(shí)例t1了,如何才能知道這個(gè)實(shí)例 中有哪些方法和屬性?這些方法和屬性有doc嗎?怎么查看?

**是不是任何的c代碼編譯成lib和dll后都會(huì)有對(duì)應(yīng)的.h頭文件,這個(gè)頭文件中就包含了類的方法和屬性信息?
會(huì)不會(huì)存在沒有頭文件,但是我們可以正常使用lib和dll所提供的類?對(duì)于這樣的情況,那么如何知道該類要如何調(diào)用,并能否查看方法的doc?
**

因?yàn)槭菑腜ython過來學(xué)C++的,以前學(xué)過c,現(xiàn)在回來學(xué)發(fā)現(xiàn)C++相對(duì)python是有點(diǎn)不方便,或者說我不會(huì)用,因?yàn)樵趐ython中查看類的屬性方法和doc非常方便,一個(gè)dir函數(shù)就出來了:

#導(dǎo)入pandas這個(gè)包
import pandas as pd
#pd.DataFrame是一個(gè)類,使用dir查看這個(gè)類中有的方法和屬性
dir(pd.DataFrame)
Out[3]:
['T',
 '_AXIS_ALIASES',
 '_AXIS_IALIASES',
 '_AXIS_LEN',
 '_AXIS_NAMES',
 '_AXIS_NUMBERS',
 '_AXIS_ORDERS',
 '_AXIS_REVERSED',
 '_AXIS_SLICEMAP',
 '__abs__',
 '__add__',
 '__and__',
 '__array__',
 '__array_wrap__',
 '__bool__',
 '__bytes__',
 ...]

#在ipython中查看doc,就是加個(gè)問號(hào)就知道怎么用了
pd.DataFrame?
Init signature: pd.DataFrame(data=None, index=None, columns=None, dtype=None, copy=False)
Docstring:
Two-dimensional size-mutable, potentially heterogeneous tabular data
structure with labeled axes (rows and columns). Arithmetic operations
align on both row and column labels. Can be thought of as a dict-like
container for Series objects. The primary pandas data structure

Parameters
----------
data : numpy ndarray (structured or homogeneous), dict, or DataFrame
    Dict can contain Series, arrays, constants, or list-like objects
index : Index or array-like
    Index to use for resulting frame. Will default to np.arange(n) if
    no indexing information part of input data and no index provided
columns : Index or array-like
    Column labels to use for resulting frame. Will default to
    np.arange(n) if no column labels are provided
dtype : dtype, default None
    Data type to force. Only a single dtype is allowed. If None, infer
copy : boolean, default False
    Copy data from inputs. Only affects DataFrame / 2d ndarray input

Examples
--------
Constructing DataFrame from a dictionary.

>>> d = {'col1': [1, 2], 'col2': [3, 4]}
>>> df = pd.DataFrame(data=d)
>>> df
   col1  col2
0     1     3
1     2     4

Notice that the inferred dtype is int64.

>>> df.dtypes
col1    int64
col2    int64
dtype: object

To enforce a single dtype:

>>> df = pd.DataFrame(data=d, dtype=np.int8)
>>> df.dtypes
col1    int8
---Return to continue, q to quit---

#使用type查看類型
In [5]: type(pd.DataFrame)
Out[5]: type
In [6]: type(pd)
Out[6]: module

那么c++中有沒有類似的方法,或者說要是有這些需求要如何處理?

還是我理解錯(cuò)了,只要能用都有.h文件的?

謝謝

回答
編輯回答
尐懶貓

C或C++程序編譯后就會(huì)失去程序中的很多信息,包括類名、方法名、字段名等,所以想在運(yùn)行期間獲得這些內(nèi)容是很難的。這一點(diǎn)與Python等支持反射的語(yǔ)言有很大不同。

至于導(dǎo)出調(diào)試信息,只是說在可執(zhí)行程序中加了一段數(shù)據(jù),用來標(biāo)識(shí)可執(zhí)行文件與源代碼的一些映射,從而讓調(diào)試變得方便。但是調(diào)試信息是給調(diào)試器用的,程序自身并不能夠直接獲取這些信息,除非你自己解析二進(jìn)制格式的可執(zhí)行文件。

所以回到你的問題上來,雖然很多語(yǔ)言都提供了反射機(jī)制,但C/C++這種相對(duì)比較“底層”的語(yǔ)言并沒有提供。

2017年3月12日 08:17
編輯回答
萢萢糖

先對(duì)比一下 python 與 c++ 代碼的運(yùn)行方式。

python
python 解釋器直接執(zhí)行 python 源碼,每一個(gè)被創(chuàng)建的 python 對(duì)象都包含有該類的屬性名、方法名等信息,所以 dir 方法可以查到。

c++
c++ 編譯器把 c++ 源碼編譯成機(jī)器碼,它不保留類的屬性名、方法名等信息,除非你要求它附帶調(diào)試符號(hào)。

不過 dll 比較特殊,它可以附帶一些導(dǎo)出(export)方法列表,里面包含有方法名稱。
這可以通過 PE 工具看到,如 Dependency Walker, http://www.dependencywalker.com/

關(guān)于調(diào)試符號(hào)請(qǐng)查閱編譯器的相關(guān)文檔,如

  1. 微軟 vc++pdb 文件, https://github.com/Microsoft/...
  2. GNU g++-g 參數(shù), https://gcc.gnu.org/onlinedoc...

需要深入研究 c++ 類如何轉(zhuǎn)換成機(jī)器碼的同學(xué),可網(wǎng)上搜索 c++ virtual function


再來說一下 .h 頭文件與 lib/dll 庫(kù)文件的關(guān)系。

一般而言,頭文件是庫(kù)開發(fā)者給庫(kù)調(diào)用者使用的一種聲明文件,調(diào)用者的 c++ 編譯器根據(jù)頭文件生成必要的機(jī)器碼以訪問類庫(kù)的屬性或方法。
如果頭文件丟失了,使用逆向工程(reverse engineering)可以造一個(gè)頭文件出來。

2017年9月2日 14:46