- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
鉴于以下内容(精简并设法突出问题):
#include <utility>
namespace Generic
{
class Whatever
{
public:
// ISSUE1 (member "swap()" has same name as non-member template)
void swap()
{
}
};
// Forward declaration of class template "Test"
template <class>
class Test;
// Prototype of function template "swap()" for class template "Test"
template<class T>
void swap(Test<T> &, Test<T> &);
/////////////////////////////////////////////////////////////////
// Class template "Test"
/////////////////////////////////////////////////////////////////
template <class T>
class Test : public T
{
public:
// Default constructor
Test() = default;
// Move constructor
Test(Test &&test)
: BaseClass(std::move(test))
{
}
// Assignment operator
Test& operator=(Test test)
{
// Note that the "Generic:" prefix here has no impact on the results
Generic::swap(*this, test);
return (*this);
}
private:
using BaseClass = T;
// ISSUE2 (qualified call - prefixing with "Generic:")
friend void Generic::swap<T>(Test &, Test &);
};
/////////////////////////////////////////////////////////////////
// Canonical swap function for class "Test" just above
/////////////////////////////////////////////////////////////////
template<class T>
void swap(Test<T> &, Test<T> &)
{
}
} // namespace Generic
int main()
{
using namespace Generic;
Test<Whatever> test1;
Test<Whatever> test2;
test2 = std::move(test1);
return 0;
}
谁能看出它不应该编译的任何原因。请特别注意评论中的 ISSUE1 和 ISSUE2。这些都会影响结果。 ISSUE1 和 ISSUE2 有 4 种组合来尝试让事情正常进行(见下表),尽管我知道另一种纠正方法,但这 4 种组合是我想要关注的。请注意,所有 4 种组合都可以在 GCC 和 Clang 中干净地编译,但只有第 4 项可以在 Visual Studio 2017(最新版本)中干净地编译。其他组合(1 到 3)失败并在下面的 RESULT 列中看到错误(编译器在所有 3 种情况下都引用了 ISSUE2 行的问题):
ISSUE1 ISSUE2 RESULT
------ ------ ------
1) Leave unchanged Leave unchanged 'Generic::Test': use of class template requires template argument list
'Generic::swap': not a function
2) Leave unchanged Remove "Generic:" qualifier syntax error: missing ';' before '<'
'swap': illegal use of type 'void'
'Generic::swap': 'friend' not permitted on data declarations
'Generic::swap': redefinition; previous definition was 'function'
unexpected token(s) preceding ';'
3) Comment out member "swap()" Leave unchanged 'Generic::Test': use of class template requires template argument list
'Generic::swap': not a function
4) Comment out member "swap()" Remove "Generic:" qualifier Works!!!
是 Visual Studio 有问题还是 GCC 和 Clang 有问题(根据我对合格/不合格名称查找规则、ADL 等的理解,前者似乎是这种情况)。谢谢。
最佳答案
呵呵,找到博客了: Two-phase name lookup support comes to MSVC .
一些相关摘录:
The original design of templates for C++ meant to do exactly what the term “template” implied: a template would stamp out families of classes and functions. It allowed and encouraged, but did not require, early checking of non-dependent names. Consequently, identifiers didn’t need to be looked up during parsing of the template definition. Instead, compilers were allowed to delay name lookup until the template was instantiated. Similarly, the syntax of a template didn’t need to be validated until instantiation. Essentially, the meaning of a name used in a template was not determined until the template was instantiated.
In accordance with these original rules, previous versions of MSVC did very limited template parsing. In particular, function template bodies were not parsed at all until instantiation. The compiler recorded the body of a template as a stream of tokens that was replayed when it was needed during instantiation of a template where it might be a candidate.
文章然后继续发布一些与n.m.’s minimal example 非常相似的代码.
It’s important to note that overloads declared after the point of the template’s definition but before the point of the template’s instantiation are only considered if they are found through argument-dependent lookup. MSVC previously didn’t do argument-dependent lookup separately from ordinary, unqualified lookup so this change in behavior may be surprising.
最后,关于使 MSVC 使用更新的两阶段模板解析的注意事项:
You’ll need to use the /permissive- conformance switch to enable two-phase lookup in the MSVC compiler included with Visual Studio 2017 “15.3”. Two-phase name lookup drastically changes the meaning of some code so the feature is not enabled by default in the current version of MSVC.
关于c++ - 模板中的友元函数(为什么这在 Visual Studio 中失败但在 GCC 和 Clang 中失败),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46469558/
我在使用以下代码时遇到问题: 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
我是一名优秀的程序员,十分优秀!