c++ - 错误 C2664 : cannot convert argument 2 from 'const ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *' to 'ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *'

标签 c++ visual-c++ com visual-studio-2013 atl

我正在将 VS2008(VC++) 代码迁移到 VS2013。我收到以下错误:

error C2664: 'HRESULT _CopyItfFromAdaptItf<IZipFileEntry>::copy(T **,ATL::CAdapt<ATL::CComPtr<T>> *)' : cannot convert argument 2 from 'const ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *' to 'ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *'

错误出现在Visual studio SDK文件中,怎么可能?。我无法修复此错误。请在下方找到输出和错误行。

非常感谢您的帮助!!

输出:

C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\atlmfc\include\atlcom.h(5818): error C2664: 'HRESULT _CopyItfFromAdaptItf<IZipFileEntry>::copy(T **,ATL::CAdapt<ATL::CComPtr<T>> *)' : cannot convert argument 2 from 'const ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *' to 'ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *'
      with
      [
          T=IZipFileEntry
      ]
      Conversion loses qualifiers
      C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\atlmfc\include\atlcom.h(5803) : while compiling class template member function 'HRESULT ATL::ICollectionOnSTLImpl<IZipFileDir1,EntryList,IZipFileEntry *,_CopyItfFromAdaptItf<IZipFileEntry>,EntryEnum>::get_Item(long,ItemType *)'
      with
      [
          ItemType=IZipFileEntry *
      ]
      C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\atlmfc\include\atlcom.h(5140) : see reference to class template instantiation 'ATL::ICollectionOnSTLImpl<IZipFileDir1,EntryList,IZipFileEntry *,_CopyItfFromAdaptItf<IZipFileEntry>,EntryEnum>' being compiled

错误代码:

if (iter != m_coll.end())
      hr = CopyItem::copy(pvar, &*iter); // Error C2664
return hr;

P.S:我无法编辑和保存这个 atlcom.h 文件。我得到访问路径被拒绝的错误信息。这是因为它是一个 SDK 文件吗?

添加了更多代码定义:

struct _CopyItfFromAdaptItf {
    static HRESULT copy(T** p1, CAdapt< CComPtr<T> >* p2) {
    if( *p1 = p2->m_T ) return (*p1)->AddRef(), S_OK;
    return E_POINTER;
}


IZipFileEntry : public IDispatch
{
    public:
        virtual /* [helpstring][id][propget] */ HRESULT STDMETHODCALLTYPE get_name( 
        /* [retval][out] */ BSTR *pVal) = 0;

        virtual /* [helpstring][id][propget] */ HRESULT STDMETHODCALLTYPE get_lastModified( 
        /* [retval][out] */ DATE *pVal) = 0;

        virtual /* [helpstring][id][propget] */ HRESULT STDMETHODCALLTYPE get_fileSize( 
        /* [retval][out] */ long *pVal) = 0;

        virtual /* [helpstring][id][propget] */ HRESULT STDMETHODCALLTYPE get_compressedSize( 
        /* [retval][out] */ long *pVal) = 0;

    };


typedef ICollectionOnSTLImpl<IZipFileDir1, EntryList, IZipFileEntry*, _CopyItfFromAdaptItf<IZipFileEntry>, EntryEnum> EntryCollection;

ICollectionOnSTLImpl的定义

class ICollectionOnSTLImpl : 
    public T
{
public:
        STDMETHOD(get_Count)(_Out_ long* pcount)
        {
        if (pcount == NULL)
            return E_POINTER;
        ATLASSUME(m_coll.size()<=LONG_MAX);

        *pcount = (long)m_coll.size();

        return S_OK;
    }
    STDMETHOD(get_Item)(
        _In_ long Index, 
        _Out_ ItemType* pvar)
    {
        //Index is 1-based
        if (pvar == NULL)
            return E_POINTER;
        if (Index < 1)
            return E_INVALIDARG;
        HRESULT hr = E_FAIL;
        Index--;
        CollType::const_iterator iter = m_coll.begin();
        while (iter != m_coll.end() && Index > 0)
        {
            iter++;
            Index--;
        }
        if (iter != m_coll.end())
            hr = CopyItem::copy(pvar, &*iter);
        return hr;
    }
    STDMETHOD(get__NewEnum)(_Outptr_ IUnknown** ppUnk)
    {
        if (ppUnk == NULL)
            return E_POINTER;
        *ppUnk = NULL;
        HRESULT hRes = S_OK;
        CComObject<EnumType>* p;
        hRes = CComObject<EnumType>::CreateInstance(&p);
        if (SUCCEEDED(hRes))
        {
            hRes = p->Init(this, m_coll);
            if (hRes == S_OK)
                hRes = p->QueryInterface(__uuidof(IUnknown), (void**)ppUnk);
        }
        if (hRes != S_OK)
            delete p;
        return hRes;
    }
    CollType m_coll;
};

最佳答案

如果没有所有代码,很难确定,但看起来我已经用以下代码重现了这个问题:

template<class T>
struct _CopyItfFromAdaptItf {
    static HRESULT copy(T** p1, CAdapt< CComPtr<T> >* p2) {
        if( *p1 = p2->m_T ) return (*p1)->AddRef(), S_OK;
        return E_POINTER;
    }
};

IDispatch** disp;
const CAdapt<CComPtr<IDispatch>> adapt; // note const here
_CopyItfFromAdaptItf<IDispatch>::copy( disp, &adapt ); //error C2664 here

编译器输出为

File(Line) : error C2664: '_CopyItfFromAdaptItf<T>::copy' :
  cannot convert parameter 2 from 'const ATL::CAdapt<T> *' to 'ATL::CAdapt<T> *'
    with
    [
      T=IDispatch
    ]
    and
    [
      T=ATL::CComPtr<IDispatch>
    ]
    and
    [
      T=ATL::CComPtr<IDispatch>
    ]
    Conversion loses qualifiers

这与您引用的输出非常匹配。

问题是 _CopyItfFromAdaptItf::copy() 想要一个指向非 const 的指针,但传递了一个指向 const 的指针。将指针传递给 const 的代码位于属于 SDK header 的 atlcom.h 中,因此您最好不要更改它。

此外,如果 _CopyItfFromAdaptItf::copy() 签名更改为接受指向 const 的指针:

static HRESULT copy(T** p1, const CAdapt< CComPtr<T> >* p2) // note addition of const

它仍然编译得很好。 _CopyItfFromAdaptItf 不在 ATL header 中,所以我猜它在您的代码中,您可以更改它。

所以看起来解决方案是简单地更改 _CopyItfFromAdaptItf::copy() 以接受指向 const 的指针,问题就会消失。

关于c++ - 错误 C2664 : cannot convert argument 2 from 'const ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *' to 'ATL::CAdapt<ATL::CComPtr<IZipFileEntry>> *' ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22806760/

相关文章:

c++ - 用 std::transform 填充 std::map(错误:无法推断参数)

c++ - 如何使用另一个 DLL 为 Qt 制作第三方插件?

c++ - const指针专门化可使用MSVC生成未解析的符号,并且适用于GCC/Clang

c# - 接口(interface)不能被实例化,但这是一个异常吗

c# - COM 接口(interface)中函数返回中的 Guid 对象

c++ - 使用 "and"运算符与 && 运算符有什么缺点吗?

c++ - 下面的代码出现了奇怪的结果(为什么输入结果 'b' )

windows - 是什么决定了进程外的 COM 服务器需要多长时间才能注意到客户端已死亡?

c++ - 我忘记了什么吗? (适用于 Windows 的双控制台缓冲区)

c++ - 使用 vector 在初始化列表中 move 构造函数