huaqingyuanjian 发表于 2024-11-27 17:27

干货!嵌入式C语言编程小知识总结

本帖最后由 huaqingyuanjian 于 2024-11-27 17:29 编辑

<p>1. 流水线被指令填满时才能发挥最大效能,即每时钟周期完成一条指令的执行(仅指单周期指令)。</p>

<p>如果程序发生跳转,流水线会被清空,这将需要几个时钟才能使流水线再次填满。因此,尽量少的使用跳转指令可以提高程序执行效率,解决发案就是尽量使用指令的&ldquo;条件执行&rdquo;功能。</p>

<p>&nbsp;</p>

<p>2. 在LPC2200系列中:可以通过过下面的程序延迟10毫秒:</p>

<p>for(i=0;i&lt;200;i++)</p>

<p>{</p>

<p>for(j=0;j&lt;200;j++);</p>

<p>}</p>

<p>&nbsp;</p>

<p>3. 通过下面语句将一个16位的变量放在两个8位的变量中。</p>

<p>//IP数据报总长度高字节</p>

<p>IpHeadUint8=(IpHead.e_ip.Crc&amp;0xff00)&gt;&gt;8;</p>

<p>//IP数据报总长度低字节</p>

<p>IpHeadUint8=IpHead.e_ip.Crc&amp;0x00ff;</p>

<p>&nbsp;</p>

<p>4. 在对全部数组元素赋初值时,可以不指定数组长度。</p>

<p>eg;inta[]={1,2,3,4,5};</p>

<p>但如果当输出第a以上的元素时,系统回输出随机数值,所以使用此方法时,不能使用超过初始值元素以上的元素。</p>

<p>&nbsp;</p>

<p>5. 由于ADS先天性的对printf不支持;因此不便于我们调试,可以利用串口输出来代替printf来调试。</p>

<p>&nbsp;</p>

<p>6. 用或运算,可使某位置为1,其它位不变</p>

<p>eg: PINSEL0 |= 0x00000005; //设置串口引脚</p>

<p>使第0位和第二位置一,其他位不变。</p>

<p>&nbsp;</p>

<p>7. 函数指针</p>

<p>1&gt; C语言中函数名直接对应于函数生成的指令代码在内存中的地址,因此函数名可以直接赋给指向函数的指针</p>

<p>2&gt; 调用函数实际上等同于&ldquo;调用指令+参数传递处理+回归位置入栈&rdquo;,本质上最核心的操作是将函数生成的目标代码的首地址赋给CPU的PC寄存器。</p>

<p>3&gt; 因为函数调用的本质是跳转到某一个地址单元的code去执行,所以可以&ldquo;调用一个根本就不存在在函数实体</p>

<p>4&gt; int (*p)();定义p是一个指向函数的指针变量,次函数返回带回整型的返回值。*P两侧的括号不能省略,表示p先于*结合,是指针变量,然后再与后面的()结合,表示此指针指向函数。</p>

<p>区别:int *p()表示这个函数的返回值是指向整型变量的指针。</p>

<p>说明:</p>

<p>(1) 指向函数的指针变量的一般定义形式为:</p>

<p>数据类型 (*指针变量名)();</p>

<p>1&gt; 此处的&ldquo;数据类型&rdquo;是指函数返回值的类型</p>

<p>(2) 返回指针值的函数:</p>

<p>类型名 *函数名(参数表)</p>

<p>eg: int * func(int x,int y)</p>

<p>func是函数名,调用它以后能返回一个指向整型数据的指针。x,y是func的形参。</p>

<p>区别方法:</p>

<p>a.从右往左找第一个括号,括号里面的是函数的形参。</p>

<p>b.括号外面的第一个标识符是函数的名字,函数前面的表示函数的返回数值。</p>

<p>&nbsp;</p>

<p>8. 数组指针</p>

<p>1&gt;int (*p)</p>

<p>表示*p有4个元素,每个元素为整型。也就是p所指的对象有4个整型元素的数组,既P是行指针。</p>

<p>2&gt; 指针数组</p>

<p>&Oslash; 一个数组,其元素均为指针类型数据,称为指针数组;即指针数组中的每一个元素都相当于一个指针变量。</p>

<p>&Oslash; 一维指针数组的定义形式为:</p>

<p>类型名 *数组名[数组长度]</p>

<p>eg:int *p:</p>

<p>作用:它用于指向若干个字符串,使字符串处理更加方便灵活。适用于一个二维字符串数组,其中每一行的字符数组的长度各不相同</p>

<p>eg:char * name[]={&ldquo;Follow me&rdquo;,&rdquo;BASIC&rdquo;,&rdquo;GreatWall&rdquo;};</p>

<p>&nbsp;</p>

<p>9. 结构体</p>

<p>1&gt; 可以用结构体变量做实参。但是用结构体变量作实参时,采取的是&ldquo;值传递&rdquo;的方式,将结构体变量所占的内存单元的内容全部顺序递给形参。形参也必须是同类型的结构体变量。</p>

<p>eg:pint(su);//注在此处su为结构体</p>

<p>注:这种传递方式在空间和时间上开销较大,如果结构体的规模较大时,开销是很可观的。</p>

<p>2&gt; 用直向结构体变量(或数组)的指针作实参,将结构体变量(或数组)的地址传给形参</p>

<p>eg:print(&amp;su);//注在此处su为结构体</p>

<p>&nbsp;</p>

<p>10. 共用体</p>

<p>1&gt; 共用体把几种不同数据类型的变量存放在同一块内存里。公用体中的变量共享同一块内存。</p>

<p>2&gt; 定义公用体类型变量的一般形式为:</p>

<p>union 共用体名</p>

<p>{</p>

<p>成员列表;</p>

<p>}变量列表;</p>

<p>&nbsp;</p>

<p>3&gt;在共用体中同一块内存可以用来存放几种不同类型的数据,但在某一时刻只能在其中存放一个成员变量。共用体变量中起作用的成员是最后一次存入的数据。</p>

<p>eg: union data</p>

<p>{</p>

<p>int i;</p>

<p>char c;</p>

<p>double d;</p>

<p>};</p>

<p>union data a;</p>

<p>共用体变量a中的成员i,c,d三个变量在内存中从同一个地址开始存储。如进行如下赋值:</p>

<p>a.i = 100;</p>

<p>a.c = &lsquo;A&rsquo;;</p>

<p>那么此时共用体变量a中的成员i已经没有值了,因为存储该值的内存现在已经被用来存储成员c的值了。</p>

<p>&nbsp;</p>

<p>3&gt; 共用体变量的长度取决于其成员的最大长度。</p>

<p>说明:</p>

<p>结构体变量所占内存的长度是各个成员的总和,每个成员分别占有自己的存储空间。共用体变量所占内存的长度是其最长成员的长度。当然,编译器出于提高访问效率的目的,在编译分配存储空间时往往要进行对齐操作。</p>

<p>对齐操作以最大基本类型为准。即以最大基本类型为基本单元。若按实际算下的长度不是基本单元的整数倍,则其实际长度应该是基本单元的整数倍。</p>

<p>(在TurboC中不进行对齐,在Linux中进行对齐)</p>

<p>&nbsp;</p>

<p>11. CPU字长与存储器位宽不一致处理</p>

<p>例如:使用共用体来解决这一冲突:</p>

<p>union send_temp{</p>

<p>uint16 words;</p>

<p>uint8 bytes;</p>

<p>}send_buff;</p>

<p>eg:send_buff.bytes=a;//此处a 是8位</p>

<p>send_buff.bytes=b;//此处 b 是8位;</p>

<p>此时就将8位字拼成了16位字存储了。</p>

<p>发送时send(send_buff.words)就可以每次发送一个16位的数据了。</p>

<p>&nbsp;</p>

<p>12. C语言符号优先级:</p>

<p>复合赋值运算符号:</p>

<p>a+=3*5;</p>

<p>等价于a=a+(3*5);</p>

<p>&nbsp;</p>

<p>13.一个常见的调试策略是把一些printf函数的调用散布于程序中,确定错误出现的具体位置。</p>

<p>但是,这些函数调用的输出结果被写入到缓冲区中,并不立即显示于屏幕上。事实上,如果程序失败,缓冲输去可能不会被实际写入,因此得到的错误位置就是错误的。</p>

<p>解决的方法是在每个用于调试的printf函数之后立即调用fflush函数即可得到。</p>

<p>printf(&ldquo;something or other&rdquo;);</p>

<p>fflush(stdout);</p>

<p>&nbsp;</p>

<p>14.关键字volatile的用法</p>

<p>volatile变量可能用于如下几种情况:</p>

<p>1&gt;设备的硬件寄存器(如:状态寄存器)</p>

<p>2&gt;一个中断服务子程序中会访问到的全局变量</p>

<p>3&gt;多线程应用中被几个任务共享的变量。</p>

<p>&nbsp;</p>

<p>15.关键字register的用法:</p>

<p>当对一个变量频繁被读写时,需要反复访问内存,从而花费大量的存取时间。为此,C语言提供了一种变量,即寄存器变量。这种变量存放在CPU的寄存器中,使用时,不需要访问内存,而直接从寄存器中读写,从而提高效率。寄存器变量的说明符是register。对于循环次数较多的循环控制变量及循环体内反复使用的变量均可定义为寄存器变量,而循环计数是应用寄存器变量的最好候选者。</p>

<p>(1) 只有局部自动变量和形参才可以定义为寄存器变量。因为寄存器变量属于动态存储方式,凡需要采用静态存储方式的量都不能定义为寄存器变量,包括:模块间全局变量、模块内全局变量、局部static变量;</p>

<p>(2) register是一个&quot;建议&quot;型关键字,意指程序建议该变量放在寄存器中,但最终该变量可能因为条件不满足并未成为寄存器变量,而是被放在了存储器中,但编译器中并不报错(在C++语言中有另一个&quot;建议&quot;型关键字:inline)。</p>

<p>&nbsp;</p>

<p>16.对于程序代码,已经被烧录在FLASH或ROM中,我们可以让CPU直接从其中读取代码执行,但通常这不是一个好办法,我们最好在系统启动后将FLASH或ROM中的目标代码拷贝入RAM中后再执行以提高取指令速度。</p>

<p>CPU对各种存储器的访问速度,基本上是:</p>

<p>CPU内部RAM &gt; 外部同步RAM &gt; 外部异步RAM &gt; FLASH/ROM</p>

<p>&nbsp;</p>

<p>17. 宏定义</p>

<p>在C语言中,宏是产生内嵌代码的唯一方法。对于嵌入式系统而言,为了能达到性能要求,宏是一种很好的代替函数的方法。</p>

<p>1&gt;宏定义&ldquo;像&rdquo;函数;</p>

<p>2&gt;宏定义不是函数,因而需要括上所有&ldquo;参数&rdquo;;</p>

<p>3&gt;宏定义可能产生副作用。因而不要给宏定义传入有副作用的&quot;参数&quot;。</p>

<p>来源:嵌入式ARM</p>

<p>嵌入式的学习之路是非常漫长且需要毅力的,学习路线不对或者学习不够深入都是很多人会遇到的问题。为大家整理了一份100多G的学习包,基本上涵盖了嵌入式学习的内容,大家按需取ᵛ:ᶠᵃʳ⁻ˢⁱᵍʰᵗ&sup2;&sup3;</p>

freebsder 发表于 2024-11-27 18:53

<p>这帖子转的好老了,还有register ... 没记错的话c99之后就删掉了</p>

今天我注册8月26日 发表于 2024-11-27 21:17

<p>啥时间干货啊</p>

许的说法 发表于 2024-12-3 09:41

<p>不错,干货满满,单片机c语言学习有帮助。。嵌入式ARM</p>
页: [1]
查看完整版本: 干货!嵌入式C语言编程小知识总结