Typedef 将测量单位放入 C 中
Typedef to put units of measurement in C
使用 typedef 将测量单位放入名称中是好习惯吗? (并重命名标准类型)像这样:
typedef int16_t MilliAmp_t; /* 1 mA */
您只是在重命名某些类型。它不会在编译器中带来任何额外的检查(除非该编译器是专门定制的;如果使用 GCC you could consider customizing it thru MELT,但该定制不是一项微不足道的任务),因为以下代码
MilliAmp_t x=0,y=1,z=2;
x = y * z;
将始终在没有警告的情况下编译(在 C 中),即使将两个电流物理相乘并将结果放入某个电流变量没有任何意义。
但是,您的 typedef 有一个小的 文档 值:如果您声明一个像
这样的原型
MilliWatt_t electical_power (MilliAmp_t current, MilliVolt_t tension);
然后 一些 读者可能会觉得这很有帮助(但也许对我没有帮助)。
是的,这样的 typedef 很常见。尽管 C 没有提供强类型来阻止您将电流量分配给电压变量,但这种做法确实可以让您一眼看出变量的含义,而无需将该信息编码到名称中。
当我亲自为小型微控制器编写物理分析代码时,我更愿意更仔细地选择范围,而不是仅仅选择“1 步 = 1 毫安”,以最大限度地提高精度。所以,我更喜欢 voltage_t
和 current_t
,然后在将它们与已知物理量相关联时使用全局转换系数。
在使用整数进行物理数学运算时,溢出和精度损失是重要的考虑因素。将变量保持在适当的物理单位有助于将中间结果保持在适当的动态范围内,从而避免这些问题,只要物理系统将其电压和电流保持在您为统一计算使用选择的范围内。
如果您有 FPU,我建议您使用它,不要为 SI 前缀烦恼,因为它会透明地处理毫、微和其他任何东西不用考虑系数。使用 typedef double amps;
仍将有助于以合理和说明的方式编写方程式代码,并且您可以在以后轻松调整整个程序的精度。一定要这样做。
正如其他人指出的那样,typedef
没有定义新类型,它引入了别名。
如果您真的想要这种级别的安全性,那么您可以将值装在 struct
中
typedef struct {
int16_t v;
} MillAmp_t ;
它需要打字开销:
#include <stdint.h>
#include <stdio.h>
typedef struct {
int16_t v;
} MilliAmp_t ;
typedef struct {
int16_t v;
} MilliOhms_t ;
typedef struct {
int16_t v;
} MilliWatt_t ;
MilliWatt_t power_loss(MilliAmp_t current,MilliOhms_t resistance){
//NB: Using int16_t makes this seriously under/overflow prone.
//So we upsize everything for the best shot at a result.
int64_t currentl=current.v;
int64_t resistancel=resistance.v;
int64_t result=currentl*currentl*resistancel/1000000L;
return (MilliWatt_t){result};
}
int main(void) {
MilliAmp_t current={1000};
MilliOhms_t resistance={2000};
MilliWatt_t power=power_loss(current,resistance);
printf("%d\n",(int)power.v);
//MilliWatt_t rewop=power_loss(resistance,current);//Compiler Error! Hurray!!
return 0;
}
但是开销并不大。你可以把它包在一个宏中:
#define VALUE(S) ((S).v)
但我不认为这有帮助。
不要被告诉您不需要此解决方案的人自动说服。如果您的计算非常复杂,涉及许多不同的单位,您可以为这种适度的开销省去很多心痛。
不过正常的约定是'mangle'把单位改成名字:
int16_ power_loss_MilliWatts(int16_t current_MilliAmps,int16_t resistance_MilliOhms);
这往往会有所帮助,因为程序员倾向于按名称引用参数并识别该后缀的意图。但显然编译器没有提供监督。
在你问之前,这个 'boxing' 不太可能在任何实际编译器上有任何内存开销或速度(当然是在没有调试的情况下编译时)。
PS:任何拒绝接受这个甚至可能有价值(在相关情况下)的人可能会用一些经典的单位转换来取乐 cock-ups。
http://mentalfloss.com/article/25845/quick-6-six-unit-conversion-disasters
还有哈勃 Space Telescope 和许多其他 cock-ups。
我已经看到很多(很多)unit-test 错误被排除了 100 倍,因为人们松散地和互换地使用术语百分比和因子。
使用 typedef 将测量单位放入名称中是好习惯吗? (并重命名标准类型)像这样:
typedef int16_t MilliAmp_t; /* 1 mA */
您只是在重命名某些类型。它不会在编译器中带来任何额外的检查(除非该编译器是专门定制的;如果使用 GCC you could consider customizing it thru MELT,但该定制不是一项微不足道的任务),因为以下代码
MilliAmp_t x=0,y=1,z=2;
x = y * z;
将始终在没有警告的情况下编译(在 C 中),即使将两个电流物理相乘并将结果放入某个电流变量没有任何意义。
但是,您的 typedef 有一个小的 文档 值:如果您声明一个像
这样的原型MilliWatt_t electical_power (MilliAmp_t current, MilliVolt_t tension);
然后 一些 读者可能会觉得这很有帮助(但也许对我没有帮助)。
是的,这样的 typedef 很常见。尽管 C 没有提供强类型来阻止您将电流量分配给电压变量,但这种做法确实可以让您一眼看出变量的含义,而无需将该信息编码到名称中。
当我亲自为小型微控制器编写物理分析代码时,我更愿意更仔细地选择范围,而不是仅仅选择“1 步 = 1 毫安”,以最大限度地提高精度。所以,我更喜欢 voltage_t
和 current_t
,然后在将它们与已知物理量相关联时使用全局转换系数。
在使用整数进行物理数学运算时,溢出和精度损失是重要的考虑因素。将变量保持在适当的物理单位有助于将中间结果保持在适当的动态范围内,从而避免这些问题,只要物理系统将其电压和电流保持在您为统一计算使用选择的范围内。
如果您有 FPU,我建议您使用它,不要为 SI 前缀烦恼,因为它会透明地处理毫、微和其他任何东西不用考虑系数。使用 typedef double amps;
仍将有助于以合理和说明的方式编写方程式代码,并且您可以在以后轻松调整整个程序的精度。一定要这样做。
正如其他人指出的那样,typedef
没有定义新类型,它引入了别名。
如果您真的想要这种级别的安全性,那么您可以将值装在 struct
typedef struct {
int16_t v;
} MillAmp_t ;
它需要打字开销:
#include <stdint.h>
#include <stdio.h>
typedef struct {
int16_t v;
} MilliAmp_t ;
typedef struct {
int16_t v;
} MilliOhms_t ;
typedef struct {
int16_t v;
} MilliWatt_t ;
MilliWatt_t power_loss(MilliAmp_t current,MilliOhms_t resistance){
//NB: Using int16_t makes this seriously under/overflow prone.
//So we upsize everything for the best shot at a result.
int64_t currentl=current.v;
int64_t resistancel=resistance.v;
int64_t result=currentl*currentl*resistancel/1000000L;
return (MilliWatt_t){result};
}
int main(void) {
MilliAmp_t current={1000};
MilliOhms_t resistance={2000};
MilliWatt_t power=power_loss(current,resistance);
printf("%d\n",(int)power.v);
//MilliWatt_t rewop=power_loss(resistance,current);//Compiler Error! Hurray!!
return 0;
}
但是开销并不大。你可以把它包在一个宏中:
#define VALUE(S) ((S).v)
但我不认为这有帮助。
不要被告诉您不需要此解决方案的人自动说服。如果您的计算非常复杂,涉及许多不同的单位,您可以为这种适度的开销省去很多心痛。
不过正常的约定是'mangle'把单位改成名字:
int16_ power_loss_MilliWatts(int16_t current_MilliAmps,int16_t resistance_MilliOhms);
这往往会有所帮助,因为程序员倾向于按名称引用参数并识别该后缀的意图。但显然编译器没有提供监督。
在你问之前,这个 'boxing' 不太可能在任何实际编译器上有任何内存开销或速度(当然是在没有调试的情况下编译时)。
PS:任何拒绝接受这个甚至可能有价值(在相关情况下)的人可能会用一些经典的单位转换来取乐 cock-ups。
http://mentalfloss.com/article/25845/quick-6-six-unit-conversion-disasters
还有哈勃 Space Telescope 和许多其他 cock-ups。 我已经看到很多(很多)unit-test 错误被排除了 100 倍,因为人们松散地和互换地使用术语百分比和因子。