一篇博客学会系列(3) —— 对动态内存管理的深度讲解以及经典笔试题的深度解析
目录
动态内存管理
1、为什么存在动态内存管理
2、动态内存函数的介绍
2.1、malloc和free
2.2、calloc
2.3、realloc
3、常见的动态内存错误
3.1、对NULL指针的解引用操作
3.2、对动态开辟空间的越界访问
3.3、对非动态开辟内存使用free释放
3.4、使用free释放一块动态开辟内存的一部分
3.5、对同一块动态内存多次释放
3.6、动态开辟内存忘记释放(内存泄漏)
4、动态内存的经典笔试题
5、C/C++程序的内存开辟
6、柔性数组
6.1、柔性数组的特点
6.2、柔性数组的使用
6.3、柔性数组的优势

动态内存管理
动态内存管理就是对堆区进行管理和操作,本篇博客将带领大家学习在堆区中创建空间、使用空间以及释放空间。

1、为什么存在动态内存管理
常见的内存开辟方式:
int val = 20;//在栈空间上开辟四个字节
char arr[10] = {0};//在栈空间上开辟10个字节的连续空间
但是上述的开辟空间的方式有两个特点:
- 空间开辟大小是固定的。
- 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。
但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。
正因为如此,C语言给了程序员一种权利:能够动态申请和管理内存空间。这时候就需要使用到动
态内存开辟了,而这就是为什么会有动态内存分配的原因,下面我们来一起学习动态内存管理。
2、动态内存函数的介绍
2.1、malloc和free

这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
- 包含头文件<stdlib.h>
- malloc函数如果开辟成功,则返回一个指向开辟好空间的指针。
- malloc函数如果开辟失败,例如开辟空间过大时,则返回一个NULL指针,因此malloc的返回值一定要做检查,否则会出现对空指针的解引用操作。
- 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
- 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器的处理方式。
【举例说明】
int main()
{//申请一块空间,用来存放10个整型int* p = (int*)malloc(sizeof(int) * 10);if (p == NULL){perror("malloc");return 1; //如果为空则不执行下面的代码直接跳出}return 0;
}
动态内存函数申请的空间 ,都需要手动释放。C语言提供了另外一个函数free,专门是用来做动态内存的释放和回收的。

- free函数用来释放动态开辟的内存
- 包含头文件<stdlib.h>
- 如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
- 如果参数 ptr 是NULL指针,则函数什么事都不做。
【举例说明】
int main()
{//申请一块空间,用来存放10个整型int* p = (int*)malloc(sizeof(int) * 10);if (p == NULL){perror("malloc");return 1; //如果为空则不执行下面的代码直接跳出}//使用int i = 0;for ( i = 0; i < 10; i++){*(p + i) = i;}for ( i = 0; i < 10; i++){printf("%d ", *(p + i));}//释放free(p);p == NULL; //虽然已经free释放了,但是p指针依然指向那个空间,此时p就是野指针了//为了防止再使用p访问该空间,将它置成NULL最为合适。return 0;
}
【运行结果】

2.2、calloc
C语言还提供了一个函数叫 calloc , calloc 函数也用来动态内存分配。

- 包含头文件<stdlib.h>
- 函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。
- malloc和calloc的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。
【举例说明】
int main()
{int* p = (int*)calloc(10, sizeof(int));if (p == NULL){perror("calloc");return 1; //如果为空则不执行下面的代码直接跳出}//使用int i = 0;for ( i = 0; i < 10; i++){printf("%d ", *(p + i));}//释放free(p);p == NULL;return 0;
}
【运行结果】

2.3、realloc
realloc函数的出现让动态内存管理更加灵活。
有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小的调整。

- ptr 是要调整的内存地址
- size 调整之后新大小
- 返回值为调整之后的内存起始位置。
- 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间。
- realloc在调整内存空间的是存在两种情况:
- 原有空间之后有足够大的空间
- 原有空间之后没有足够大的空间

- 情况1:当是情况1的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化。
- 情况2:当是情况2的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适大小的连续空间来使用。将旧空间中的数据拷贝到新的空间中,释放掉旧的空间,然后返回新空间的地址,这样函数返回的是一个新的内存地址。
【举例说明】
注意:realloc也可能开辟空间失败,失败是返回NULL。因此不能直接将realloc开辟的空间直接赋值给原指针p,因为这样做会导致当realloc开辟失败时p直接被置成NULL了,那么就意味着不但realloc没有调整大小,反而把p原有的内容丢失了。所以此处需要用一个tmp先接收返回值,当判断了 返回值不为NULL时再将tmp赋值给p。
int main()
{int* p = (int*)calloc(10, sizeof(int));if (p == NULL){perror("calloc");return 1; //如果为空则不执行下面的代码直接跳出}//使用int i = 0;for ( i = 0; i < 10; i++){printf("%d ", *(p + i));}//空间不够,realloc调整为20个intint* tmp = (int*)realloc(p, 20 * sizeof(int));if (tmp != NULL){p = tmp;}//使用//释放free(p);p == NULL;return 0;
}
3、常见的动态内存错误
3.1、对NULL指针的解引用操作
int main()
{int* p = (int*)malloc(INT_MAX / 4);//不做返回值判断,就可能使用NULL指针*p = 20;//如果p的值是NULL,就会有问题free(p);return 0;
}
3.2、对动态开辟空间的越界访问
int main()
{int i = 0;int* p = (int*)malloc(10 * sizeof(int));if (NULL == p){return 1;}for (i = 0; i <= 10; i++){*(p + i) = i;//当i是10的时候越界访问}free(p);return 0;
}
3.3、对非动态开辟内存使用free释放
free只能用于释放malloc、calloc、realloc开辟的空间。
int main()
{int a = 10;int* p = &a;free(p);//错误操作return 0;
}
3.4、使用free释放一块动态开辟内存的一部分
只释放一部分会报错。尽量避免让p自己移动位置,如果非要移动,应该再定义一个指针,让新定义的指针动。
int main()
{int* p = (int*)malloc(100);p++;free(p);//p不再指向动态内存的起始位置return 0;
}
3.5、对同一块动态内存多次释放
重复释放会导致报错,而为了防止对p进行重复释放,就应该使用free和NULL组合,即free(p); p = NULL;。当p被置成空指针后再对其进行free不会有任何问题。
因为上文讲解free时说过:free中如果参数 ptr 是NULL指针,则函数什么事都不做。
int main()
{int* p = (int*)malloc(100);free(p);free(p);//重复释放return 0;
}

【正确做法】
int main()
{int* p = (int*)malloc(100);free(p);p = NULL;free(p);//重复释放return 0;
}
3.6、动态开辟内存忘记释放(内存泄漏)
在test内使用malloc开辟的空间,在没有free释放之前退出了test,一旦退出了test并且没有通过return带回指针p,那么该空间就永远没有人能找得到了,变成垃圾保存在内存中,造成内存泄漏,这是非常可怕的。
void test()
{int* p = (int*)malloc(40);if (NULL != p){*p = 20;}
}int main()
{test();while (1)//这里表示程序还在一直运行,不会结束,例如服务器{;}return 0;
}
4、动态内存的经典笔试题
以下这些题的原题出自 :《高质量C/C++编程》
【题目1】
请问运行Test 函数会有什么样的结果?
void GetMemory(char* p)
{p = (char*)malloc(100);
}
void Test(void)
{char* str = NULL;GetMemory(str);strcpy(str, "hello world");printf(str);
}int main()
{Test();return 0;
}
【解析】
程序崩溃,内存泄漏,所以不会打印任何内容。
GetMemory的参数是指针,指针内的内容是NULL,即此时形参p也存放着一个NULL,然后用malloc给p开辟了100字节的空间,将这100个字节的空间的首地址存入p指针中,p就指向了这100个字节的空间。此时GetMemory函数完成操作后并没有free释放空间或者return返回指向空间的地址就退出函数了,那么由p指向的这块空间就存在内存泄漏的问题,因为是值传递这里改变p并不会影响str,此时str仍然是空指针NULL,无法正常进行strcpy操作,发生了程序对NULL的解引用操作,程序崩溃。

【题目2】
请问运行Test 函数会有什么样的结果?
char* GetMemory(void)
{char p[] = "hello world";return p;
}
void Test(void)
{char* str = NULL;str = GetMemory();printf(str);
}int main()
{Test();return 0;
}
【解析】
打印出乱码。
数组开辟在栈区当中,并且在GetMemory内创建,出了GetMemory函数就销毁。当用return返回p时,虽然p指向的地址值被返回了,但是p指向的地址的内容在出了GetMemory函数后已经被销毁,此时str接收到的指针是野指针,而对野指针操作就是非法访问内存。
这类问题统称返回栈空间地址的问题。

【题目3】
请问运行Test 函数会有什么样的结果?
void GetMemory(char** p, int num)
{*p = (char*)malloc(num);
}
void Test(void)
{char* str = NULL;GetMemory(&str, 100);strcpy(str, "hello");printf(str);
}int main()
{Test();return 0;
}
【解析】
可以成功打印hello。
此处GetMemory的参数为传址调用,*p就等于str,因此对*p进行动态内存开辟就等于对str动态内存开辟,所以可以正常打印处hello。但是malloc动态开辟的空间并没有用free释放,存在内存泄漏的风险。

【题目4】
请问运行Test 函数会有什么样的结果?
void Test(void)
{char* str = (char*)malloc(100);strcpy(str, "hello");free(str);if (str != NULL){strcpy(str, "world");printf(str);}
}int main()
{Test();return 0;
}
【解析】
str指向malloc开辟的空间,当使用free释放str,但是并没有对str置空,即str = NULL;,此时str是一个内容不为空的野指针,if(str != NULL)为真,因此会执行if中的strcpy,对野指针进行操作导致非法访问内存,此时程序崩溃,不会执行printf打印语句。
总结 :需要牢记,free和NULL要配套使用,释放完空间之后立即将指针置空,可以避开很多错误。
5、C/C++程序的内存开辟

C/C++程序内存分配的几个区域:
- 栈区(stack):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。
- 堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配方式类似于链表。
- 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
- 代码段:存放函数体(类成员函数和全局函数)的二进制代码。
有了这幅图,我们就可以更好的理解 static关键字 修饰局部变量的例子了。
实际上普通的局部变量是在栈区分配空间的,栈区的特点是在上面创建的变量出了作用域就销毁。
但是被static修饰的变量存放在数据段(静态区),数据段的特点是在上面创建的变量,直到程序结束才销毁,所以生命周期变长。
6、柔性数组
也许你从来没有听说过柔性数组(flexible array)这个概念,但是它确实是存在的。
C99中,结构中的最后一个元素允许是未知大小的数组,这就叫做『柔性数组』成员。
【举例说明】
struct S
{int i;int a[];//柔性数组成员————未知大小的数组
};
6.1、柔性数组的特点
- 结构中的柔性数组成员前面必须至少一个其他成员。
- sizeof 返回的这种结构大小不包括柔性数组的内存。
【举例说明】
struct S
{int i; //int是4个字节int a[];//并没有计算该数组大小
};int main()
{printf("%d\n", sizeof(struct S)); //结果就为4return 0;
}
到这里也就能理解了为什么 结构中的柔性数组成员前面必须至少一个其他成员,因为如果前面根本没有其他成员,那么sizeof就无法计算出大小。
6.2、柔性数组的使用
包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。
【举例说明】
简单来说就是:如果想要给数组开辟20个字节的空间,那么malloc开辟的时候必须加上其他成员的大小,也就是sizeof(struct S)的大小,即一次开辟了24个字节的空间,前面4个字节供成员 i 使用。
struct S
{int i;int a[];//柔性数组成员
};int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S) + 20); //4+20if (ps == NULL){perror("malloc");return 1;}free(ps);ps = NULL;return 0;
}

同样的,当开辟的空间不够,使用realloc调整大小时也需要加上其他成员的大小:
int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S) + 20); //4+20if (ps == NULL){perror("malloc");return 1;}//调整大小 20->40struct S* ptr = (struct S*)realloc(ps, sizeof(struct S) + 40);//4+40if (ptr != NULL){ps = ptr;}else{perror("realloc");return 1;}free(ps);ps = NULL;return 0;
}
6.3、柔性数组的优势
当讲解完柔性数组之后,有读者就会产生疑惑了,这个柔性数组感觉还挺麻烦的,那么它到底有什么优势呢?接下来我就用两段代码比较让大家知道柔性数组的优势。
【代码1】
使用指针a模拟柔性数组,对指针进行动态开辟。
struct S
{int i;int* a;
};int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S));if (ps == NULL){perror("malloc1");return 1;}ps->a = (int*)malloc(20);if (ps->a == NULL){perror("malloc2");return 1;}//使用//....//释放free(ps->a);ps->a = NULL;free(ps);ps = NULL;return 0;
}
【代码1内存图解】

【代码2】
struct S
{int i;int a[];//柔性数组成员
};int main()
{struct S* ps = (struct S*)malloc(sizeof(struct S) + 20); //4+20if (ps == NULL){perror("malloc");return 1;}//使用//....//释放free(ps);ps = NULL;return 0;
}
【代码2内存图解】

上面的【代码1】和【代码2】完成的功能都是一致的,【代码1】就是在不使用柔性数组时要完成该功能的基本写法。
但是我们能看出一些不同:
- 【代码1】中需要先给结构体动态开辟一块空间,然后再对结构体中的a指针再动态开辟一块空间,这里就使用了两次malloc来动态开辟,增加了代码量。而使用了柔性数组的【代码2】只需要对结构体整体使用malloc动态开辟一次适合的大小即可。
- 【代码1】中由于malloc开辟了两次空间,因此也需要使用两次free释放空间,并且释放顺序还不能错,必须先释放成员a指向的空间,再释放结构体空间。而【代码2】只需要释放一次。
因此使用柔性数组实现有两个好处:
- 方便操作:只需要一次malloc和free就可以把所有内存分配好与释放掉。
- 减少内存碎片和提高访问速度:如果在内存中频繁开辟空间,内存和内存之间就很容易留下一些缝隙,而这些缝隙又称之为内存碎片,内存碎片越多内存利用率就越低。并且连续的内存有益于提高访问速度。

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!
如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!
如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!
相关文章:
一篇博客学会系列(3) —— 对动态内存管理的深度讲解以及经典笔试题的深度解析
目录 动态内存管理 1、为什么存在动态内存管理 2、动态内存函数的介绍 2.1、malloc和free 2.2、calloc 2.3、realloc 3、常见的动态内存错误 3.1、对NULL指针的解引用操作 3.2、对动态开辟空间的越界访问 3.3、对非动态开辟内存使用free释放 3.4、使用free释放一块动态…...
【C++ techniques】虚化构造函数、虚化非成员函数
constructor的虚化 virtual function:完成“因类型而异”的行为;constructor:明确类型时构造函数;virtual constructor:视其获得的输入,可产生不同的类型对象。 //假如写一个软件,用来处理时事…...
蓝牙核心规范(V5.4)11.6-LE Audio 笔记之初识音频位置和通道分配
专栏汇总网址:蓝牙篇之蓝牙核心规范学习笔记(V5.4)汇总_蓝牙核心规范中文版_心跳包的博客-CSDN博客 爬虫网站无德,任何非CSDN看到的这篇文章都是盗版网站,你也看不全。认准原始网址。!!! 音频位置 在以前的每个蓝牙音频规范中,只有一个蓝牙LE音频源和一个蓝牙LE音频接…...
mysql双主+双从集群连接模式
架构图: 详细内容参考: 结果展示: 178.119.30.14(主) 178.119.30.15(主) 178.119.30.16(从) 178.119.30.17(从)...
嵌入式中如何用C语言操作sqlite3(07)
sqlite3编程接口非常多,对于初学者来说,我们暂时只需要掌握常用的几个函数,其他函数自然就知道如何使用了。 数据库 本篇假设数据库为my.db,有数据表student。 nonamescore4嵌入式开发爱好者89.0 创建表格语句如下: CREATE T…...
RandomForestClassifier 与 GradientBoostingClassifier 的区别
RandomForestClassifier(随机森林分类器)和GradientBoostingClassifier(梯度提升分类器)是两种常用的集成学习方法,它们之间的区别分以下几点。 1、基础算法 RandomForestClassifier:随机森林分类器是基于…...
计组——I/O方式
一、程序查询方式 CPU不断轮询检查I/O控制器中“状态寄存器”,检测到状态为“已完成”之后,再从数据寄存器取出输入数据。 过程: 1.CPU执行初始化程序,并预置传送参数;设置计数器、设置数据首地址。 2. 向I/O接口发…...
jsbridge实战2:Swift和h5的jsbridge通信
[[toc]] demo1: 文本通信 h5 -> app 思路: h5 全局属性上挂一个变量app 接收这个变量的内容关键API: navigation代理 navigationAction.request.url?.absoluteString // 这个变量挂载在 request 的 url 上 ,在浏览器实际无法运行,因…...
集合原理简记
HashMap 无论在构造函数是否指定数组长度,进行的都是延迟初始化 构造函数作用: 阈值:threshold,每次<<1 ,数组长度 负载因子 无参构造:设置默认的负载因子 有参:可以指定初始容量或…...
机器学习的超参数 、训练集、归纳偏好
一、介绍 超参数(Hyperparameters)和验证集(Validation Set)是机器学习中重要的概念,用于调整模型和评估其性能。 超参数: 超参数是在机器学习模型训练过程中需要手动设置的参数,而不是从数据…...
Leetcode1071. 字符串的最大公因子(三种方法,带详细解析)
Leetcode1071. 字符串的最大公因子 对于字符串 s 和 t,只有在 s t … t(t 自身连接 1 次或多次)时,我们才认定 “t 能除尽 s”。 给定两个字符串 str1 和 str2 。返回 最长字符串 x,要求满足 x 能除尽 str1 且 x 能…...
如何像人类一样写HTML之图像标签,超链接标签与多媒体标签
文章目录 前言一、图像标签1.1 什么是图像标签?2.2 如何使用图像标签? 二、超链接标签2.1 什么是超链接标签?2.2 如何使用超链接标签? 三、多媒体标签3.1 什么是多媒体标签?3.2 如何使用多媒体audio标签?3.…...
1300*C. Rumor(并查集贪心)
解析: 并查集,求每个集合的最小费用。 每次合并集合的时候,根节点保存当前集合最小的费用。 #include<bits/stdc.h> using namespace std; #define int long long const int N1e55; int n,m,a[N],p[N],cnt[N]; int find(int x){retur…...
python实用小代码(数据分析向)
一、python实现31省市名称替换 首先定义了一个字典province_dict,其中包含每个省市的无缩写名称与其标准名称之间的映射。然后,我们使用map()函数将每个省市名称从无缩写名称转换为标准名称,并将结果存储在新列省市标准名称中。 province_di…...
【oncmdmsg 鼠标】2023/8/19 上午9:50:14
2023/8/19 上午9:50:14 oncmdmsg 鼠标 2023/8/19 上午9:50:22 抱歉,您提到的 “oncmdmsg” 和 “鼠标” 是什么?请提供更多上下文信息,我将尽力帮助您。 2023/8/19 上午9:51:43 OnCmdMsg 2023/8/19 上午9:52:21 “OnCmdMsg” 是一个在 MFC (Microsoft Foundation Cla…...
插入排序:简单而有效的排序方法
在计算机科学中,排序算法是一个重要且常见的主题,它们用于对数据进行有序排列。插入排序(Insertion Sort)是其中一个简单但有效的排序算法。本文将详细解释插入排序的原理和步骤,并提供Java语言的实现示例。 插入排序的…...
OpenGL之光照贴图
我们需要拓展之前的系统,引入漫反射和镜面光贴图(Map)。这允许我们对物体的漫反射分量和镜面光分量有着更精确的控制。 漫反射贴图 我们希望通过某种方式对物体的每个片段单独设置漫反射颜色。我们仅仅是对同样的原理使用了不同的名字:其实都是使用一张覆盖物体的图像,让我…...
隐私交易成新刚需,Unijoin 凭什么优势杀出重围?
随着区块链技术的普及和发展,全球加密货币用户在持续增长,根据火币研究院公布的数据,2022年全球加密用户已达到 3.2亿人,目前全球人口总数超过了 80亿,加密货币用户渗透率已达到了 4%。 尤其是在 2020 年开启的 DeFi 牛…...
小谈设计模式(12)—迪米特法则
小谈设计模式(12)—迪米特法则 专栏介绍专栏地址专栏介绍 迪米特法则核心思想这里的“朋友”指当前对象本身以参数形式传入当前对象的对象当前对象的成员变量直接引用的对象目标 Java程序实现程序分析 总结 专栏介绍 专栏地址 link 专栏介绍 主要对目…...
Foxit PDF
Foxit PDF 福昕PDF 软件,可以很好的编辑PDF文档。 调整PDF页面大小 PDF文档中,一个页面大,一个页面小 面对这种情况,打开Foxit PDF 右键单击需要调整的页面,然后选择"调整页面大小". 可以选择…...
idea大量爆红问题解决
问题描述 在学习和工作中,idea是程序员不可缺少的一个工具,但是突然在有些时候就会出现大量爆红的问题,发现无法跳转,无论是关机重启或者是替换root都无法解决 就是如上所展示的问题,但是程序依然可以启动。 问题解决…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
脑机新手指南(八):OpenBCI_GUI:从环境搭建到数据可视化(下)
一、数据处理与分析实战 (一)实时滤波与参数调整 基础滤波操作 60Hz 工频滤波:勾选界面右侧 “60Hz” 复选框,可有效抑制电网干扰(适用于北美地区,欧洲用户可调整为 50Hz)。 平滑处理&…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...
学校招生小程序源码介绍
基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码,专为学校招生场景量身打造,功能实用且操作便捷。 从技术架构来看,ThinkPHP提供稳定可靠的后台服务,FastAdmin加速开发流程,UniApp则保障小程序在多端有良好的兼…...
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
引擎版本: 3.8.1 语言: JavaScript/TypeScript、C、Java 环境:Window 参考:Java原生反射机制 您好,我是鹤九日! 回顾 在上篇文章中:CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
均衡后的SNRSINR
本文主要摘自参考文献中的前两篇,相关文献中经常会出现MIMO检测后的SINR不过一直没有找到相关数学推到过程,其中文献[1]中给出了相关原理在此仅做记录。 1. 系统模型 复信道模型 n t n_t nt 根发送天线, n r n_r nr 根接收天线的 MIMO 系…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
让回归模型不再被异常值“带跑偏“,MSE和Cauchy损失函数在噪声数据环境下的实战对比
在机器学习的回归分析中,损失函数的选择对模型性能具有决定性影响。均方误差(MSE)作为经典的损失函数,在处理干净数据时表现优异,但在面对包含异常值的噪声数据时,其对大误差的二次惩罚机制往往导致模型参数…...
