空间换时间,查表法的经典例子

前言

上一篇分享了:C语言精华知识:表驱动法编程实践

这一篇再分享一个查表法经典的例子。

我们怎么衡量一个函数/代码块/算法的优劣呢?这需要从多个角度看待。本篇笔记我们先不考虑代码可读性、规范性、可移植性那些角度。

在我们嵌入式中,我们需要根据实际资源的情况来设计我们的代码。

比如当我们能用的存储器空间极其有限的情况,我之前就有遇到这样子的情况,我能用的flash空间只有4KB,但是要实现的功能很多,稍微不注意就超了,这种情况下我们就得多考虑程序占用方面的问题。

如果我们的存储器空间很足,有时候可以牺牲一些存储器空间来换取我们程序的运行速度。查表法就是以空间换取时间的典型例子。下面看一个经典的例子:

基础例子

编写程序统计一个4bit数据(0x0~0x0F)中1的个数。这里提供两种方法:

1、方法一:常规法

常规法就是依次判断这个4bit的数据的每一位是否为1,并用一个计数变量把1的个数记录下来:

左右滑动查看全部代码>>>

#include <stdio.h>/* 测试结果 */
struct test_res
{unsigned int data;  /* 数据         */unsigned int count; /* 数据中1的个数 */
};struct test_res get_test_res(unsigned int data)
{/* 保存测试结果 */struct test_res res;/* 保证数据总会在0~0xf之间 */unsigned int temp = data & 0xf;  res.count = 0;res.data = temp;/* 循环判断每一位 */for (int i = 0; i < 4; i++){if (temp & 0x01){res.count++;}temp >>= 1;}return res;
}int main(void)
{struct test_res res = {0};for (int i = 0; i < 32; i++){res = get_test_res(i);printf("%2d中二进制位为1的个数有%d\n", res.data, res.count);}return 0;
}

运行结果:

unsigned int temp = data & 0xf; 语句就是为了保证数据都是在0x0~0xf之间,即0~15为一个周期,如果输入的数据为16,则当做0来看待,输入的数据为17,则当做1来看待……

2、方法二:查表法

这个例子也可以用查表法来做,把0x0~0xF中的所有数据中每个数据的1的个数都记录下来,存放到一个表中。

这样一来,数据数据中1的个数就建立起了一一对应关系,我们就可以通过数组索引来获取我们想要的结果:

左右滑动查看全部代码>>>

int table[16] = {0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4};struct test_res get_test_res(unsigned int data)
{/* 保存测试结果 */struct test_res res;/* 保证数据总会在0~0xf之间 */unsigned int temp = data & 0xf;  /* 获取结果 */res.data = temp;res.count = table[temp];return res;
}

常规法使用for循环的方式来实现,缺点是占用了不少处理器的时间;查表法的优点弥补了常规法的不足,但是额外占用了一些静态空间。

这里针对这个应用而言处理的数据还是比较简单的,数据范围只是0x0~0xF之间,所以这两种方式可能也都差不多。

那如果以上题目稍微改一下:编写程序统计一个8bit、16bit数据中1的个数。查表法换取的时间就比较明显了。

延伸例子

下面我们先来看一下编写程序统计一个8bit(0x0~0xFF)数据中1的个数的情况。

1、常规法

把以上代码稍微改一下就可以:

左右滑动查看全部代码>>>

struct test_res get_test_res(unsigned int data)
{/* 保存测试结果 */struct test_res res;/* 保证数据总会在0~0xf之间 */ unsigned int temp = data & 0xff;  res.count = 0;res.data = temp;/* 循环判断每一位 */for (int i = 0; i < 16; i++){if (temp & 0x01){res.count++;}temp >>= 1;}return res;
}

运行结果:

2、查表法

上面的数据范围仅仅是0x0~0xF,数据量比较少,建立数据表也比较容易。

这里的数据量范围变成了0x0~0xFF,比原来多了两百多个数据,这也还可以接受,也还可以全都列出来。

但是针对这里的这个问题有更好的方法:

在这个问题中,8bit的数据可以看做两个4bit数据,这样就可以共用上面4bit数据的数据表。所以我们只要把2个4bit数据的1的个数相加,就是最后的结果。

获取8bit数据1的个数:

左右滑动查看全部代码>>>

struct test_res get_test_res(unsigned int data)
{/* 保存测试结果 */struct test_res res;/* 保证数据总会在0~0xf之间 */unsigned int temp = data & 0xff;  /* 获取低4位中1的个数 */unsigned int low_data = temp & 0xf;unsigned int low_cnt = table[low_data];/* 获取高4位中1的个数 */unsigned int high_data = (temp >> 4) & 0xf;unsigned int high_cnt = table[high_data];/* 结果 */res.count = low_cnt + high_cnt;res.data = temp;return res;
}

同样的,获取16bit数据也是类似的,把16bit数据当做4个4bit数据。

针对以上这个查表法的例子我们可以总结出:

1、数据表的确定要合适。像上面8bit的情况再重新创建一个数据表把表元素列出来也还可以接受。但是如果是16bit这样子大数据的情况,建立这么大的数据表也不太现实。所以需要考虑如何建立一个合适的数据表。

2、需要权衡空间换取时间是否值得。像16bit这样子大数据的情况,全部列出来的话会大幅度的增加我们的存储开销,这种以空间换时间的情况可能会得不偿失。

推荐阅读:

    专辑|Linux文章汇总

    专辑|程序人生

    专辑|C语言

嵌入式Linux

微信扫描二维码,关注我的公众号 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/467344.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux内核系统架构介绍

28年前(1991年8月26日)Linus公开Linux的代码&#xff0c;开启了一个伟大的时代。这篇文章从进程调度&#xff0c;内存管理&#xff0c;设备驱动&#xff0c;文件系统&#xff0c;网络等方面讲解Linux内核系统架构。Linux的系统架构是一个经典的设计&#xff0c;它优秀的分层和模…

这道笔试题竟然运行不出错

#读者提供的面试题下面这张截图是一个读者在面试的时候遇到的题目&#xff0c;是哪个公司的我就不说出来了&#xff0c;我在微信朋友圈发了这个题目后&#xff0c;有几个好友给我留言说自己也写了这道题。题目&#xff1a;下面这段代码有什么问题&#xff1f;#后续然后我就用这…

android加号底部导航栏,EasyNavigation Android 底部导航栏████几行代码实现 Tab 导航(随意定制加号,带红点消息提示) @codeKK Android开源站...

几行代码轻松实现底部导航栏(Tab 文字图片高度随意更改)&#xff1b;中间可添加加号按钮&#xff0c;也可添加文字&#xff1b;(足够的属性满足你需要实现的加号样式)如果还不能满足、中间可添加自定义 View&#xff1b;Tab 中随意添加小红点提示、数字消息提示&#xff1b;点击…

在工厂的这几天

上周在工厂呆了6天支持我们产品量产&#xff0c;说下自己在工厂看到的一些事情&#xff0c;可能对大家对嵌入式行业有一定的认识&#xff0c;这样大家在选择方向会有些借鉴作用。产品由研发到生产是一个过程&#xff0c;只有经受过量产考验的产品&#xff0c;才能说这个产品真正…

Arm华为NXP睿赛德大咖云集!2020中国嵌入式技术大会嘉宾揭晓

展会即将开幕▲扫码领门票从电子信息产业趋势看&#xff0c;以自动驾驶、智能机器人和智慧安防为代表的AIOT&#xff08;智联网&#xff09;正在成为新一轮科技创新制高点。5G 布置将大大提升端侧智能&#xff0c;助力AIOT 发展。从技术层面看&#xff0c;AIOT 是AI 技术嵌入到…

c语言变量加常量,C语言(二)---常量与变量(示例代码)

一、进制1.1 二进制1.1 简介二进制数据是用0和1两个数码来表示的数。它的基数是2&#xff0c;进位规则是“逢二进1”&#xff0c;借位规则是“借一当二”。当前的计算机系统使用的基本上是二进制系统。1.2 二进制数据表示法二进制没有2只有0和1&#xff0c;逢2进1&#xff0c;其…

使用CodeSmith要注意的几点

新的项目即将开始了&#xff0c;项目经理老蔡说上次项目的持久层框架不太适合B/S项目&#xff0c;原因是反射用得太多了&#xff0c;并且数据层内部自己又封装了好几层&#xff0c;所以决定从新设计一个执行效率更高的框架。 老蔡设计了下他构思好的持久层和数据层的接口&am…

Alsa里面恶心的DAPM

相关文章音频系统&#xff0c;Alsa 里面的buff 是怎么计算的&#xff1f;为什么需要超过48k的采样音频&#xff1f;我在MTK平台下调试音频ALSA音频几个重要的参数openwrt 音频开发(干货)Ai音箱和Linux音频驱动小谈Android 音频数据流分析之程序员干架产品经理正文做音频的很多初…

从Linus Torvalds一封发飙的电邮开始谈设备树究竟是棵什么树?

[导读] 新版的U-Boot以及内核都引入了设备树&#xff0c;那么这究竟是棵什么样的树呢&#xff1f;长啥样&#xff1f;有啥用&#xff1f;为啥弄个这样的树&#xff1f;本文基于对设备树标准的理解&#xff0c;来学习整理一下相关的要点&#xff0c;供大家参考。Linux为啥要设备…

html两个div间有白线,html-在特定浏览器宽度下,白线出现在渐变填充div的末尾

我有一个ID为#gradient_div的div,其背景图像设置为线性渐变.仅在某些浏览器窗口宽度下,线性渐变的末尾与div #gradient_div的末尾之间才出现缝隙.当我拉伸和收缩浏览器窗口时,这条白线消失并重新出现.似乎与边距有关&#xff1a;>当我将边距设置为margin&#xff1a;0 1&…

为什么读书了,还不如那些初中毕业的同学

今天群里特别热闹&#xff0c;大家聊了很多&#xff0c;我在旁边看着&#xff0c;很舒服&#xff0c;可能也有人跟我一样&#xff0c;看着这些B神吹牛。像读书时候的卧谈会&#xff0c;大家都聊的很嗨&#xff0c;听的人更嗨&#xff0c;我就是那个喜欢听大家吹牛的人&#xff…

为什么说卷积神经网络,是深度学习算法应用最成功的领域之一?

目前&#xff0c;作为深度学习的代表算法之一&#xff0c;卷积神经网络&#xff08;Convolutional Neural Networks&#xff0c;CNN&#xff09;在计算机视觉、分类等领域上&#xff0c;都取得了当前最好的效果。后来&#xff0c;基于深度神经网络和搜索树的智能机器人“AlphaG…

C语言博客作业--函数

C语言博客作业--函数 一&#xff0c;PTA实验作业 6-2 简单实现x的n次方 1.本题PTA提交列表 2.设计思路 double mypow( double x, int n )函数&#xff08;1&#xff09;定义y1&#xff0c;i用于循环变量&#xff08;2&#xff09;for(i1;i<n;i){ yy*x}用循环实现x的n次方&am…

Linux系统编程 / triggerhappy 源码分析(3.select 的应用)

哈喽&#xff0c;我是老吴&#xff0c;继续记录我的学习心得。一、进步的滞后性我们期望进步是线性&#xff1a;每一个人付出一些努力后&#xff0c;都希望它有立竿见影的效果。现实是&#xff1a;做出努力后&#xff0c;结果的显现往往滞后。只有在几个月或几年后&#xff0c;…

今天,给我妈打电话聊了我爸

晚上&#xff0c;给我妈打电话&#xff0c;我爸前两天病了&#xff0c;前几年的大病虽然慢慢康复&#xff0c;但是因为年纪大了&#xff0c;身体机能也慢慢变弱&#xff0c;总是有一些大大小小的问题。前两天我发消息给我爸&#xff0c;我说我们要进新房子了&#xff0c;虽然房…

A Simple Note on P4FPGA: A Rapid Prototyping Framework for P4

论文&#xff1a;P4FPGA: A Rapid Prototyping Framework for P4 Github&#xff1a;https://github.com/p4fpga Reference: Han Wang, Robert Soule ́, Huynh Tu Dang, Ki Suh Lee, Vishal Shrivastav, Nate Foster, and Hakim Weatherspoon. 2017. P4FPGA : A Rapid Prototy…

客户端回调

Client Callback 是ASP.NET 2.0新增的一个特性。简单的说&#xff0c;就是在不刷新页面的情况下&#xff0c;用javascript向服务器端传递参数、调用服务器端的方法、并且得到服务器端的返回值进行处理。 1> Why Client Callback HTTP是无状态的协议。在HTTP协议之上开发的…

万事开头难 - 介绍IMX6ULL启动方式

不同开发板&#xff0c;启动方式不一样&#xff0c;今天我们来介绍imx6ull开发板的启动方式&#xff0c;这非常重要。若不了解清楚启动方式&#xff0c;后面的所有开发工作便无从谈起。本文摘自100ask_imx6ull 开发板 配套学习手册-《嵌入式Linux应用开发完全手册_韦东山全系列…

可以在中断服务程序执行malloc吗?

这是微信群里面的一个群友提的问题&#xff0c;原问题如下&#xff1a;今天遇到一个面试问题 请教各位老哥 中断服务程序能否malloc &#xff1f;为什么&#xff1f;我回答是 不行 因为中断服务程序应该要尽量简短&#xff0c;且执行完中断服务程序会跳出中断函数&#xff0c…

Android Input 子系统初探

Android系统基于Linux内核实现&#xff0c;内核作为整个操作系统的核心&#xff0c;对下&#xff0c;它负责整个硬件的驱动、实现对硬件器件的控制管理&#xff1b;对上&#xff0c;它提供各种系统所需的核心功能。Android系统支持的输入设备较多&#xff0c;如按键、触摸屏、手…