Как я могу написать макрос "зажим" / "клип" / "связанный" для возврата значения в заданном диапазоне?

Я часто пишу что-то вроде

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

Я хотел бы иметь возможность написать это в виде одного однострочного макроса. Он не должен иметь побочных эффектов, так же, как существующие системные макросы MIN и MAX, и должен работать для тех же типов данных, что и MIN и MAX.

Может кто-нибудь показать мне, как превратить это в один макрос?

5 ответов

Решение

Это без побочных эффектов и работает для любого примитивного числа:

#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,

Взято с этого сайта http://developer.gnome.org/glib/2.34/glib-Standard-Macros.html

#define CLAMP(x, low, high)  (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x)))

Может быть, вы хотите попробовать это так:

template <class T> 
const T& clamp(const T& value, const T& low, const T& high) {
    return value < low ? low:
           value > high? high:
                         value;
}
 #define MAX(a, b) (((a) > (b)) ? (a) : (b)) 
 #define MIN(a, b) (((a) > (b)) ? (b) : (a))

превращение в одну директиву #define не будет очень читабельным.

Используя только одну операцию сравнения:

static inline int clamp(int value, int min, int max) {
    return min + MIN((unsigned int)(value - min), max - min)
}
Другие вопросы по тегам