成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

那些年我們踩過的坑:C 語言柔性數組其實超簡單!

開發(fā)
今天咱們來聊一個看起來高大上,其實超級實用的 C 語言知識點——柔性數組。

大家好啊,我是小康。

今天咱們來聊一個看起來高大上,其實超級實用的 C 語言知識點——柔性數組。

別被這個名字唬住,啥叫"柔性"?簡單說就是大小可變、長度不固定的數組。學會這招,分分鐘提升你的程序設計水平!

一、啥是柔性數組?先別慌!

你肯定用過普通數組吧?比如:int nums[10]。這種數組大小一旦定了就是10個元素,多一個少一個都不行,死板得很!

而柔性數組是啥呢?它是 C99 標準引入的一個神奇特性,允許我們在結構體的最后聲明一個大小未知的數組。是不是聽著很玄乎?別著急,看完你就懂了!

二、柔性數組長啥樣?

struct FlexArray {
    int length;     // 記錄數組長度
    double scores[]; // 這就是柔性數組!注意這里沒寫大小
};

看到了嗎?這個scores數組后面的中括號是空的!這就是柔性數組的寫法。它必須是結構體的最后一個成員,前面必須至少有一個其他成員(通常用來記錄數組的實際長度)。

三、為啥要用柔性數組?有啥好處?

想象一下這個場景:你要管理不同學生的成績,有的學生選了 3 門課,有的選了 8 門課。用普通數組咋辦?

方法一:定一個夠大的數組,比如:

struct Student {
    int id;
    int courseCount;
    double scores[30]; // 寫死30個,夠大就行
};

// 使用方式
struct Student xiaoming;
xiaoming.id = 1001;
xiaoming.courseCount = 5;
xiaoming.scores[0] = 85.5;
// ...

問題來了,太浪費空間了!小明只選了 5 門課,但我們卻給他預留了 30 門課的空間。而且,萬一有學霸選了超過 30 門課呢?改代碼重新編譯?這也太麻煩了!

方法二:用指針和動態(tài)內存,比如:

struct Student {
    int id;
    int courseCount;
    double *scores; // 指針,指向另一塊內存
};

// 使用方式
struct Student *xiaoming = (struct Student*)malloc(sizeof(struct Student));
xiaoming->id = 1001;
xiaoming->courseCount = 5;

// 再分配一次內存給成績數組
xiaoming->scores = (double*)malloc(5 * sizeof(double));
xiaoming->scores[0] = 85.5;
// ...

// 釋放內存時要記得釋放兩次!
free(xiaoming->scores); // 先釋放數組
free(xiaoming);         // 再釋放結構體

這種方式雖然靈活,但每次都要分兩次申請內存:一次給結構體,一次給 scores 指向的數組。 內存不連續(xù),訪問效率低,而且容易忘記釋放內存(特別是那個 scores 指針指向的內存,很多人只釋放了結構體,忘了釋放數組,造成內存泄漏)。

這時候,柔性數組就顯得特別聰明了!

四、柔性數組是怎么用的?實戰(zhàn)來了!

#include <stdio.h>
#include <stdlib.h>

// 定義一個帶柔性數組的結構體
struct Student {
    int id;          // 學號
    int courseCount; // 課程數量
    double scores[]; // 柔性數組,存儲成績
};

int main() {
    int courses = 5; // 小明選了5門課
    
    // 計算需要的總內存:結構體固定部分 + 柔性數組部分
    struct Student *xiaoming = (struct Student*)malloc(sizeof(struct Student) + courses * sizeof(double));
    
    // 初始化小明的信息
    xiaoming->id = 1001;
    xiaoming->courseCount = courses;
    
    // 設置小明的5門課成績
    xiaoming->scores[0] = 85.5; // 數學
    xiaoming->scores[1] = 92.0; // 英語
    xiaoming->scores[2] = 78.5; // 物理
    xiaoming->scores[3] = 96.0; // 化學
    xiaoming->scores[4] = 88.5; // 生物
    
    // 計算平均分
    double sum = 0;
    for (int i = 0; i < xiaoming->courseCount; i++) {
        sum += xiaoming->scores[i];
    }
    
    printf("學號%d的小明平均分是:%.2f\n", xiaoming->id, sum / xiaoming->courseCount);
    
    // 釋放內存,只需要free一次!
    free(xiaoming);
    
    return0;
}

運行結果:

學號1001的小明平均分是:88.10

五、柔性數組的內存布局,一圖看懂!

假設我們有這樣的結構體:

struct FlexArray {
    int length;
    double scores[];
};

內存中的樣子大概是:

+-------------+-------------+-------------+-------------+
| length (4B) | scores[0]   | scores[1]   | scores[2]   | ...
+-------------+-------------+-------------+-------------+
               ↑
        柔性數組的起始位置

所有數據都在一塊連續(xù)的內存中,訪問超快,而且只需要分配和釋放一次內存!

六、柔性數組的注意事項(踩坑警告??)

必須放在結構體最后:柔性數組必須是結構體的最后一個成員。

至少有一個其他成員:結構體中必須有至少一個其他成員(通常用來記錄柔性數組的長度)。

不占結構體大小:柔性數組不計入結構體的 sizeof 大小。

struct Test {
    int n;
    int arr[];
};

printf("結構體大小:%zu\n", sizeof(struct Test)); // 輸出:結構體大小:4

不能直接定義變量:不能直接定義結構體變量,必須用指針和動態(tài)內存。

// 錯誤寫法
struct Test t;  // 不行!柔性數組沒地方存
               // 注意:雖然在 VS2022 等現代編譯器中可能編譯通過
               // 但這是不規(guī)范的,柔性數組沒有實際存儲空間,使用會導致內存越界!

// 正確寫法
struct Test *pt = (struct Test*)malloc(sizeof(struct Test) + 10 * sizeof(int));

七、柔性數組vs指針成員,差別在哪?

有人可能會問:用結構體里的指針成員不也能實現類似功能嗎?

struct WithPointer {
    int length;
    int *data;  // 指針成員
};

struct WithFlexible {
    int length;
    int data[]; // 柔性數組
};

區(qū)別大了去了:

  • 內存布局:柔性數組的數據緊跟在結構體后面,是一塊連續(xù)內存;指針方式數據在另一個地方,是兩塊不連續(xù)的內存。
  • 內存操作次數:柔性數組只需要分配和釋放一次內存;指針方式需要分配和釋放兩次。
  • 訪問效率:柔性數組訪問更快,內存連續(xù),對 CPU 緩存更友好。
  • 代碼簡潔度:柔性數組代碼更簡潔,不容易出現忘記釋放內存的問題。

八、實戰(zhàn)案例:實現一個簡單的動態(tài)字符串

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

typedefstruct {
    size_t length;  // 字符串長度
    char data[];    // 柔性數組
} MyString;

// 創(chuàng)建字符串
MyString* createString(const char* text) {
    size_t len = strlen(text);
    
    // 分配內存:結構體大小 + 字符串長度 + 1(給'\0'留位置)
    MyString* str = (MyString*)malloc(sizeof(MyString) + len + 1);
    
    str->length = len;
    strcpy(str->data, text);  // 復制字符串內容
    
    return str;
}

// 打印字符串
void printString(const MyString* str) {
    printf("長度: %zu, 內容: %s\n", str->length, str->data);
}

int main() {
    // 創(chuàng)建一個字符串
    MyString* hello = createString("Hello, 柔性數組!");
    
    // 打印字符串信息
    printString(hello);
    
    // 內存釋放,只需要一次free
    free(hello);
    
    return0;
}

運行結果:

長度: 16, 內容: Hello, 柔性數組!

總結:柔性數組到底香在哪?

  • 內存連續(xù):數據緊湊,訪問效率高
  • 一次分配:避免多次 malloc/free,減少內存碎片
  • 一次釋放:不容易造成內存泄漏
  • 靈活方便:可以根據需要分配剛好夠用的內存

是不是感覺 C 語言突然變得更強大了?柔性數組這個小技巧,在很多底層庫和系統(tǒng)編程中都有廣泛應用,比如 Linux 內核中就大量使用了這個技術。

好了,今天的 C 語言小課堂到此結束!下次我們再聊其他有趣的編程技巧。

掌握了柔性數組這個小技巧,是不是感覺自己的 C 語言技能又升級了?

寫在最后:技術成長沒有"固定數組"

就像柔性數組一樣,我們的學習之路也不該限定死板的大小。從 C 語言基礎到高級技巧,從編程小白到技術大牛,每個階段都需要不同"長度"的知識儲備。

責任編輯:趙寧寧 來源: 跟著小康學編程
相關推薦

2024-04-01 08:05:27

Go開發(fā)Java

2025-05-26 02:22:00

2023-12-21 11:46:48

C語言柔性數組開發(fā)

2017-07-17 15:46:20

Oracle并行機制

2020-11-18 10:16:23

人工智能機器學習技術

2025-04-03 12:30:00

C 語言隱式類型轉換代碼

2018-01-10 13:40:03

數據庫MySQL表設計

2019-02-19 09:46:58

美圖容器化Kubernetes

2019-04-18 11:43:40

軟件卸載PC端

2020-05-18 08:58:33

Python開發(fā)工具

2024-05-06 00:00:00

緩存高并發(fā)數據

2015-03-24 16:29:55

默認線程池java

2020-03-18 14:28:43

偽概念手機真相

2021-12-28 08:17:41

循環(huán) forgo

2025-04-29 10:17:42

2023-03-13 13:36:00

Go擴容切片

2018-09-11 09:14:52

面試公司缺點

2022-07-06 11:47:27

JAVAfor循環(huán)

2023-12-14 17:34:22

Kubernetes集群K8s

2021-10-28 19:10:02

Go語言編碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 欧美一级免费看 | 国产色| 日韩精品一区二区三区在线观看 | 亚洲欧美视频 | 国产成人免费视频网站高清观看视频 | 精品免费 | 97国产精品 | 国产成人在线播放 | 欧美一区二区三区视频 | 99久久电影| 91免费电影 | 黄色免费观看 | 午夜视频在线观看网站 | 自拍偷拍小视频 | aa级毛片毛片免费观看久 | 亚洲美女在线一区 | 蜜桃精品视频在线 | 毛片一区 | 日韩成人在线电影 | 免费性视频 | 欧美性受 | 国产欧美精品区一区二区三区 | 九色91视频 | 国产第一亚洲 | 亚州中文字幕 | 中文字幕人成乱码在线观看 | 国产精品福利在线 | 久久亚洲国产精品日日av夜夜 | 欧美激情亚洲 | 丝袜天堂 | 日韩视频91 | 成人激情视频网 | 国产高清精品一区二区三区 | 成人免费影院 | 日韩中文字幕 | 在线视频99| 精品欧美一区二区精品久久久 | 欧美一级在线观看 | 午夜影院在线观看 | 一片毛片 | 一本在线|