在访问者上下文中,我需要在访问 child 之前临时设置一个变量,然后再恢复该变量。我正在使用以下代码,但我确信有一种更优雅、更正确的方法可以做到这一点:
template <typename TYPE> class TemporaryAssignment {
protected:
TYPE& mVariable;
TYPE mOriginalValue;
public:
TemporaryAssignment(TYPE& inVariable, TYPE inValue)
: mVariable(inVariable), mOriginalValue(inVariable) {
mVariable = inValue;
}
~TemporaryAssignment(void) {
mVariable = mOriginalValue;
}
};
这使我可以编写如下内容:
{
...
TemporaryAssignment<string> t(myVariable, myTemporaryValue);
visitChildren();
...
}
// previous value of myVariable is restored
当临时赋值对象超出范围时,变量将恢复到它以前的值。执行此操作的更好方法是什么?
最佳答案
除了析构函数可以抛出外,在我看来还不错,这很糟糕。 swap 与原始值而不是赋值(编辑:这涉及 std::string,但请参阅注释以了解可能存在的问题,这些类的用户友好性低于 字符串)。
如果您从这部分代码中退一步,也许您可以找到一种根本不需要设置临时值的方法。对象中共享可变状态的坏处与可变全局变量坏处的原因相同,但程度较轻,因为它只会弄乱你的类,而不是弄乱你的整个程序。
例如,也许您可以复制整个对象,为变量设置一个新值并访问拷贝而不是访问自己。显然,这不一定可行或有效,您必须根据具体情况寻找替代方案。也许就子项而言,拷贝可能很浅(即引用相同的子对象),这可能足以使其便宜。
关于用法,您可以像这样推断类型(未经测试的代码):
template <typename T, typename ARG>
TemporaryAssignment<T> temp_value(T &var, ARG &&newvalue) {
return TemporaryAssignment(var, std::forward<ARG>(newValue));
}
用法:
auto t = temp_value(myVariable, myTemporaryValue);
然后您需要一个用于 TemporaryAssignment 的移动构造函数:
template <typename TYPE> class TemporaryAssignment {
// change data member
TYPE *mVariable;
TYPE mOriginalValue;
public:
TemporaryAssignment(TYPE &inVariable, TYPE inValue)
: mVariable(&inVariable), mOriginalValue(std::move(inVariable)) {
*mVariable = std::move(inValue);
}
TemporaryAssignment(TemporaryAssignment &&rhs) {
mOriginalValue = std::move(rhs.mOriginalValue);
mVariable = rhs.mVariable;
rhs.mVariable = 0;
}
~TypeAssignment() {
using std::swap;
if (mVariable) {
swap(*mVariable, mOriginalValue);
}
}
// can't remember whether this is needed
TemporaryAssignment(const TemporaryAssignment &) = delete;
TemporaryAssignment &operator=(const TemporaryAssignment &) = delete;
TemporaryAssignment &operator=(TemporaryAssignment &&) = delete;
};
我想过为 TemporaryAssignment 指定 operator= 以使用法看起来像一个赋值,但我没有想出任何好办法。
auto t = (temporary(myVariable) = myTemporaryValue);
是合理的,但您可能不希望 TemporaryAssignment 定义 operator=,因为以下含义:
t = otherTemporaryValue;
不一定清楚,可能不应该被允许。也许从 temporary 返回第二个类,它从它的 operator= 返回 TemporaryAssignment。
关于c++ - 在 C++ 中恢复临时赋值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13383683/
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:
我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我
我明白了:x,(y,z)=1,*[2,3]x#=>1y#=>2z#=>nil我想知道为什么z的值为nil。 最佳答案 x,(y,z)=1,*[2,3]右侧的splat*是内联扩展的,所以它等同于:x,(y,z)=1,2,3左边带括号的列表被视为嵌套赋值,所以它等价于:x=1y,z=23被丢弃,而z被分配给nil。 关于ruby-带括号和splat运算符的并行赋值,我们在StackOverflow上找到一个类似的问题: https://stackoverflow
我最近与一位同事讨论了以下Ruby语法:value=ifa==0"foo"elsifa>42"bar"else"fizz"end我个人并没有看到太多这种逻辑,但我的同事指出,这实际上是一种相当普遍的Rubyism。我试着用谷歌搜索这个主题,但没有找到任何文章、页面或SO问题来讨论它,这让我相信这可能是一种非常实际的技术。然而,另一位同事发现语法令人困惑,而是将上面的逻辑写成这样:ifa==0value="foo"elsifa>42value="bar"elsevalue="fizz"end缺点是value=的重复声明和隐式elsenil的丢失,如果我们想使用它的话。这也感觉它与Ruby
我有一个数组数组,想将元素附加到子数组。+=做我想做的,但我想了解为什么push不做。我期望的行为(并与+=一起工作):b=Array.new(3,[])b[0]+=["apple"]b[1]+=["orange"]b[2]+=["frog"]b=>[["苹果"],["橙子"],["Frog"]]通过推送,我将推送的元素附加到每个子数组(为什么?):a=Array.new(3,[])a[0].push("apple")a[1].push("orange")a[2].push("frog")a=>[[“苹果”、“橙子”、“Frog”]、[“苹果”、“橙子”、“Frog”]、[“苹果”、“
有没有办法让Ruby能够做这样的事情?classPlane@moved=0@x=0defx+=(v)#thisiserror@x+=v@moved+=1enddefto_s"moved#{@moved}times,currentxis#{@x}"endendplane=Plane.newplane.x+=5plane.x+=10putsplane.to_s#moved2times,currentxis15 最佳答案 您不能在Ruby中覆盖复合赋值运算符。任务在内部处理。您应该覆盖+,而不是+=。plane.a+=b与plane.a=
出于某种原因,heroku尝试要求dm-sqlite-adapter,即使它应该在这里使用Postgres。请注意,这发生在我打开任何URL时-而不是在gitpush本身期间。我构建了一个默认的Facebook应用程序。gem文件:source:gemcuttergem"foreman"gem"sinatra"gem"mogli"gem"json"gem"httparty"gem"thin"gem"data_mapper"gem"heroku"group:productiondogem"pg"gem"dm-postgres-adapter"endgroup:development,:t
我有一个rspec模拟对象,一个值赋给了属性。我正在努力在我的rspec测试中满足这种期望。只是想知道语法是什么?代码:defcreate@new_campaign=AdCampaign.new(params[:new_campaign])@new_campaign.creationDate="#{Time.now.year}/#{Time.now.mon}/#{Time.now.day}"if@new_campaign.saveflash[:status]="Success"elseflash[:status]="Failed"endend测试it"shouldabletocreat
我是Ruby和这个网站的新手。下面两个函数是不同的,一个在函数外修改变量,一个不修改。defm1(x)x我想确保我理解正确-当调用m1时,对str的引用被复制并传递给将其视为x的函数。运算符当调用m2时,对str的引用被复制并传递给将其视为x的函数。运算符+创建一个新字符串,赋值x=x+"4"只是将x重定向到新字符串,而原始str变量保持不变。对吧?谢谢 最佳答案 String#+::str+other_str→new_strConcatenation—ReturnsanewStringcontainingother_strconc