0

前言:我发现一个答案被标记为重复而没有实际检查它是否解决了一个人的问题,这很烦人。我以前问过这个问题,但没有成功。特别是,在 C++ 表达式模板类中实现 Matlab 的冒号:运算符没有回答这个问题。它是固定数量的变量与可变数量的变量+整数范围和整数值的可变组合。如果您仍然认为正确的方向,那么请为我陈述的问题提供有效的解决方案,或者如果您不能这样做,请不要将其简单地标记为重复。

=> 在你点击那个重复按钮之前,请检查它是否回答了我的问题。谢谢。

我试图找到最优雅的方式来定义位掩码。此位掩码是一个整数,它定义了地图上对象的可见性(32 个级别,因此位 0..31 定义了 32 个级别中每个级别的可见性)。bitmaks 创建助手应该能够处理可变长度的整数列表,以及整数范围 - 以及这些的任何组合。

例如,它可能看起来像:

int visibilityMask = CreateVisibilityMask([1..12], 16, 22);

我想这个真的很难。但这不可能吗?

4

4 回答 4

1

根据 Sarien 的回答:

class MaskCreator
{
  public:
    MaskCreator& AddRange(int from,int to){
      for(int i= from; i<=to; ++i){
        m_list.push_back(i);
      }
      return *this;
    }

    MaskCreator& Add(int i){
      m_list.push_back(i);
      return *this;
    }

    MaskCreator& AddMulti( varargstuff ){
      m_list.push_back(i);
      return *this;
    }

    unsigned int GetMask();

  private:
    vector<int> m_list;
}

// usage:
   unsigned int mask = MaskCreator().Add(3).Add(7).AddRange(16,25).AddMulti(28,30,31).GetMask();

显然 AddMulti 可以代替 Add;

于 2013-05-03T14:47:15.317 回答
0

使它成为 VisibilityMask(Range(1, 12)).Add(16).Add(22) ,它会很容易。

您只需要为 Range 对象(或两个 int)和一个 int 提供一个带有重载的构造函数和一个 Add 函数。如果您返回当前对象,您可以像这样链接调用。

如果你想要更漂亮的语法,你也可以重载 operator<< 或 operator()。但我不认为这是一个好主意。我只会重载那些用于输出 (<<) 和仿函数 (()) 以使您的代码更具可读性。

您可能会超载逗号运算符或一些此类恶作剧,但我不推荐它。坚持使用惯用的 C++,即使这意味着 Range(1, 12) 现在比 [1..12] 多了 4 个字符。您也不应该使用可变参数函数(参见此处)。

于 2013-05-03T14:10:57.337 回答
0

可能是通过使用可变参数我不经常使用它们可以是一个可变参数列表包含向量吗?如果是的话,你可以做这样的事情:

std::vector<int> SingleIndices( int vararglist)
{
  std::vector<int> vec;
  /// push all ints 
  return vec;
}

std::vector<int> Range(int from, int to)
{
  std::vector<int> vec;
  for(int i= from; i<=to; ++i){
    vec.push_back(i);
  }
  return vec;
}


int CreateVisibilityMask( vectorvarargs );

//usage:
unsigned int mask = CreateVisibilityMask(Range(1,5),SingleIndices(8,9,23), Range(25, 30));

无论如何,这似乎过于工程化并且效率不高。

于 2013-05-03T14:38:15.020 回答
0

我想到的另一个不错的选择是简单地使用以下形式的 var-arg int 列表:

int mask = getBitMask(1, 4, 6,-10, 15, 20,-24);

然后你处理 args 并找到一个负数,那么这意味着最后一个数字和当前数字的绝对值之间的所有索引形成一个范围。

唯一的缺点是,没有语义编译时检查来禁止类似的事情

int mask = getBitMask(-2, 4, 6); // allthough this may be valid and just mean bit 0 to 2

或者

int mask = getBitMask(4, -6, -10); 
于 2013-05-03T17:06:59.020 回答