我知道 sizeof 运算符不会评估其表达式参数来获得答案。但它不是模板的非扣除上下文之一。所以我想知道它如何与模板交互,特别是模板参数推导。例如,以下内容摘自 C++ 模板:完整指南:
template<typename T>
class IsClassT {
private:
typedef char One;
typedef struct { char a[2]; } Two;
template<typename C> static One test(int C::*);
template<typename C> static Two test(...);
public:
enum { Yes = sizeof(IsClassT<T>::test<T>(0)) == 1 };
enum { No = !Yes };
};
这个类型函数 决定了,正如它的名字所暗示的,模板参数是否是一个类类型。该机制本质上是以下条件测试:
sizeof(IsClassT<T>::test<T>(0)) == 1
但是请注意,函数模板参数是显式的(在本例中为 T)并且函数参数是 int 类型的计划 0 >,它不是 指向类 C 的 int 成员的指针 类型。在正常的函数模板参数推导中,当 T 确实是类类型并且函数参数只是一个 0 时,推导 static One test(int C::* ); 应该失败,因为在模板参数推导期间不允许隐式转换(0 用作空指针类型)并且(我猜?)SFINAE 应该启动并且重载解析会被选择
static Two test(...);
但是,由于整个表达式都包含在 sizeof 运算符中,因此似乎无需强制转换即可传递 0。
谁能澄清一下:
sizeof 运算符的非求值性质使传递 0 成功?和0 在此上下文中无关紧要,我们可以选择任何参数代替 0,例如 0.0、100 甚至用户定义的类型?结论:我在C++ Primer 中找到了关于函数模板显式参数的部分。我引用“正常转换适用于明确指定的参数”和“出于同样的原因,允许对参数进行正常转换
使用普通类型定义(§ 16.2.1,第 680 页),正常转换也适用
for arguments whose template type parameter is explicitly specified”。所以这个问题中的0实际上是隐式转换为指向成员的空指针(指针转换)。
最佳答案
模板参数推导是在实例化函数时完成的。这是作为函数重载的一部分完成的(以及此处不适用的其他上下文)。在 TAD 中,函数参数的类型用于推导模板参数,但不一定使用所有参数。这就是“非推导上下文”的来源。如果模板参数出现在函数签名内的非推导上下文中,则无法从实际参数推导它。
sizeof(T)实际上是 T 的非推导上下文,但它是如此明显以至于没有人愿意提及它。例如
template< int N> class A {};
template<typename T> void f(A<sizeof(T)>);
f(A<4>());
编译器不会选择具有sizeof(T)==4 的随机T .
现在您的示例实际上没有 sizeof在函数模板的参数列表中,因此“非推导上下文”是一个无关紧要的考虑因素。也就是说,了解“sizeof 不评估其表达式参数”的含义很重要。这意味着不计算表达式 value,但计算表达式 type。在您的示例中,IsClassT<T>::test<T>(0)不会在运行时调用,但其类型在编译时确定。
关于c++ - sizeof 运算符是否会导致模板参数推导发生?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33405720/
给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
这个问题在这里已经有了答案:Checktoseeifanarrayisalreadysorted?(8个答案)关闭9年前。我只是想知道是否有办法检查数组是否在增加?这是我的解决方案,但我正在寻找更漂亮的方法:n=-1@arr.flatten.each{|e|returnfalseife
我正在为一个项目制作一个简单的shell,我希望像在Bash中一样解析参数字符串。foobar"helloworld"fooz应该变成:["foo","bar","helloworld","fooz"]等等。到目前为止,我一直在使用CSV::parse_line,将列分隔符设置为""和.compact输出。问题是我现在必须选择是要支持单引号还是双引号。CSV不支持超过一个分隔符。Python有一个名为shlex的模块:>>>shlex.split("Test'helloworld'foo")['Test','helloworld','foo']>>>shlex.split('Test"
我不确定传递给方法的对象的类型是否正确。我可能会将一个字符串传递给一个只能处理整数的函数。某种运行时保证怎么样?我看不到比以下更好的选择:defsomeFixNumMangler(input)raise"wrongtype:integerrequired"unlessinput.class==FixNumother_stuffend有更好的选择吗? 最佳答案 使用Kernel#Integer在使用之前转换输入的方法。当无法以任何合理的方式将输入转换为整数时,它将引发ArgumentError。defmy_method(number)
请帮助我理解范围运算符...和..之间的区别,作为Ruby中使用的“触发器”。这是PragmaticProgrammersguidetoRuby中的一个示例:a=(11..20).collect{|i|(i%4==0)..(i%3==0)?i:nil}返回:[nil,12,nil,nil,nil,16,17,18,nil,20]还有:a=(11..20).collect{|i|(i%4==0)...(i%3==0)?i:nil}返回:[nil,12,13,14,15,16,17,18,nil,20] 最佳答案 触发器(又名f/f)是
两者都可以defsetup(options={})options.reverse_merge:size=>25,:velocity=>10end和defsetup(options={}){:size=>25,:velocity=>10}.merge(options)end在方法的参数中分配默认值。问题是:哪个更好?您更愿意使用哪一个?在性能、代码可读性或其他方面有什么不同吗?编辑:我无意中添加了bang(!)...并不是要询问nobang方法与bang方法之间的区别 最佳答案 我倾向于使用reverse_merge方法:option
我有一个包含多个键的散列和一个字符串,该字符串不包含散列中的任何键或包含一个键。h={"k1"=>"v1","k2"=>"v2","k3"=>"v3"}s="thisisanexamplestringthatmightoccurwithakeysomewhereinthestringk1(withspecialcharacterslike(^&*$#@!^&&*))"检查s是否包含h中的任何键的最佳方法是什么,如果包含,则返回它包含的键的值?例如,对于上面的h和s的例子,输出应该是v1。编辑:只有字符串是用户定义的。哈希将始终相同。 最佳答案