値の相対的な分布を維持しながら、数値のコレクションをある範囲から別の範囲に変換する必要があります。
たとえば、ランダムに生成されたfloatを含むベクトルは、可能なunsigned char値(0..255)に収まるようにスケーリングできます。型変換を無視すると、これは、提供された入力が何であれ(たとえば、-1.0から1.0)、すべての数値が0.0から255.0(またはその前後)にスケーリングされることを意味します。
この変換を実行するためのテンプレートクラスを作成しました。これは、次を使用してコレクションに適用できますstd::transform
。
template <class TYPE>
class scale_value {
const TYPE fmin, tmin, ratio;
public:
TYPE operator()(const TYPE& v) {
TYPE vv(v);
vv += (TYPE(0) - fmin); // offset according to input minimum
vv *= ratio; // apply the scaling factor
vv -= (TYPE(0) - tmin); // offset according to output minimum
return vv;
}
// constructor takes input min,max and output min,max
scale_value(const TYPE& pfmin, const TYPE& pfmax, const TYPE& ptmin, const TYPE& ptmax)
: fmin(pfmin), tmin(ptmin), ratio((ptmax-tmin)/(pfmax-fmin)) { }
// some code removed for brevity
};
ただし、上記のコードは実数(、、 ...)に対してのみ正しく機能しfloat
ますdouble
。整数はスケールアップ時に機能しますが、それでも全体の比率によってのみ機能します。
float scale_test_float[] = {0.0, 0.5, 1.0, 1.5, 2.0};
int scale_test_int[] = {0, 5, 10, 15, 20};
// create up-scalers
scale_value<float> scale_up_float(0.0, 2.0, 100.0, 200.0);
scale_value<int> scale_up_int(0, 20, 100, 200);
// create down-scalers
scale_value<float> scale_down_float(100.0, 200.0, 0.0, 2.0);
scale_value<int> scale_down_int(100, 200, 0, 20);
std::transform(scale_test_float, scale_test_float+5, scale_test_float, scale_up_float);
// scale_test_float -> 100.0, 125.0, 150.0, 175.0, 200.0
std::transform(scale_test_int, scale_test_int+5, scale_test_int, scale_up_int);
// scale_test_int -> 100, 125, 150, 175, 200
std::transform(scale_test_float, scale_test_float+5, scale_test_float, scale_down_float);
// scale_test_float -> 0.0, 0.5, 1.0, 1.5, 2.0
std::transform(scale_test_int, scale_test_int+5, scale_test_int, scale_down_int);
// scale_test_int -> 0, 0, 0, 0, 0 : fails due to ratio being rounded to 0
この問題に対する私の現在の解決策は、内部のすべてをscale_value
として保存しdouble
、必要に応じて型変換を使用することです。
TYPE operator()(const TYPE& v) {
double vv(static_cast<double>(v));
vv += (0.0 - fmin); // offset according to input minimum
vv *= ratio; // apply the scaling factor
vv -= (0.0 - tmin); // offset according to output minimum
return static_cast<TYPE>(vv);
}
これは、値が丸められるのではなく切り捨てられるため、整数のエラーはありますが、ほとんどの場合に機能します。たとえば、からにスケーリングして{0,5,10,15,20}
から戻るにスケーリング0..20
すると、が得られます。20..35
{0,4,9,14,20}
だから、私の質問は、これを行うためのより良い方法はありますか?sのコレクションをスケーリングする場合float
、型変換はかなり冗長に見えますが、sをスケーリングする場合、int
切り捨てが原因でエラーが発生します。
余談ですが、この目的のためにブーストで何か(少なくとも明らかなものは何もありません)を見つけられなかったことに驚きました。多分私はそれを逃しました-さまざまな数学ライブラリが私を混乱させます。
編集:特定のタイプに特化できることはわかっoperator()
ていますが、これは多くのコード重複を意味し、テンプレートの便利な部分の1つを無効にします。たとえば、すべての非浮動小数点型(short、int、uint、...)に1回特化する方法がない限り。