包含DShow.h在BDS2006上打破了VCL AnsiString :: sprintf()

时间:2016-10-12 13:39:52

标签: c++ c++builder directshow ansistring

我终于有时间升级我的视频捕捉课程了。我想比较 VFW (我到目前为止使用的)和 DirectShow 。正如预期的那样, DirectShow 更快,但当我添加信息文本时,AnsiString::sprint()突然不再是AnsiString的成员。

经过一番努力,我找到了一个解决方法,因为AnsiString::printf()仍然有效,但我很好奇如何解决这个问题。也许某些定义来自dshow.hdstring.h是冲突的?

我删除了所有不必要的代码以显示此问题:

//$$---- Form CPP ----
//---------------------------------------------------------------------------
#include <vcl.h>
#include <dshow.h>
#pragma hdrstop

#include "Unit1.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm1 *Form1;
//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponent* Owner)
    : TForm(Owner)
{
}
//---------------------------------------------------------------------------
void __fastcall TForm1::Timer1Timer(TObject *Sender)
{
    static int i=0;
    Caption=AnsiString().sprintf("%i",i);               // this does not work
    AnsiString s; s.printf("%i",i); Caption=s;  // this does work
    i++;
}
//---------------------------------------------------------------------------

它只是一个简单的VCL表单应用程序,上面有一个TTimerTTimer正在递增计数器i并将其输出到表单Caption中。 DirectX 库甚至没有链接,只包含标题!

链接器输出错误:

[C++ Error] Unit1.cpp(20): E2316 'sprintf_instead_use_StringCbPrintfA_or_StringCchPrintfA' is not a member of 'AnsiString'

如果我在这一行交换vcl.h和dshow.h includes, the compiler stops in dstring.h`:

AnsiString& __cdecl         sprintf(const char* format, ...); // Returns *this

出现此错误消息:

[C++ Error] dstring.h(59): E2040 Declaration terminated incorrectly

因此,显然存在一些冲突(AnsiString关键字是问题)。将dshow.h放入namespace也无济于事。

有没有人有任何线索?

Q1。如何解决这个问题?

Q2。究竟是什么/导致了这个?

我能想到并且应该工作的唯一解决方案(但我想尽可能避免)是创建 OBJ (或 DLL )使用 DirectShow 的东西,然后将其链接到标准的 VCL 项目中,而不包括dshow.h,当然导出必须没有任何有趣的东西,太

2 个答案:

答案 0 :(得分:1)

我没有这个版本的dshow.h和dstring.h,所以我不能自己查看,但是从你引用的错误消息来看,它似乎在dshow.h中的某个地方或者它们的依赖项中它们声明了&#34; sprintf的&#34;宏。您可以查看是否可以找到它。

为了防止这种行为,您需要删除此宏。使用

CAST(NOW() AS DATE)

在包含dshow.h的行之后。

答案 1 :(得分:1)

问题不在于dshow.h本身,而在于strsafe.h,而dshow.h默认包含{。}}。

strsafe.h包含以下代码 1

#ifndef STRSAFE_NO_DEPRECATE
// Deprecate all of the unsafe functions to generate compiletime errors. If you do not want
// this then you can #define STRSAFE_NO_DEPRECATE before including this file
#ifdef DEPRECATE_SUPPORTED

...
#pragma deprecated(sprintf)
...

#else // DEPRECATE_SUPPORTED

...
#undef sprintf
#define sprintf     sprintf_instead_use_StringCchPrintfA_or_StringCbPrintfA;
...

#endif  // DEPRECATE_SUPPORTED
#endif  // !STRSAFE_NO_DEPRECATE

1 对于许多其他已弃用的“不安全”C函数,有类似的#pragma#define语句。

如果未定义STRSAFE_NO_DEPRECATEDEPRECATE_SUPPORTED(在这种情况下就是这种情况),#define sprintf的使用会导致任何类型的所有后续引用 sprintf符号在编译期间被视为sprintf_instead_use_StringCchPrintfA_or_StringCbPrintfA;

这就是您收到编译器错误的原因。在vcl.h之前包含strsafe.h时,首先包含dstring.h,因此编译器会看到AnsiString::sprintf()方法的正确的声明,然后{在编译器看到您的strsafe.h代码之前,{1}}被包含在内(可能是Unit1.h),因此您对Timer1Timer()的调用实际上是在尝试调用失败的AnsiString().sprint("%i",i)

当您交换AnsiString().sprintf_instead_use_StringCchPrintfA_or_StringCbPrintfA;("%i",i)vcl.h包含时,dshow.h中的#define sprintf语句会在包含strsafe.h之前得到处理,因此编译器会看到以下内容dstring.hAnsiString::sprintf()方法的声明失败:

dstring.h

为了防止出现这种情况,可能AnsiString& __cdecl sprintf_instead_use_StringCchPrintfA_or_StringCbPrintfA;(const char* format, ...); // Returns *this 后使用#undef sprintf语句,就像JeffRSon建议的那样。但正确解决方案是在#include <dshow.h>之前定义STRSAFE_NO_DEPRECATE。你可以通过以下任何一种方式做到这一点:

  1. #include <strsafe.h>声明之前将#define STRSAFE_NO_DEPRECATE添加到您的代码中

  2. #include <dshow.h>添加到“项目选项”中的“条件”列表中。

  3. 此解决方案在MSDN上描述:

    About Strsafe.h

      
        
    • 在文件中包含Strsafe.h时,将弃用由Strsafe.h函数替换的旧函数。尝试使用这些旧函数将导致编译器错误,告诉您使用较新的函数。如果要覆盖此行为,请在包含Strsafe.h之前包括以下语句。

      STRSAFE_NO_DEPRECATE
    •   
    • 要仅允许字符计数功能,请在包含Strsafe.h之前包含以下语句。

      #define STRSAFE_NO_DEPRECATE
      
    •   
    • 要仅允许字节计数功能,请在包含Strsafe.h之前包含以下语句。

      #define STRSAFE_NO_CB_FUNCTIONS
      
    •   

    另一个受支持的解决方案是在#define STRSAFE_NO_CCH_FUNCTIONS 之前定义NO_DSHOW_STRSAFE,以便它不再包含#include <dshow.h>,这要归功于strsafe.h中的此代码:

    dshow.h