来自 std::chrono 的编译器警告但未使用

dou*_*oug 5 c++ compiler-warnings visual-studio

注意:此错误仅在发布和调试模式下的 x64 项目中出现。

std::chrono使用 VC2019 的这段代码出现警告级别为 3 的奇怪警告。这是一段处理命令行标志的精简代码。我已经删除了大部分与问题无关的内容。

#if 1   // enable bug
#include <chrono>   // excluding this also eliminates chrono warnings
using CorrectedIntType=int;
#else
using CorrectedIntType=size_t;
#endif

#include <iostream>
#include <vector>
#include <string>
#include <type_traits>

using std::vector;
using std::string;

namespace {
    void fixup(const std::string& argcmd, std::string& arg) { arg = argcmd; }

    template<class T>
    void procVal(std::vector<std::string>& arglist, CorrectedIntType idx, T& arg)
    {
        fixup(arglist[idx], arg);
        arglist.erase(arglist.begin() + idx);
    }

    template<class T, class ...TA>
    void procVal(std::vector<std::string>& arglist, CorrectedIntType idx, T& arg, TA&...argv)
    {
        procVal(arglist, idx, arg);
        procVal(arglist, idx, argv...);
    }

    template<class T, class ...TA>
    bool procFlag(const char* pc, std::vector<std::string>& arglist, T& arg1, TA&...argv)
    {
        std::string flag(pc);
        for (size_t i = 0; i < arglist.size(); i++)
        {
            if (arglist[i] == flag)
            {
                arglist.erase(arglist.begin() + i);
                procVal(arglist, i, arg1);      // process one argument after flag
                return true;
            }
        }
        return false;
    }
}

int main()
{
    string outfile;
    vector<string> test = { "test" };
    procFlag("-o", test, outfile);      // assigns test[0] to outfile and removes it
    std::cout << outfile << '\n';
}
Run Code Online (Sandbox Code Playgroud)

警告:

1>Source.cpp
1>C:\Users\mgray\Documents\Visual Studio 2017\Projects\CommandLineCPP\stackoverflow\Source.cpp(35,1): warning C4267: 'argument': conversion from 'size_t' to 'CorrectedIntType', possible loss of data
1>C:\Users\mgray\Documents\Visual Studio 2017\Projects\CommandLineCPP\stackoverflow\Source.cpp(54): message : see reference to function template instantiation 'bool `anonymous-namespace'::procFlag<std::string,>(const char *,std::vector<std::string,std::allocator<std::string>> &,T &)' being compiled
1>        with
1>        [
1>            T=std::string
1>        ]
1>C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\VC\Tools\MSVC\14.24.28314\include\chrono(632): message : see reference to class template instantiation 'std::chrono::duration<double,std::ratio<1,1>>' being compiled
1>C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\VC\Tools\MSVC\14.24.28314\include\chrono(178): message : see reference to class template instantiation 'std::chrono::duration<__int64,std::nano>' being compiled
1>C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise\VC\Tools\MSVC\14.24.28314\include\chrono(610): message : see reference to class template instantiation 'std::chrono::time_point<std::chrono::steady_clock,std::chrono::nanoseconds>' being compiled
Run Code Online (Sandbox Code Playgroud)

虽然代码可以工作,但即使存在int -<> size_t合法警告的转换问题,当顶部的宏设置为 0 时,所有警告都会消失。因此,size_t 和 int 之间的大小差异会以某种方式触发 chrono 消息。我担心这些chrono警告的存在,因为它不涉及。这是VS2019的bug吗?关于为什么chrono会出现警告引用有什么想法吗?

Fed*_*dor 0

这是一个有效的警告,它与您自己的代码和类型无关,<chrono>但与您自己的代码和CorrectedIntType类型无关。这是一个没有的简化代码<chrono>https ://gcc.godbolt.org/z/qf9v8TEh7

在 的定义中procVal,第二个参数是CorrectedIntType

void procVal(std::vector<std::string>& arglist, CorrectedIntType idx, T& arg)
Run Code Online (Sandbox Code Playgroud)

但它是从有价值的地方调用procFlagsize_t

bool procFlag(const char* pc, std::vector<std::string>& arglist, T& arg1)
    ...
    for (size_t i = 0; i < arglist.size(); i++)
    ...
            procVal(arglist, i, arg1);
Run Code Online (Sandbox Code Playgroud)

i因此,人们也可以通过将类型更改为来修复警告CorrectedIntType