- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
下面的代码可以使用 g++ 4.8.1 (mingw) 以及 http://gcc.godbolt.org/ 上的各种最新 clang 和 gcc 版本正确编译。 ,但是对于 MSVC2013 更新 4,它失败了,显然是由于 typedef typename A<T>::value_type value_type;
线。编译器给出以下错误:
x.cpp(30): error C2893: Failed to specialize function template '
void B<C,int>::bar(void)
' With the following template arguments: 'MemberFn=void C::baz(int)
'
更简单的 typedef typedef T value_type;
有效。
我做错了什么吗?或者这是 Microsoft C++ 编译器中的已知错误?
补充问题:
typedef T value_type;
)还是从基类中提取类型(例如 typedef typename A<T>::value_type value_type;
,或 C++11 using typename A<T>::value_type;
)? (注意:我现在正在对冲 C++03 兼容性,因此避免了 using
。)这里有一些不确定的讨论:Use typedef/using from templated base class in derived class我问的原因是如果 typedef T value_type;
无论如何都是首选,我不需要那么担心。#include <cstdio>
template <typename T>
struct A {
typedef A<T> base_type;
typedef T value_type;
};
template <typename Derived, typename T>
struct B : public A<T> {
typedef Derived derived_type;
//typedef T value_type; // this works
typedef typename A<T>::value_type value_type; // this fails in MSVC 2013
//using typename A<T>::value_type; // this fails in MSVC 2013 too
template<void (derived_type::*MemberFn)(value_type) >
void bar()
{
(static_cast<derived_type*>(this)->*MemberFn)(42);
}
};
struct C : public B<C, int> {
void baz(int i)
{
std::printf("baz(%d)\n", i);
}
void foo()
{
bar<&C::baz>();
}
};
int main(int, char *[])
{
C c;
c.foo();
}
更新#1:这是一个来自框架的简化测试用例。我不是要求对结构进行一般性批评。我不希望这个结构在没有上下文的情况下有意义。
更新#2:这是一个相关问题,讨论是否 using
与 typename
组合有效: C++ template inheritance issue with base types
更新#3:我已在 Microsoft Connect 上提交了一份公开错误报告。如果您可以重现该问题,并认为这是一个错误,请对该错误点赞:https://connect.microsoft.com/VisualStudio/feedback/details/1740423
最佳答案
我有一个可能的解决方案(我只能用 VS2015 测试这个)
如果您使用基本类型作为模板参数本身,它会被解析并且您可以访问 value_type
。
template <typename T>
struct A {
typedef T value_type;
typedef A<T> base_type;
};
template <typename Derived, typename T, typename Base = A<T>>
struct B : public Base
{
typedef Derived derived_type;
typedef void (derived_type::*member_func_type)(typename Base::value_type);
template<member_func_type MemberFn>
void bar()
{
(static_cast<derived_type*>(this)->*MemberFn)(42);
}
};
struct C : public B<C, int>
{
void baz(int i)
{
std::printf("baz(%d)\n", i);
}
void foo()
{
bar<&C::baz>();
}
};
或者我更喜欢的解决方案:
template <typename T>
struct A {
typedef A<T> base_type;
typedef T value_type;
};
template <typename Derived, typename T, typename Base = A<T>>
struct B : public Base
{
typedef Derived derived_type;
typedef typename Base::base_type base_type;
typedef typename Base::value_type value_type;
typedef void (derived_type::*member_func_type)(value_type);
template<member_func_type MemberFn>
void bar()
{
(static_cast<derived_type*>(this)->*MemberFn)(42);
}
};
我必须承认这个解决方案有它的缺陷,比如用其他东西覆盖第三个模板参数的能力。至少其他基类需要声明 base_type
和 value_type
。
编辑:使用 static_assert
可以防止更改 Base
模板参数。
template <typename Derived, typename T, typename Base = A<T> >
struct B : public Base
{
static_assert(std::is_same<Base, typename A<T>>::value, "Redefinition of template parameter Base is not allowed");
typedef Derived derived_type;
typedef typename Base::base_type base_type;
typedef typename Base::value_type value_type;
typedef void (derived_type::*member_func_type)(value_type);
template<member_func_type MemberFn>
void bar()
{
(static_cast<derived_type*>(this)->*MemberFn)(42);
}
};
例子:
template <typename T>
struct D {
typedef D<T> base_type;
typedef T value_type;
};
struct E : public B<C, int, D<int>>
{
};
结果:
error C2338: Redefinition of template parameter Base is not allowed
更新:
更改 B
的模板参数的顺序会改变行为。这里原来的代码只有T
和Derived
的顺序改变了。
#include <cstdio>
template <typename T>
struct A {
typedef A<T> base_type;
typedef T value_type;
};
template <typename T, typename Derived>
struct B : public A<T> {
typedef Derived derived_type;
//typedef T value_type; // this works
typedef typename A<T>::value_type value_type; // this fails in MSVC 2013
//using typename A<T>::value_type; // this fails in MSVC 2013 too
template<void (derived_type::*MemberFn)(value_type) >
void bar()
{
(static_cast<derived_type*>(this)->*MemberFn)(42);
}
};
struct C : public B<int, C> {
void baz(int i)
{
std::printf("baz(%d)\n", i);
}
void foo()
{
bar<&C::baz>();
}
};
int main(int, char *[])
{
C c;
c.foo();
}
这可以编译并且工作正常。
我仍然不能完全确定这是一个错误。
更新
所以这似乎是 MSVC 中缺少的功能。 MSVC(直到 2015/14.0)似乎不支持“两阶段名称查找”。
Darran Rowe: VC hasn't implemented three C++98/03 features: two-phase name lookup, dynamic exception specifications, and export. Two-phase name lookup remains unimplemented in 2015, but it's on the compiler team's list of things to do, pending codebase modernization. Dynamic exception specifications also remain unimplemented (VC gives non-Standard semantics to throw() and ignores other forms), but they were deprecated in C++11 and nobody cares about them now that we have noexcept. It's unlikely that we'll ever implement them, and there's even been talk of removing them from C++17. Finally, export was removed in C++11.
来源C++11/14/17 Features In VS 2015 RTM
在 2012 年有一个请求该功能的错误,但它在没有评论的情况下被关闭:support two-phase name lookup - by Ivan Sorokin
所以看起来你在这里做的一切,但 MSVC 只是不支持 C++ 标准的这一部分。
关于c++ - 函数模板特化失败 : coding error or MSVC2013 bug?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32312733/
我在使用以下代码时遇到问题: function http_file_exists($url){ $f=fopen($url,"r"); if($f){ fclose($f); retu
我已经通过 Git 部署到 Azure 几个月了,没有出现重大问题,但现在我似乎遇到了一个无法克服的错误。 我创建了一个新的 Azure 网站,为正在开发的项目创建单独的预览链接。我在新站点上设置了
我已经通过flutter创建了一个App并完成了它,我想在flutter文档中阅读时进行部署。 我收到此错误: FAILURE: Build failed with an exception. * W
我在Windows 10中使用一些简单的Powershell代码遇到了这个奇怪的问题,我认为这可能是我做错了,但我不是Powershell的天才。 我有这个: $ix = [System.Net.Dn
我正在尝试使用 RapidJSON 解析从服务器接收到的数据。以下是收到的确切字符串: [ { "Node": "9478149a08f9", "Address": "172.17
我尝试为 ios 编译 OpenCV。我总是收到这些错误。我用不同版本的opencv试了一下,结果都是一样的。 我运行这个:python 平台/ios/build_framework.py ios_o
我在一台机器上做基本的发布/订阅,我的客户端是 StackExchange-Redis 的 C# 客户端,我在同一台机器上运行基于 Windows 的 Redis 服务器(服务器版本 2.8.4) 当
我有这段代码,但无法执行,请帮我解决这个问题 连接 connect_error) { die ("connection failed: " . $terhubung->connect_erro
我在 tomcat 上运行并由 maven 编译的 Web 应用程序给出了以下警告和错误。我可以在本地存储库中看到所有 JAR,但有人可以帮忙吗。 WARNING: Failed to scan JA
我正在 Windows 8 上使用 Android Studio 开发一个 android 应用程序,我正在使用一些 native 代码。突然间我无法编译我的 C 文件。当我运行 ndk-build
下面的代码对类和结构的成员进行序列化和反序列化。序列化工作正常,但我在尝试使用 oarch >> BOOST_SERIALIZATION_NVP(outObj); 反序列化时遇到了以下错误; 代码中是
如果我运行此命令“rspec ./spec/requests/api/v1/password_reset_request_spec.rb”,此文件中的所有测试都会通过。 但是,当我运行“rspec”时
我在尝试执行测试以使用 Protractor 上传文件时出错,我的代码是这个 it('it should be possible to upload a file', function() {
System.loadLibrary("nativefaceswap"); 当我运行我的应用程序时,我在 Android Studio 中发现了此类错误。在logcat中显示: java.lang.U
我希望有人能帮助我!使用任何方法或命令行的任何 SSL/HTTPS 调用均无效。 我在 Windows 10 中使用 Ubuntu Server 18.04 作为子系统。我的问题是昨天才开始出现的,因
通过删除这两个值将日期字段从 null=True 和 Blank=True 更改为 required 时,使用 db.alter 命令时遇到问题。 当以下行被注释掉时,迁移运行不会出现问题。
我第一次使用 Heroku 尝试创建应用程序(使用 SendGrid 的 Inbound Parse Webhook"和 Twilio SMS 通过电子邮件发送和接收 SMS 消息)。通过 Virtu
我正在将我的 swift 项目更新到 Xcode 7 上的 Swift 2.0。xcode 在构建项目时报告了以下错误: 命令/Applications/Xcode.app/Contents/Deve
在我的代码中,SSL 库函数 SSL_library_init() 没有按预期返回 1。我如何才能看到它返回了什么错误? 我在 SSL_library_init() 之后调用了 SSL_load_er
我正在尝试运行在以下链接中找到的答案: Asynchronously Load the Contents of a Div 但是当我这样做时,我会遇到我不太理解的错误。 我的代码: $(documen
我是一名优秀的程序员,十分优秀!