0

为了学习 C++ 中的复合赋值,我创建了以下代码来演示它们的作用:

int b05;
int b06 = 13;

b05 = 49;
b05 += b06; // b05 = b05 + b06
cout << "(+=) compound assignment: " << b05 << endl;

b05 = 49;
b05 -= b06; // b05 = b05 - b06
cout << "(-=) compound assignment: " << b05 << endl;

b05 = 49;
b05 *= b06; // b05 = b05 * b06
cout << "(*=) compound assignment: " << b05 << endl;

b05 = 49;
b05 /= b06; // b05 = b05 / b06
cout << "(/=) compound assignment: " << b05 << endl;

b05 = 49;
b05 %= b06; // b05 = b05 % b06
cout << "(%=) compound assignment: " << b05 << endl;

b05 = 49;
b05 >>= b06; // b05 = b05 >> b06
cout << "(>>=) compound assignment: " << b05 << endl;

b05 = 49;
b05 <<= b06; // b05 = b05 << b06
cout << "(<<=) compound assignment: " << b05 << endl;

b05 = 49;
b05 &= b06; // b05 = b05 & b06
cout << "(&=) compound assignment: " << b05 << endl;

b05 = 49;
b05 ^= b06; // b05 = b05 ^ b06
cout << "(^=) compound assignment: " << b05 << endl;

b05 = 49;
b05 |= b06; // b05 = b05 | b06
cout << "(|=) compound assignment: " << b05 << endl;

如您所见,我必须重新分配 49 的值,b05因为前面的操作修改了这些值。

有没有办法绕过这样做?还是有更有效的方法来实现相同的输出?(我很感激一个代码示例)

4

4 回答 4

2

你可以用宏做这样的事情:

#define compound(op) \
    b05 = 49; \
    b05 op b06; \
    std::cout << "(" << #op << ") compound assignment: " << b05 << std::endl;

然后你这样称呼它:

int b05, b06 = 13;

compound(+=)
compound(/=)
// etc...

这实际上是在编译时进行文本替换。在您拥有 的每个地方compound(...),它将被复合宏的文本op替换,并被您在括号中提供的任何内容替换(在这种情况下,是某种运算符)。

要查看此操作,请执行此操作,g++ -E <codefile>您将看到宏(以及任何包含)已被扩展。

于 2016-08-08T22:57:22.423 回答
2

首先,没有特别的理由将变量命名为b05and b06,所以让我们使用aand b,它看起来不那么难看。

从根本上说,您想要复制您的一对值(两个整数4913)并将副本传递给将更改它们的代码。

最简单的解决方案是创建 2 个额外的变量,将值分配给它们:

int a0 = 49, b0 = 13;

然后每次你想使用它们时,你可以复制分配:

a = a0; b = b0;

这将避免重复常量(您只指定一次) 4913但是您仍然必须重复复制变量的运算符。

任何进一步的改进都不会让你避免这个副本,它仍然必须在每次操作之前完成,但你可以通过一些技巧来避免一些源代码重复:

(1) 使用结构。这将让您将两个值封装在一个中:

struct params {
  int a;
  int b;
}

params p0;
p0.a = 49;
p0.b = 13;

params p;

...
p = p0;
p.a += p.b; // p.a = p.a + p.b
cout << "(+=) compound assignment: " << p.b << endl;
...

现在你只重复一个命令p = p0而不是两个。

(2) 使用值参数:

void test1(int a, int b) {
  a += b; cout << "(+=) compound assignment: " << a << endl;
}
void test2(int a, int b) {
  a -= b; cout << "(-=) compound assignment: " << a << endl;
}
...

int main() {
  int a = 49; int b = 13;
  test1(a, b);
  test2(a, b);
  ...
  return 0;
}

虽然没有显式分配 a 和 b,但每次复制仍然发生:实际参数abfrommain被复制到函数的形式参数变量中,巧合的是,也命名为aand b

(3) 使用函数引用可以避免重复代码。这是每个操作的匿名函数数组的示例:

typedef int OperationFn(int, int)

struct OperationInfo {
  std::string label;
  OperationFn op;
}

OperationInfo operations[9] = {
  { "+=", [](int a, int b) { return a += b } },
  { "-=", [](int a, int b) { return a -= b } },
  { "*=", [](int a, int b) { return a *= b } },
  { "/=", [](int a, int b) { return a /= b } },
  { "%=", [](int a, int b) { return a %= b } },
  { ">>=", [](int a, int b) { return a >>= b } },
  { "<<=", [](int a, int b) { return a <<= b } },
  { "&=", [](int a, int b) { return a &= b } },
  { "^=", [](int a, int b) { return a ^= b } },
}

int main() {
  for (int i = 0; i < 9; i++) {
    const OperationInfo& oi = operations[i];
    cout << "(" << oi.label << ") compound assignment: ";
    cout << oi.op(49, 13) << endl;
  }
  return 0;
}

代码只提到了一次参数替换(oi.op(49, 13)行),但它是在一个循环中,所以它会有效地执行所有 9 次。虽然代码看起来更简洁、更紧凑,但并没有提高效率:您仍然在每次测试之前复制这两个值。

但是可以进行一种优化:您可以对第二个参数使用共享的 const 值,因为它从未被修改过:

const int b = 13;

然后不要b绕过,只需使用b.

注意:请原谅我的错别字。我没有方便的编译器,所以我无法验证这段代码。它可能有一些小错误。如果你找到了请留下评论,我会更新我的帖子。

于 2016-08-08T23:24:49.697 回答
1

您误解了复合操作。它们只影响表达式的左值。

#include <iostream>

int main() {
    int b05 = 10; int b06 = 5;
    b05 |= b06;
    std::cout << "b05 " << b05 << ", b06 " << b06 << "\n";
}

http://ideone.com/5nT7pR

输出:

b05 15, b06 5

b06 未修改。

于 2016-08-08T22:57:19.470 回答
0

也许你可以做一个这样的功能:

class Obj
{
public:
    Obj& operator+= (const Obj &) { return *this; }
    Obj& operator-= (const Obj &) { return *this; }
};

void testFunction(std::function<Obj&(Obj *, const Obj &)> functionToTest)
{
  Object object;
  Object otherObject;
  functionToTest(&object, otherObject);
  cout << "(+=) function test: " << object << endl;
}

int main(void)
{
  std::function<Obj&(Obj *, const Obj &)> functionToTest(&Obj::operator+=);
  testFunction(functionToTest);

  std::function<Obj&(Obj *, const Obj &)> functionToTest2(&Obj::operator-=);
  testFunction(functionToTest2);
}

这段代码不是 100% 正确的,但它应该给你的想法和原则是合理的。当然,它不适用于原始类型。

于 2016-08-08T23:34:15.667 回答