草庐IT

c++ - 未定义模板的隐式实例化 'class'

coder 2023-06-04 原文

在我的库中尝试为 const 和非 const 模板参数提供函数时,我遇到了一个奇怪的问题。以下源代码是一个最小的示例现象:

#include <iostream>


template<typename some_type>
struct some_meta_class;

template<>
struct some_meta_class<int>
{
    typedef void type;
};



template<typename some_type>
struct return_type
{
    typedef typename some_meta_class< some_type >::type test;

    typedef void type;
};



template<typename type>
typename return_type<type>::type foo( type & in )
{
    std::cout << "non-const" << std::endl;
}

template<typename type>
void foo( type const & in )
{
    std::cout << "const" << std::endl;
}


int main()
{
    int i;

    int const & ciref = i;
    foo(ciref);
}

我尝试为 foo 实现非 const 版本和 const 版本,但不幸的是,这段代码无法在 CLANG 3.0 和 gcc 4.6.3 上编译。

main.cpp:18:22: error: implicit instantiation of undefined template 'some_meta_class'

因此,出于某种原因,编译器希望将非 const 版本的 foo 用于 const int 引用。这显然会导致上面的错误,因为 some_meta_class 没有实现。奇怪的是,如果您进行以下更改之一,代码编译得很好并且可以工作:

  • 取消注释/删除非常量版本
  • 取消注释/删除 return_type::test 的 typedef

这个例子当然是简约和纯粹的学术。在我的库中,我遇到了这个问题,因为 const 和非 const 版本返回不同的类型。我通过使用部分特化的辅助类解决了这个问题。

但是为什么上面的例子会导致这种奇怪的行为呢?为什么编译器不想使用 const 版本有效且匹配更好的非 const 版本?

最佳答案

原因是执行函数调用解析的方式,以及模板参数推导和替换。

  1. 首先,执行名称查找。这为您提供了两个名称匹配的函数 foo() .

  2. 其次,执行类型推导:对于具有匹配名称的每个模板函数,编译器尝试推导将产生的函数模板参数一个可行的比赛。你得到的错误发生在这个阶段。

  3. 第三,重载分辨率进入游戏。这只是在执行了类型推导并且确定了用于解析调用的可行函数的签名之后,这是有道理的:编译器只有在发现所有候选人的确切签名。

您收到与非常量重载相关的错误并不是因为编译器选择它作为解决调用的最可行候选者(这将是第 3 步),而是因为编译器在实例化时产生了错误其返回类型以确定其签名,在第 2 步中。

不是很明显为什么这会导致错误,因为人们可能认为 SFINAE 适用(替换失败不是错误)。为了澄清这一点,我们可以考虑一个更简单的例子:

template<typename T> struct X { };

template<typename T> typename X<T>::type f(T&) { }  // 1
template<typename T> void f(T const&) { }           // 2

int main()
{
    int const i = 0;
    f(i); // Selects overload 2
}

在本例中,SFINAE 适用:在步骤 2 中,编译器将推导出 T对于上述两个重载中的每一个,并尝试确定它们的签名。在过载 1 的情况下,这会导致 替换失败:X<const int>没有定义任何 type (在 typedef 中没有 X)。然而,由于 SFINAE,编译器只是丢弃它,并发现重载 2 是可行的匹配。因此,它选择了它。

现在让我们稍微改变一下示例,以反射(reflect) 您的示例:

template<typename T> struct X { };

template<typename Y>
struct R { typedef typename X<Y>::type type; };

// Notice the small change from X<T> into R<T>!
template<typename T> typename R<T>::type f(T&) { }  // 1
template<typename T> void f(T const&) { }           // 2

int main()
{
    int const i = 0;
    f(i); // ERROR! Cannot instantiate R<int const>
}

改变的是重载 1 不再返回 X<T>::type ,而是R<T>::type .这又与X<T>::type 相同因为 typedef R 中的声明,所以人们可能期望它产生相同的结果。但是,在这种情况下,您会收到编译错误。为什么?

标准有答案(第 14.8.3/8 段):

If a substitution results in an invalid type or expression, type deduction fails. An invalid type or expression is one that would be ill-formed if written using the substituted arguments. [...] Only invalid types and expressions in the immediate context of the function type and its template parameter types can result in a deduction failure.

显然,第二个示例(以及您的示例)在嵌套上下文中生成错误,因此 SFINAE 不适用。我相信这回答了你的问题。

顺便说一句,有趣的是,自 C++03 以来发生了变化,更一般地说(第 14.8.2/2 段):

[...] If a substitution in a template parameter or in the function type of the function template results in an invalid type, type deduction fails. [...]

如果您对为什么事情发生变化的原因感到好奇, this paper 可能会给你一个想法。

关于c++ - 未定义模板的隐式实例化 'class',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14626033/

有关c++ - 未定义模板的隐式实例化 'class'的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - rails : "missing partial" when calling 'render' in RSpec test - 2

    我正在尝试测试是否存在表单。我是Rails新手。我的new.html.erb_spec.rb文件的内容是:require'spec_helper'describe"messages/new.html.erb"doit"shouldrendertheform"dorender'/messages/new.html.erb'reponse.shouldhave_form_putting_to(@message)with_submit_buttonendendView本身,new.html.erb,有代码:当我运行rspec时,它失败了:1)messages/new.html.erbshou

  3. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  4. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  5. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  6. ruby - 在 jRuby 中使用 'fork' 生成进程的替代方案? - 2

    在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',

  7. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  8. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  9. ruby 正则表达式 - 如何替换字符串中匹配项的第 n 个实例 - 2

    在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg

  10. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

随机推荐