- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我对用 C 语言编写相对较新。我使用在网上和打印品中找到的资源自学。这是我的第一个真正的 C 编程项目。一定要热爱在职培训。
我正在用 C 语言编写一些代码,该代码正在 Texas Instruments C6701 数字信号处理器上使用。具体来说,我正在编写一组通信函数来通过串行端口进行接口(interface)。
我正在进行的项目有一个现有的数据包协议(protocol),用于通过串行端口发送数据。这是通过传递一个指向要传输的数据及其字节长度的指针来实现的。我所要做的就是将要传输的字节写入内存中的“数组”中(发送器将该字节序列复制到缓冲区中并进行传输)。
我的问题涉及如何最好地格式化要传输的数据,我必须发送的数据由几种不同的数据类型组成(unsigned char、unsigned int、float 等...)。我无法将所有内容扩展为 float(或 int),因为我的通信带宽有限并且需要使数据包尽可能小。
本来想用数组来格式化数据,
unsigned char* dataTx[10];
dataTx[0]=char1;
dataTx[1]=char2;
etc...
这可以工作,除非我的数据不是所有的字符,有些是无符号整数或无符号短整型。
为了处理短整型和整型,我使用了位移位(暂时忽略小端与大端)。
unsigned char* dataTx[10];
dataTx[0]=short1>>8;
dataTx[1]=short1;
dataTx[2]=int1>>24;
dataTx[3]=int1>>16;
etc...
但是,我相信另一种(也是更好的?)方法来做到这一点是使用指针和指针算术。
unsigned char* dataTx[10]
*(dataTx+0) = int1;
*(dataTx+4) = short1;
*(dataTx+6) = char1;
etc...
我的问题(最后)是,哪种方法(位移或指针算术)是更可接受的方法?另外,跑得更快吗? (我也有运行时限制)。
我的要求:数据连续位于内存中,没有间隙、中断或填充。
我对结构的了解还不够,还不知道结构是否可以作为解决方案。具体来说,我不知道结构是否总是连续且不间断地分配内存位置。我读到一些内容表明它们以 8 字节 block 进行分配,并且可能引入填充字节。
现在我倾向于指针方法。感谢您阅读这篇看似很长的文章。
最佳答案
通常您会使用位移位方法,因为许多芯片不允许您将例如 4 字节整数复制到奇数字节地址(或者更准确地说,复制到从奇数字节地址)。这称为对齐。如果可移植性是一个问题,或者您的 DSP 不允许错位访问,则需要进行转移。如果您的 DSP 由于未对齐的访问而导致性能受到严重影响,您可能会担心。
但是,我不会编写如图所示的针对不同类型进行手动转换的代码。我希望使用函数(可能是内联)或宏来处理数据的序列化和反序列化。例如:
unsigned char dataTx[1024];
unsigned char *dst = dataTx;
dst += st_int2(short1, dst);
dst += st_int4(int1, dst);
dst += st_char(str, len, dst);
...
以函数形式,这些函数可能是:
size_t st_int2(uint16_t value, unsigned char *dst)
{
*dst++ = (value >> 8) & 0xFF;
*dst = value & 0xFF;
return 2;
}
size_t st_int4(uint32_t value, unsigned char *dst)
{
*dst++ = (value >> 24) & 0xFF;
*dst++ = (value >> 16) & 0xFF;
*dst++ = (value >> 8) & 0xFF;
*dst = value & 0xFF;
return 4;
}
size_t st_char(unsigned char *str, size_t len, unsigned char *dst)
{
memmove(dst, str, len);
return len;
}
诚然,这样的函数会让代码变得乏味;另一方面,它们也减少了出错的机会。您可以决定名称是否应为 st_uint2()
而不是 st_int2()
——事实上,您可以决定长度是否应以字节为单位(如此处)或以位为单位(如在参数类型中)。只要你始终如一且无聊,你就可以做你想做的事。您还可以将这些函数组合成更大的函数来封装整个数据结构。
现代编译器可能不需要掩码操作(& 0xFF
)。很久以前,我似乎记得它们对于避免某些平台上的某些编译器偶尔出现问题是必要的(因此,我的代码可以追溯到 20 世纪 80 年代,其中包含此类屏蔽操作)。上述平台可能已经安息了,所以我认为它们(仍然)在那里可能纯粹是偏执。
请注意,这些函数以大端顺序传递数据。这些函数可以在大端和小端机器上“按原样”使用,并且数据将在两种类型上正确解释,因此您可以使用此代码让不同的硬件通过网络进行通信,并且将会有没有沟通不畅。如果您要传达浮点值,则必须更多地担心网络上的表示形式。尽管如此,您可能应该以平台无关的格式传输数据,以便芯片类型之间的互操作尽可能简单。 (这也是我使用其中包含数字的类型大小的原因;特别是“int”和“long”在不同平台上可能意味着不同的事物,但 4 字节有符号整数仍然是 4 字节有符号整数,即使您是不幸 - 或幸运 - 足以拥有一台具有 8 字节整数的机器。)
关于c - 将不同数据类型的值按顺序写入内存?或者,具有多种数据类型的数组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7182402/
我在具有 2CPU 和 3.75GB 内存 (https://aws.amazon.com/ec2/instance-types/) 的 c3.large Amazon EC2 ubuntu 机器上运
我想通过用户空间中的mmap-ing并将地址发送到内核空间从用户空间写入VGA内存(视频内存,而不是缓冲区),我将使用pfn remap将这些mmap-ed地址映射到vga内存(我将通过 lspci
在 Mathematica 中,如果你想让一个函数记住它的值,它在语法上是很轻松的。例如,这是标准示例 - 斐波那契: fib[1] = 1 fib[2] = 1 fib[n_]:= fib[n] =
我读到动态内存是在运行时在堆上分配的,而静态内存是在编译时在堆栈上分配的,因为编译器知道在编译时必须分配多少内存。 考虑以下代码: int n; cin>>n; int a[n]; 如果仅在运行期间读
我是 Python 的新手,但我之前还不知道这一点。我在 for 循环中有一个基本程序,它从站点请求数据并将其保存到文本文件但是当我检查我的任务管理器时,我发现内存使用量只增加了?长时间运行时,这对我
我正在设计一组数学函数并在 CPU 和 GPU(使用 CUDA)版本中实现它们。 其中一些函数基于查找表。大多数表占用 4KB,其中一些占用更多。基于查找表的函数接受一个输入,选择查找表的一两个条目,
读入一个文件,内存被动态分配给一个字符串,文件内容将被放置在这里。这是在函数内部完成的,字符串作为 char **str 传递。 使用 gdb 我发现在行 **(str+i) = fgetc(aFil
我需要证实一个理论。我正在学习 JSP/Java。 在查看了一个现有的应用程序(我没有写)之后,我注意到一些我认为导致我们的性能问题的东西。或者至少是其中的一部分。 它是这样工作的: 1)用户打开搜索
n我想使用memoization缓存某些昂贵操作的结果,这样就不会一遍又一遍地计算它们。 两个memoise和 R.cache适合我的需要。但是,我发现缓存在调用之间并不可靠。 这是一个演示我看到的问
我目前正在分析一些 javascript shell 代码。这是该脚本中的一行: function having() { memory = memory; setTimeout("F0
我有一种情况,我想一次查询数据库,然后再将整个数据缓存在内存中。 我得到了内存中 Elasticsearch 的建议,我用谷歌搜索了它是什么,以及如何在自己的 spring boot 应用程序中实现它
我正在研究 Project Euler (http://projecteuler.net/problem=14) 的第 14 题。我正在尝试使用内存功能,以便将给定数字的序列长度保存为部分结果。我正在
所以,我一直在做 Java 内存/注意力游戏作业。我还没有达到我想要的程度,它只完成了一半,但我确实让 GUI 大部分工作了......直到我尝试向我的框架添加单选按钮。我认为问题可能是因为我将 JF
我一直在尝试使用 Flask-Cache 的 memoize 功能来仅返回 statusTS() 的缓存结果,除非在另一个请求中满足特定条件,然后删除缓存。 但它并没有被删除,并且 Jinja 模板仍
我对如何使用 & 运算符来减少内存感到非常困惑。 我可以回答下面的问题吗? clase C{ function B(&$a){ $this->a = &$a; $thi
在编写代码时,我遇到了一个有趣的问题。 我有一个 PersonPOJO,其 name 作为其 String 成员之一及其 getter 和 setter class PersonPOJO { priv
在此代码中 public class Base { int length, breadth, height; Base(int l, int b, int h) { l
Definition Structure padding is the process of aligning data members of the structure in accordance
在 JavaScript Ninja 的 secret 中,作者提出了以下方案,用于在没有闭包的情况下内存函数结果。他们通过利用函数是对象这一事实并在函数上定义一个属性来存储过去调用函数的结果来实现这
我正在尝试找出 map 消耗的 RAM 量。所以,我做了以下事情;- Map cr = crPair.collectAsMap(); // 200+ entries System.out.printl
我是一名优秀的程序员,十分优秀!