宏很好。 模板很好。 几乎任何它的工作都很好。
示例是OpenGL;但该技术是C ++特有的,并且不依赖于OpenGL的知识。
精确问题:
我想要一个表达式E;我不需要指定唯一名称;这样就可以在定义了E的地方调用构造函数,并在块E的末尾调用析构函数。
例如,考虑:
class GlTranslate {
GLTranslate(float x, float y, float z); {
glPushMatrix();
glTranslatef(x, y, z);
}
~GlTranslate() { glPopMatrix(); }
};
手动解决方案:
{
GlTranslate foo(1.0, 0.0, 0.0); // I had to give it a name
.....
} // auto popmatrix
现在,我不仅有glTranslate,还有很多其他的PushAttrib / PopAttrib调用。我宁愿不必为每个var提出一个唯一的名称。是否存在涉及宏模板的一些技巧......或者其他会自动创建变量的变量,在定义点调用构造函数;和块结束时调用的析构函数?
谢谢!
答案 0 :(得分:61)
我个人不会这样做,只是想出独特的名字。但是如果你想这样做,一种方法是使用if
和for
的组合:
#define FOR_BLOCK(DECL) if(bool _c_ = false) ; else for(DECL;!_c_;_c_=true)
您可以像
一样使用它FOR_BLOCK(GlTranslate t(1.0, 0.0, 0.0)) {
FOR_BLOCK(GlTranslate t(1.0, 1.0, 0.0)) {
...
}
}
每个名称都在不同的范围内,不会发生冲突。内部名称隐藏外部名称。 if
和for
循环中的表达式是常量,应该可以由编译器轻松优化。
如果你真的想要传递一个表达式,你可以使用ScopedGuard技巧(参见Most Important const
),但是需要更多的工作来编写它。但好的一面是,我们可以摆脱for
循环,让我们的对象评估为false
:
struct sbase {
operator bool() const { return false; }
};
template<typename T>
struct scont : sbase {
scont(T const& t):t(t), dismiss() {
t.enter();
}
scont(scont const&o):t(o.t), dismiss() {
o.dismiss = true;
}
~scont() { if(!dismiss) t.leave(); }
T t;
mutable bool dismiss;
};
template<typename T>
scont<T> make_scont(T const&t) { return scont<T>(t); }
#define FOR_BLOCK(E) if(sbase const& _b_ = make_scont(E)) ; else
然后,您提供正确的enter
和leave
功能:
struct GlTranslate {
GLTranslate(float x, float y, float z)
:x(x),y(y),z(z) { }
void enter() const {
glPushMatrix();
glTranslatef(x, y, z);
}
void leave() const {
glPopMatrix();
}
float x, y, z;
};
现在你可以在用户端没有名字的情况下完全编写它:
FOR_BLOCK(GlTranslate(1.0, 0.0, 0.0)) {
FOR_BLOCK(GlTranslate(1.0, 1.0, 0.0)) {
...
}
}
如果你想一次传递多个表达式,这有点棘手,但你可以编写一个表达式模板,作用于operator,
以将所有表达式收集到scont
中。
template<typename Derived>
struct scoped_obj {
void enter() const { }
void leave() const { }
Derived const& get_obj() const {
return static_cast<Derived const&>(*this);
}
};
template<typename L, typename R> struct collect
: scoped_obj< collect<L, R> > {
L l;
R r;
collect(L const& l, R const& r)
:l(l), r(r) { }
void enter() const { l.enter(); r.enter(); }
void leave() const { r.leave(); l.leave(); }
};
template<typename D1, typename D2>
collect<D1, D2> operator,(scoped_obj<D1> const& l, scoped_obj<D2> const& r) {
return collect<D1, D2>(l.get_obj(), r.get_obj());
}
#define FOR_BLOCK(E) if(sbase const& _b_ = make_scont((E))) ; else
您需要从scoped_obj<Class>
继承RAII对象,如下面的节目
struct GLTranslate : scoped_obj<GLTranslate> {
GLTranslate(float x, float y, float z)
:x(x),y(y),z(z) { }
void enter() const {
std::cout << "entering ("
<< x << " " << y << " " << z << ")"
<< std::endl;
}
void leave() const {
std::cout << "leaving ("
<< x << " " << y << " " << z << ")"
<< std::endl;
}
float x, y, z;
};
int main() {
// if more than one element is passed, wrap them in parentheses
FOR_BLOCK((GLTranslate(10, 20, 30), GLTranslate(40, 50, 60))) {
std::cout << "in block..." << std::endl;
}
}
所有这些都不涉及虚函数,所涉及的函数对编译器是透明的。实际上,上面的GLTranslate
更改为将一个整数添加到全局变量中,当再次减去它时,以及下面定义的GLTranslateE
时,我做了一个测试:
// we will change this and see how the compiler reacts.
int j = 0;
// only add, don't subtract again
struct GLTranslateE : scoped_obj< GLTranslateE > {
GLTranslateE(int x):x(x) { }
void enter() const {
j += x;
}
int x;
};
int main() {
FOR_BLOCK((GLTranslate(10), GLTranslateE(5))) {
/* empty */
}
return j;
}
事实上,优化级别-O2
的GCC会输出:
main:
sub $29, $29, 8
ldw $2, $0, j
add $2, $2, 5
stw $2, $0, j
.L1:
add $29, $29, 8
jr $31
我没想到,它优化得很好!
答案 1 :(得分:36)
如果您的编译器支持__COUNTER__
(它可能会支持),您可以尝试:
// boiler-plate
#define CONCATENATE_DETAIL(x, y) x##y
#define CONCATENATE(x, y) CONCATENATE_DETAIL(x, y)
#define MAKE_UNIQUE(x) CONCATENATE(x, __COUNTER__)
// per-transform type
#define GL_TRANSLATE_DETAIL(n, x, y, z) GlTranslate n(x, y, z)
#define GL_TRANSLATE(x, y, z) GL_TRANSLATE_DETAIL(MAKE_UNIQUE(_trans_), x, y, z)
有关
{
GL_TRANSLATE(1.0, 0.0, 0.0);
// becomes something like:
GlTranslate _trans_1(1.0, 0.0, 0.0);
} // auto popmatrix
答案 2 :(得分:10)
我认为现在可以这样做:
struct GlTranslate
{
operator()(double x,double y,double z, std::function<void()> f)
{
glPushMatrix(); glTranslatef(x, y, z);
f();
glPopMatrix();
}
};
然后在代码中
GlTranslate(x, y, z,[&]()
{
// your code goes here
});
显然,需要C ++ 11
答案 3 :(得分:0)
一个答案中描述的规范方法是使用lambda表达式作为块,在C ++中,您可以轻松编写模板函数
with<T>(T instance, const std::function<void(T)> &f) {
f(instance);
}
并像使用它
with(GLTranslate(...), [] (auto translate) {
....
});
但是,想要一种避免在您的范围内定义名称的机制的最常见原因是长函数/方法,它们可以完成很多事情。如果这种问题一直困扰着您,您可以尝试使用非常短的方法/函数尝试一种现代的OOP /干净代码启发式的样式,如果这种问题持续困扰您?