作者amamoimi (佛仔)
看板C_and_CPP
標題[問題]char 指標問題
時間Sun Oct 15 14:49:24 2023
不好意思我程式新手又來擾民了@@
https://onlinegdb.com/Vm941gQ0_
這是我在書上看到的程式碼
功能是把變數byte by byte的交換
但是我看不太懂那個swap函數...
為什麼可以隨便把參數冠上一個(char* )啊?
譬如a跟b明明就是int
把&a跟&b 前面加一個(char *)是什麼意思啊?又為什麼特別指定char呢?
chatgpt 的解釋是說(char*)x是在告訴compiler要把x跟y當作a sequence of bytes.
為啥!?
在本版獲益良多,希望各位這次也能不吝指教
謝謝大家!
----
Sent from
BePTT on my OPPO CPH1943
--
※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 1.200.241.52 (臺灣)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/C_and_CPP/M.1697352566.A.A8C.html
1F:推 ssdoz2sk: 因為以基礎的資料型態來說,只有 char, int, float, dou10/15 15:01
2F:→ ssdoz2sk: ble 與 signed, unsigned, 的各式組合,而其中 char 為10/15 15:01
3F:→ ssdoz2sk: 1 byte。(char *) 是把整個資料當作一個長度為 n 的 by10/15 15:01
4F:→ ssdoz2sk: te array 來看。10/15 15:01
5F:推 Lhmstu: char = 1 byte 一般來說是 8 bits,這邊轉成char*一次對110/15 15:06
6F:→ Lhmstu: byte 進行互換(原本丟進函數後為已轉為 void*),也就是10/15 15:06
7F:→ Lhmstu: 說“接下來我一次要操作一個byte”這樣的意思,所以後面+10/15 15:06
8F:→ Lhmstu: +操作都是把pointer移動一個byte,移動幾次則是根據你丟10/15 15:06
9F:→ Lhmstu: 進去的size決定10/15 15:06
10F:推 ssdoz2sk: 補充一下,我習慣會重新定義 typedef unsigned char10/15 15:08
11F:→ ssdoz2sk: UINT8; 。會比較直觀易懂,資料就是一個個 byte ,10/15 15:08
12F:→ ssdoz2sk: 不會跟字串搞混。10/15 15:08
13F:→ amamoimi: 謝謝z大的詳細解釋!我懂為什麼要用char了,但我還是不10/15 15:14
14F:→ amamoimi: 太懂為什麼(char*)a是合法的寫法...a明明就是一個整數10/15 15:14
15F:→ amamoimi: 變數,為什麼可以創造一個char指標指向一個存著整數的10/15 15:14
16F:→ amamoimi: 記憶體位置啊@@10/15 15:14
17F:推 Lhmstu: int 是4 bytes,你可以想像一個byte是一個箱子。而int*是10/15 15:19
18F:→ Lhmstu: 指向排好的箱子的頭,可以一次移動改寫操作這四個箱子,10/15 15:19
19F:→ Lhmstu: 如果你只想一次改動一個箱子( 1 byte),就可以轉成char*10/15 15:19
20F:→ Lhmstu: ,一次只對一個byte做處理,這就是強制性別轉換。也是系10/15 15:19
21F:→ Lhmstu: 統大了之後容易有bug的地方(x10/15 15:19
22F:→ saxontai: 有stdint.h了,可以不用再自己typedef10/15 15:38
23F:推 wulouise: use case不合理,什麼情況要swap不同type的資料?你要不10/15 15:39
24F:→ wulouise: 要貼一下書名給大家看,有可能你的書太舊或寫得不好10/15 15:39
25F:→ wulouise: 最早c只有char代表8bits type,後來才有uint8_t10/15 15:42
26F:→ amamoimi: 書名是「第一次學C++就上手」10/15 15:56
27F:→ amamoimi: 我好像稍微懂了...10/15 15:56
28F:→ amamoimi: 所以其實可以assign一個地址(不管那個地址存放什麼變10/15 15:56
29F:→ amamoimi: 數)給任何指標(指向任何型態)囉?(混亂...其實不管10/15 15:56
30F:→ amamoimi: 什麼資料型態都是0101嘛)10/15 15:56
※ 編輯: amamoimi (1.200.241.52 臺灣), 10/15/2023 16:38:26
※ 編輯: amamoimi (1.200.241.52 臺灣), 10/15/2023 16:39:09
31F:→ amamoimi: 想再問一下 void*型態的參數是只要是地址都可以接收嗎10/15 16:40
32F:→ amamoimi: ?10/15 16:40
33F:→ Lhmstu: 是的,void *是萬用。如果要處理任何資料,都需要先把voi10/15 16:53
34F:→ Lhmstu: d*轉成對應想要處理的型別的指標10/15 16:53
35F:推 Lhmstu: 你可以嘗試 unsigned int x = 258 然後強制轉型成(unsig10/15 17:04
36F:→ Lhmstu: ned char) 然後看結果,因為一個byte最多存256個值(258會10/15 17:04
37F:→ Lhmstu: 進位用到下一個byte)10/15 17:04
38F:→ amamoimi: 了解了 謝謝大大!10/15 17:12
39F:推 johnjohnlin: 建議還是使用std::swap10/15 19:47
40F:推 wulouise: byte by byte copy不一定比較快,沒必要自己寫swap10/15 20:44
41F:推 Dracarys: 因為char可以alias任何type10/16 01:42
44F:→ leolarrel: 你可能要先回去複習計概,重新理解什麼是記憶體10/16 12:55
45F:→ leolarrel: 不然版友回答得再多妳還是有聽卻自以為懂10/16 12:56
46F:→ amamoimi: 謝謝 不過我我們同時修程式跟計概,所以我現在還在學習10/16 14:10
47F:→ amamoimi: 計概中@@10/16 14:10
※ 編輯: amamoimi (140.138.21.80 臺灣), 10/16/2023 14:20:30
48F:推 descent: 推薦 C语言程序设计现代方法第2版, 很好的入門書10/16 14:45
謝謝大大推薦!
49F:→ amamoimi: 另外想問一下10/16 14:48
51F:→ amamoimi: 這時剛剛用來測試的程式10/16 14:48
52F:→ amamoimi: 我不小心把cout<<*b寫成10/16 14:48
53F:→ amamoimi: cout<<b 結果為什麼output是一樣的勒xd10/16 14:48
54F:推 yvb: 計概教到 little endian 了嗎?10/16 15:27
55F:推 Dracarys: 因為你呼叫到吃const char*的operator<<了,就像你寫st10/16 15:29
56F:→ Dracarys: d::cout << “M”;10/16 15:29
→ amamoimi: 謝謝 稍微測試一下明白了~(這邊是書讀的不夠熟不好意思xd
10/16 18:02
※ 編輯: amamoimi (1.200.241.52 臺灣), 10/16/2023 18:29:31
57F:→ leolarrel: 哪一家爛補習班不先教計概就教C語言阿? c/cpp語言又不 10/17 12:38
58F:→ leolarrel: 像python 是高階語言,中低階語言很吃系統底層知識 10/17 12:39
59F:→ leolarrel: 還有你上次問的遞迴,因為你沒有根本上理解遞迴的核心義 10/17 12:47
60F:→ leolarrel: 意,所以才會問出那樣的問題. 10/17 12:48
61F:推 yvb: 可能是學校排課程的問題吧. 10/17 15:41
62F:推 closer76: 我二十多年前在大學就是在計概課學C的啊!XDD 10/17 17:12
63F:推 Richun: 如果從140那個IP來看,大一課程同時排計概跟程式設計常見 10/17 17:12
64F:→ Richun: 不過先教C++這個做法真的頗神奇,怎麼不是先教C或python? 10/17 17:13
65F:→ closer76: 嚴格來說當年我們系上根本沒有單獨的C語言課,教授只有 10/17 17:13
66F:→ closer76: 安排助教在晚上開C的補習課,真的需要的人就問助教。 10/17 17:14
67F:推 Richun: 在學校的話TA時間就過去問爆,C++複雜度很高,陷阱很多。 10/17 17:16
68F:推 closer76: 同意,C++真的複雜太多。學得越多越怕踩到陷阱 XD 10/17 17:19
69F:推 closer76: @amamoimi: 你要不要試著把 a=77 改成 a=21325 試試? 10/17 17:26
70F:→ closer76: 這樣 cout<<b 和 cout<<*b 結果就會不一樣喔! 10/17 17:26
71F:→ closer76: 然後你可以自己思考一下為什麼會這樣。 10/17 17:26
72F:→ amamoimi: 有喔 我昨天花了一些時間測試了各種東西~感覺又複習了 10/17 17:35
73F:→ amamoimi: 不少東西 10/17 17:35
74F:→ amamoimi: 感謝各位大前輩願意建議與指點本菜鳥!! 10/17 17:35
75F:推 DerLuna: 這真的好難懂 10/18 21:03
76F:推 CoNsTaR: 有 std::byte 可以用 10/19 03:45