class 和 struct的區別

C++中的struct對C中的struct進行了擴充,它已經不再只是一個包含不同數據類型的數據結構了,它已經獲取了太多的功能。
struct能包含成員函數嗎? 能!
struct能繼承嗎? 能!!
struct能實現多態嗎? 能!!!
?

既然這些它都能實現,那它和class還能有什么區別?

最本質的一個區別就是默認的訪問控制:?

默認的繼承訪問權限

struct是public的class是private的
你可以寫如下的代碼:
struct A
{
? char a;
};
struct B : A
{
? char b;
};

這個時候B是public繼承A的。

如果都將上面的struct改成class,那么B是private繼承A的。這就是默認的繼承訪問權限。?

所以我們在平時寫類繼承的時候,通常會這樣寫:
class?B : public A

就是為了指明是public繼承,而不是用默認的private繼承。

?

當然,到底默認是public繼承還是private繼承取決于子類而不是基類

我的意思是,struct可以繼承class,同樣class也可以繼承struct,那么默認的繼承訪問權限是看子類到底是用的struct還是class。如下:

?

struct A{};class B : A{}; //private繼承
struct C : B{}; //public繼承

?

struct作為數據結構的實現體,它默認的數據訪問控制是public的,而class作為對象的實現體,它默認的成員變量訪問控制是private的

?

我依舊強調struct是一種數據結構的實現體,雖然它是可以像class一樣的用。我依舊將struct里的變量叫數據,class內的變量叫成員,雖然它們并無區別。


到底是用struct還是class,完全看個人的喜好,你可以將程序里所有的class全部替換成struct,它依舊可以很正常的運行。但我給出的最好建議,還是:當你覺得你要做的更像是一種數據結構的話,那么用struct,如果你要做的更像是一種對象的話,那么用class。?

當然,我在這里還要強調一點的就是,對于訪問控制,應該在程序里明確的指出,而不是依靠默認,這是一個良好的習慣,也讓你的代碼更具可讀性。?

說到這里,很多了解的人或許都認為這個話題可以結束了,因為他們知道struct和class的“唯一”區別就是訪問控制。很多文獻上也確實只提到這一個區別。?

但我上面卻沒有用“唯一”,而是說的“最本質”,那是因為,它們確實還有另一個區別,雖然那個區別我們平時可能很少涉及。

那就是:“class”這個關鍵字還用于定義模板參數,就像“typename”。但關鍵字“struct”不用于定義模板參數。這一點在Stanley B.Lippman寫的Inside the C++ Object Model有過說明。?

問題討論到這里,基本上應該可以結束了。但有人曾說過,他還發現過其他的“區別”,那么,讓我們來看看,這到底是不是又一個區別。還是上面所說的,C++中的struct是對C中的struct的擴充,既然是擴充,那么它就要兼容過去C中struct應有的所有特性。例如你可以這樣寫:?

struct A //定義一個struct
{
?? char c1;
?? int n2;
?? double db3;
};
A a={'p', 7, 3.1415926}; //定義時直接賦值
?

也就是說struct可以在定義的時候用{}賦初值。那么問題來了,class行不行呢?將上面的struct改成class,試試看。報錯!噢~于是那人跳出來說,他又找到了一個區別。我們仔細看看,這真的又是一個區別嗎??

你試著向上面的struct中加入一個構造函數(或虛函數),你會發現什么?
對,struct也不能用{}賦初值了
的確,以{}的方式來賦初值,只是用一個初始化列表來對數據進行按順序的初始化,如上面如果寫成A a={'p',7};則c1,n2被初始化,而db3沒有。這樣簡單的copy操作,只能發生在簡單的數據結構上,而不應該放在對象上。加入一個構造函數或是一個虛函數會使struct更體現出一種對象的特性,而使此{}操作不再有效。?

事實上,是因為加入這樣的函數,使得類的內部結構發生了變化。而加入一個普通的成員函數呢?你會發現{}依舊可用。其實你可以將普通的函數理解成對數據結構的一種算法,這并不打破它數據結構的特性。?

那么,看到這里,我們發現即使是struct想用{}來賦初值,它也必須滿足很多的約束條件,這些條件實際上就是讓struct更體現出一種數據機構而不是類的特性。?

那為什么我們在上面僅僅將struct改成class,{}就不能用了呢?

其實問題恰巧是我們之前所講的——訪問控制!你看看,我們忘記了什么?對,將struct改成class的時候,訪問控制由public變為private了,那當然就不能用{}來賦初值了。加上一個public,你會發現,class也是能用{}的,和struct毫無區別!!!?

做個總結,從上面的區別,我們可以看出,struct更適合看成是一個數據結構的實現體,class更適合看成是一個對象的實現體。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/444936.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/444936.shtml
英文地址,請注明出處:http://en.pswp.cn/news/444936.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

leetcode206 反轉鏈表

反轉一個單鏈表。 示例: 輸入: 1->2->3->4->5->NULL 輸出: 5->4->3->2->1->NULL 進階: 你可以迭代或遞歸地反轉鏈表。你能否用兩種方法解決這道題? 經典題不解釋 /*** Definition for singly-linked list.* public class ListNode…

淺議柔性數組

很多時候,柔性數組應用在了變長結構體中,如: StructPacket {Int state; Int len;

leetcode 152 乘積最大子序列

給定一個整數數組 nums ,找出一個序列中乘積最大的連續子序列(該序列至少包含一個數)。 示例 1: 輸入: [2,3,-2,4] 輸出: 6 解釋: 子數組 [2,3] 有最大乘積 6。 示例 2: 輸入: [-2,0,-1] 輸出: 0 解釋: 結果不能為 2, 因為 [-2,-1] 不是子…

PaperNotes(5)-Conditional Generative Adversarial Nets

Conditional GAN 論文閱讀筆記Abstract1 Introduction2 Related Work3 Conditional Adversarial Nets3.1 Generative Adversarial Nets3.2 Conditional Adversarial Nets4 Experimental Results4.1 Unimodal4.2 Multimodal5 Future Work6.思考文章地址:https://arxi…

蛙泳姿勢教學

偶爾看到分享的一篇日志,記錄下,忙過這段時間努力學蛙泳。 蛙泳配合有一個順口溜,在講解蛙泳動作要領之前先介紹給大家:“劃手腿不動,收手再收腿,先伸胳膊后蹬腿,并攏伸直漂一會兒。”從順口溜中…

leetcode238 除本身以外數組的乘積

給定長度為 n 的整數數組 nums,其中 n > 1,返回輸出數組 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘積。 示例: 輸入: [1,2,3,4] 輸出: [24,12,8,6] 說明: 請不要使用除法,且在 O(n) 時間復雜度內完…

C 和c++的一些雜想,想到哪兒寫到哪兒

關于C和c++一直有好多的程序猿在研究,研究區別研究相似的地方,究竟用那個預言好,沒有確定的說法,要看你做什么了。 初始化操作: 在初始化的時候,我們都知道C語言一般都是這樣處理的: int a=12; C++ 呢,除了這樣復制初始化之外還可以直接初始化: int a(12); 啊…

C++(2)--mac使用VScode 進行C++編譯、運行、調試

mac 使用VScode 進行C開發1.編譯的基礎概念2. mac 編譯c代碼2.1 查看編譯器情況2.2 安裝插件C/C,C/C Clang Command Adapte2.3新建一個C project2.3.1本地新建文件夾2.3.2新建mian.cpp文件2.3.3 編寫hello word demo2.4 代碼編譯,運行,調試2.…

boost庫linux編譯安裝

0.下載 1.解壓boost_1_49_0.tar.g然后放到/opt/ 2. 進入解壓后的文件夾 cd /opt/boost_1_49_0 3.將boost安裝配置在/boost/prefix目錄下 不過之前先 mkdir -p /boost/prefix

leetcode136 只出現一次的數字

給定一個非空整數數組,除了某個元素只出現一次以外,其余每個元素均出現兩次。找出那個只出現了一次的元素。 說明: 你的算法應該具有線性時間復雜度。 你可以不使用額外空間來實現嗎? 示例 1: 輸入: [2,2,1] 輸出: 1 示例 2: …

C++(3)--編譯、gdb調試

3--編譯和執行過程1.編譯2.gdb調試gdb 查coreGCC是一個編譯套件,是一個以"gcc"命令為首的源碼施工隊。施工隊的成員有gcc、cpp、as、ld四個成員 預處理–宏定義展開,頭文件引入-- cpp 等價于 gcc -E編譯–C語言->匯編語言–gcc -S匯編–匯…

leetcode94 二叉樹的中序遍歷

給定一個二叉樹,返回它的中序 遍歷。 示例: 輸入: [1,null,2,3] 1 \ 2 / 3 輸出: [1,3,2] 進階: 遞歸算法很簡單,你可以通過迭代算法完成嗎? 遞歸 /*** Definition for a binary tree node.* public class TreeNode …

使用動態鏈接庫

1. 動態鏈接庫是程序運行時加載的庫,當動態鏈接庫正確安裝后,所有的程序都可以使用動態庫來運行程序。動態鏈接庫是目標文件的集合,目標文件在動態鏈接庫中的組織方式是按照特殊方式形成的。庫中函數和變量的地址是相對地址,不是絕對地址,其真實地址在調用動態庫的程序加載…

算法(29)--兩棵樹匹配

樹匹配1.劍指 Offer 26. 樹的子結構2.劍指 Offer 27. 二叉樹的鏡像3.劍指 Offer 28. 對稱的二叉樹1.劍指 Offer 26. 樹的子結構 判斷:小樹B是否是大樹A的一部分,需要以大樹A的每個為根節點進行匹配判斷。 算法:判斷兩個節點是否相等&#xf…

leetcode647 回文子串

給定一個字符串,你的任務是計算這個字符串中有多少個回文子串。 具有不同開始位置或結束位置的子串,即使是由相同的字符組成,也會被計為是不同的子串。 示例 1: 輸入: "abc" 輸出: 3 解釋: 三個回文子串: "a", "…

windows下關于Objective-C開發環境的配置

最近IOS一直很火,也想學習以以下OC開發,一般裝個虛擬機,然后裝個mac,我相信大多數人的機子跑不起來或者很卡,所以之前借鑒了一個文章,就是關于在windows下配置OC開發環境,這里我把自己的安裝說一…

如何選擇商鋪投資

如何選擇商鋪? 選擇商鋪時,需要考慮哪些因素呢? 聶先生表示,要看地段,包括周圍的商業環境、停車方便程度、汽車流量、未來發展前景、視覺效果等因素。此外,地域經濟也很重要,必須要有一個好的投…

PaperNotes(6)-GAN/DCGAN/WGAN/WGAN-GP/WGAN-SN-網絡結構/實驗效果

GAN模型網絡結構實驗效果演化1.GAN1.1網絡結構1.2實驗結果2.DCGAN2.1網絡結構2.2實驗結果3.WGAN3.1網絡結構3.2實驗結果4.WGAN-GP4.1網絡結構4.2實驗結果5.WGAN-SN5.1網絡結構5.2實驗結果小結1.GAN 文章: https://arxiv.org/pdf/1406.2661.pdf 代碼: Py…

Spring Security使用

Spring Security 在web應用開發中,安全無疑是十分重要的,選擇Spring Security來保護web應用是一個非常好的選擇。 Spring Security 是spring項目之中的一個安全模塊,可以非常方便與spring項目無縫集成。特別是在spring boot項目中加入sprin…

nginx python webpy 配置安裝

安裝webpy$ wget http://webpy.org/static/web.py-0.34.tar.gz$ tar xvzf web.py-0.34.tar.gz$ cd web.py-0.34$ sudo python setup.py install安裝 Fluphttp://www.saddi.com/software/flup/dist/flup-1.0.2.tar.gz$ wget http://www.saddi.com/software/flup/dist/flup-1.0.2…