9

我经常发现自己在写类似的东西

int computedValue = ...;
return MAX(0, MIN(5, computedValue));

我希望能够把它写成一个单行宏。它必须没有副作用,就像现有系统宏 MIN 和 MAX 一样,并且应该适用于与 MIN 和 MAX 相同的数据类型。

谁能告诉我如何把它变成一个宏?

4

5 回答 5

18

这没有副作用,适用于任何原始数字:

#define MIN(A,B)    ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __a : __b; })
#define MAX(A,B)    ({ __typeof__(A) __a = (A); __typeof__(B) __b = (B); __a < __b ? __b : __a; })

#define CLAMP(x, low, high) ({\
  __typeof__(x) __x = (x); \
  __typeof__(low) __low = (low);\
  __typeof__(high) __high = (high);\
  __x > __high ? __high : (__x < __low ? __low : __x);\
  })

可以这样使用

int clampedInt = CLAMP(computedValue, 3, 7);
double clampedDouble = CLAMP(computedValue, 0.5, 1.0);

其他建议的名称而不是CLAMP可以是VALUE_CONSTRAINED_LOW_HIGH, BOUNDS, CLIPPED.

于 2013-02-08T10:17:32.087 回答
3

取自该站点http://developer.gnome.org/glib/2.34/glib-Standard-Macros.html#CLAMP:CAPS

#define CLAMP(x, low, high)  (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x)))
于 2013-02-08T09:58:01.860 回答
2

也许你想这样尝试:

template <class T> 
const T& clamp(const T& value, const T& low, const T& high) {
    return value < low ? low:
           value > high? high:
                         value;
}
于 2013-02-08T09:46:57.860 回答
1
 #define MAX(a, b) (((a) > (b)) ? (a) : (b)) 
 #define MIN(a, b) (((a) > (b)) ? (b) : (a))

将它放在一个#define 指令中不会很容易阅读。

于 2013-02-08T09:56:45.113 回答
-1

仅使用一个比较操作:

static inline int clamp(int value, int min, int max) {
    return min + MIN((unsigned int)(value - min), max - min)
}
于 2013-02-08T09:55:28.397 回答