`

使用 Visual C++ 2008 功能包加强 Windows 应用程序

阅读更多
C++ Plus
使用 Visual C++ 2008 功能包加强 Windows 应用程序
Kenny Kerr

本文以 Visual C++ 功能包的预发布版为基础。文中包含的所有信息均有可能变更。
本文讨论:
  • Microsoft 基础类库更新
  • 使用 C++ 对功能区进行编程
  • 使用 C++ 实现选项卡式 MDI
  • 多态函数和智能指针
本文使用了以下技术:
Visual Studio 2008,MFC
作为一名 Visual C++ 开发人员 ,您在最近几年可能有点被冷落的感觉,因为与 Visual C#® 相比,似乎 Microsoft 向 Visual C++® 添加的新特性和新功能要少很多。事实上,尽管 Visual C++ 编译器在性能、安全性和标准符合性等方面始终在不断改进,但很长时间以来在新库和生产率功能方面却做的比较少。后来虽然更新了 MFC 以更好地支持 Windows Vista® ,但仍有许多工作需要完成。
但是现在,为了对那些使用本机代码和 MFC 的开发人员提供更好的支持,Microsoft 发布了 Visual C++ 2008 功能包。以下是 Visual C++ 更新的一些主要内容。
此功能包包括用于构建现代用户界面的一大组新 MFC 类。它还包括作为技术报告 1 (TR1) 的一部分添加到标准 C++ 库中的大量功能。TR1 是 C++ 委员会所采用的首个针对标准 C++ 库的主要更新和添加内容。
多年以来,传统模式的单个和多个文档/视图应用程序、菜单、工具栏和对话框一直是 MFC 开发的一个主要方面。如果想让 MFC 应用程序看起来更现代一点,用户需要自己想办法。
现在则完全不同了。MFC 现在包括了许多新的用户界面模式,其中甚至还有类似于 Microsoft® Office 和 Visual Studio® 中的可停靠窗格。它还完全支持 Microsoft Office 功能区用户界面以及众多其他新控件、对话框和窗口等。
接下来,我将演示 MFC 中的两个新用户界面功能:Office 功能区和选项卡式多文档界面 (MDI)。

Office 功能区用户界面
到目前为止,我相信您已见过了新的 2007 Microsoft Office 系统功能区元素,并且您可能想知道如何在自己的应用程序中营造出这种效果。令人高兴的是,现在可以非常轻松地向 MFC 框架窗口添加功能区栏。
许多新功能都依赖于新版本的 CwinApp、CFrameWnd 和 CMDIFrameWnd 类;这些类代表着大多数 MFC 应用程序的基础。CWinAppEx 由 CwinApp 派生而来,应该用作应用程序对象的基类。CFrameWndEx 由 CframeWnd 派生而来,应该用作单文档界面 (SDI) 框架窗口的基类。同样,CMDIFrameWndEx 由 CMDIFrameWnd 派生而来,应该用作 MDI 框架窗口的基类。这些新的基类提供了支持众多新用户界面功能(如可停靠、可调整大小的窗口窗格以及工作区持久性等)所需的全部要素。
图 1 显示了可支持功能区栏的最小应用程序对象。如您所见,Application 类由 CwinAppEx 派生而来,可实现大家所熟悉的 InitInstance 成员函数(通常用于创建应用程序的主窗口)。千万不要忘记调用 SetRegistryKey 成员函数来设置应用程序设置的注册表位置,因为框架类要依赖于它。然后,InitInstance 继续以通常的方式创建主窗口。
class Application : public CWinAppEx
{
public:

virtual BOOL InitInstance();

};

BOOL Application::InitInstance()
{
SetRegistryKey(L"SampleCompany\\SampleProduct");

m_pMainWnd = new MainWindow;
m_pMainWnd->ShowWindow(m_nCmdShow);
m_pMainWnd->UpdateWindow();

return TRUE;
}

图 2 中的代码显示了具有功能区栏和应用程序按钮的一个最小 SDI 框架窗口。应用程序按钮并不是必需的,但通常会与功能区栏结合使用,为应用程序提供各种各样的主菜单,以代替传统的“文件”菜单。
class MainWindow : public CFrameWndEx
{
DECLARE_MESSAGE_MAP()

public:

MainWindow();

private:

int OnCreate(CREATESTRUCT* createStruct);

CMFCRibbonBar m_ribbon;
CMFCRibbonApplicationButton m_appButton;
};

BEGIN_MESSAGE_MAP(MainWindow, CFrameWndEx)
ON_WM_CREATE()
END_MESSAGE_MAP()

MainWindow::MainWindow()
{
Create(0, // class name
L"MFC Ribbon Sample Application");
}

int MainWindow::OnCreate(CREATESTRUCT* createStruct)
{
if (-1 == __super::OnCreate(createStruct))
{
return -1;
}

if (-1 == m_ribbon.Create(this))
{
return -1;
}

m_appButton.SetImage(IDB_APP_BUTTON);

m_ribbon.SetApplicationButton(&m_appButton,
CSize(45, 45));

CMFCRibbonMainPanel* appButtonMenu =
m_ribbon.AddMainCategory(L"Menu",
IDB_APP_BUTTON_MENU_SMALL,
IDB_APP_BUTTON_MENU_LARGE);

appButtonMenu->Add(new CMFCRibbonButton(ID_FILE_NEW,
L"&New",
0, // small image index
0)); // large image index

appButtonMenu->Add(new CMFCRibbonButton(ID_FILE_OPEN,
L"&Open...",
1, // small image index
1)); // large image index

appButtonMenu->AddToBottom(new CMFCRibbonMainPanelButton(ID_APP_EXIT,
L"E&xit",
15));
//small image index

CMFCRibbonCategory* category = m_ribbon.AddCategory(L"Home",
IDB_RIBBON_CAT_HOME_SMALL,
IDB_RIBBON_CAT_HOME_LARGE);

CMFCRibbonPanel* panel = category->AddPanel(L"Clipboard");

panel->Add(new CMFCRibbonButton(ID_EDIT_PASTE,
L"Paste",
0, // small image index
0)); // large image index

panel->Add(new CMFCRibbonButton(ID_EDIT_CUT, L"Cut", 1));
panel->Add(new CMFCRibbonButton(ID_EDIT_COPY, L"Copy", 2));
panel->Add(new CMFCRibbonButton(ID_EDIT_SELECT_ALL,
L"Select All", -1));

m_ribbon.AddCategory(L"Insert",
IDB_RIBBON_CAT_HOME_SMALL,
IDB_RIBBON_CAT_HOME_LARGE);

CMFCVisualManager::SetDefaultManager(
RUNTIME_CLASS(CMFCVisualManagerOffice
2007));
CMFCVisualManagerOffice2007::SetStyle
(CMFCVisualManagerOffice2007::Office
2007_LunaBlue);

return 0;
}

从概念上讲,功能区由多个被称为类别的选项卡组成,每个选项卡都承载着一组面板。反过来,这些面板又承载着功能区元素或控件,它们分别代表特定于应用程序的各种操作。如果功能区承载着应用程序按钮(左上角的大圆按钮),则在用户单击应用程序按钮时所显示的弹出窗口中也会显示一个面板,它被视为功能区的主类别。
CMFCRibbonBar 类可实现功能区栏本身,而 CMFCRibbonApplicationButton 类则代表功能区栏所承载且显示在窗口框架左上角的应用程序按钮。功能区栏通常是在 WM_CREATE 消息处理程序中创建和准备的。要创建功能区栏,只需调用 CMFCRibbonBar 的 Create 成员函数,以提供将其附加到其中的窗口框架的地址即可。然后根据需要填充它。AddMainCategory 成员函数将主类别添加到功能区,并返回一个指向 CMFCRibbonMainPanel(您可以向其中添加将要显示在此面板中的功能区元素)的指针。
通过调用 AddCategory 成员函数,可向其中添加更多的类别来表示功能区的选项卡。AddCategory 返回一个指向 CMFCRibbonCategory 对象(您可使用其 AddPanel 成员函数向其中添加面板)的指针。AddPanel 返回一个指向 CMFCRibbonPanel 对象(可像使用功能区的主面板一样向其中添加功能区元素)的指针。最后,您可使用 CMFCVisualManager::SetDefaultManager 静态成员函数来设置负责处理框架窗口的样式和外观的可视化管理器。图 3 显示了功能区应用程序的可能外观(假定您已为功能区栏上的按钮添加了必要的事件处理程序)。
图 3功能区示例应用程序

选项卡式多文档界面
MFC 一直都支持 MDI 实现及其文档/视图体系结构,但图 4 中所示的传统 MDI 早已过时,用户可能会认为您的应用程序从 Windows® 95 开始就再也没有更新过。现在,绝大多数用户都期望可通过窗口边缘的选项卡来访问多个文档,而这正是新的 CMDIFrameWndEx MDI 框架窗口所提供的功能。
图 4石器时代的窗口 (单击该图像获得较大视图)
您需要更新多文档/视图应用程序对象以支持新的框架窗口。图 5 显示了满足需要的最小应用程序对象。它类似于传统的 MDI 应用程序对象,但有几点值得一提。
class Application : public CWinAppEx
{
DECLARE_MESSAGE_MAP()

public:

virtual BOOL InitInstance();

};

BEGIN_MESSAGE_MAP(Application, CWinAppEx)
ON_COMMAND(ID_FILE_NEW, &CWinAppEx::OnFileNew)
END_MESSAGE_MAP()

BOOL Application::InitInstance()
{
SetRegistryKey(L"SampleCompany\\SampleProduct");

VERIFY(InitContextMenuManager());

AddDocTemplate(new CMultiDocTemplate(IDR_CHILDFRAME,
RUNTIME_CLASS(Document),
RUNTIME_CLASS(CMDIChildWndEx),
RUNTIME_CLASS(View)));

MainWindow* mainWindow = new MainWindow();
VERIFY(mainWindow->LoadFrame(IDR_MAINFRAME));

m_pMainWnd = mainWindow;
m_pMainWnd->ShowWindow(m_nCmdShow);
m_pMainWnd->UpdateWindow();

return TRUE;
}

首先,子窗口框架的运行时类是 CMDIChildWndEx,而非传统的 CMDIChildWnd 类。要准备在选项卡式视图之间进行切换时使用的菜单管理器,还需调用 InitContextMenuManager 函数。
图 6 显示了最小 MDI 框架窗口。同样,您会非常高兴地看到开启此功能是多么地简单。实际只需调用 EnableMDITabbedGroups 成员函数来启用 MDI 选项卡式分组功能即可。CMDITabInfo 类提供了各种成员变量,可使用它们来自定义选项卡式分组的外观和行为。顾名思义,它甚至还允许用户拖动不同的视图来创建垂直或水平对齐的选项卡组。图 7 显示了其可能的外观。
class MainWindow : public CMDIFrameWndEx
{
DECLARE_DYNCREATE(MainWindow)
DECLARE_MESSAGE_MAP()

private:

int OnCreate(CREATESTRUCT* createStruct);

};

IMPLEMENT_DYNCREATE(MainWindow, CMDIFrameWndEx)

BEGIN_MESSAGE_MAP(MainWindow, CMDIFrameWndEx)
ON_WM_CREATE()
END_MESSAGE_MAP()

int MainWindow::OnCreate(CREATESTRUCT* createStruct)
{
if (-1 == __super::OnCreate(createStruct))
{
return -1;
}

CMDITabInfo tabInfo;
tabInfo.m_bAutoColor = true;
tabInfo.m_bDocumentMenu = true;
EnableMDITabbedGroups(true, tabInfo);

return 0;
}

图 7现代选项卡式 MDI 应用程序 (单击该图像获得较大视图)

标准 C++ 库中的新功能
正如我所提到的,功能包还包括作为 TR1 的一部分添加到标准 C++ 库中的大量附加功能。其中包括支持引用计数的智能指针、多态函数包装、基于哈希表的容器、正则表达式等等。下面我将介绍其中的一些新 TR1 功能。

多态函数对象
在许多应用程序中都有一个至关重要的功能,就是能够将函数作为一个值加以引用并能够将其作为参数来传递或存储起来以备今后使用。此概念可用于实现各种常见的构造,包括回调函数、事件处理程序和异步编程功能等。但是,函数在 C++ 中非常难于处理。函数设计的驱动力主要源自与 C 的兼容性的要求以及对优良性能的要求。尽管实现了这些目标,但在将函数视为可存储、可传递并最终能够异步调用的对象方面,却并未能使其变得简单一些。让我们来看一看 C++ 中常见的一些类似函数的构造。
首先,是一个不错的古老非成员函数:
int Add(int x, int y)
{
return x + y;
}
正常情况下,可通过如下方法调用它:
int result = Add(4, 5);
ASSERT(4 + 5 == result);
另一个常见的类似函数的构造是函数对象(即算符):
class AddFunctor
{
public:
int operator()(int x, int y) const
{
return x + y;
}
};
由于它实现调用运算符,因此可像使用函数一样来使用函数对象:
   AddFunctor fo;

int result = fo(4, 5);
ASSERT(4 + 5 == result);
接下来是非静态成员函数:
   class Adder
{
public:
int Add(int x, int y) const
{
return x + y;
}
};
当然,调用成员函数需要使用对象:
Adder adder;

int result = adder.Add(4, 5);
ASSERT(4 + 5 == result);
到目前为止一切顺利。现在,假设您需要将这些类似函数的构造存储起来以备今后使用。可按如下方式定义一个能存储指向非成员函数的指针的类型:
typedef int (*FunctionPointerType)(int x, int y);
也可将函数指针作为函数来使用:
FunctionPointerType fp = &Add;

int result = fp(4, 5);
ASSERT(4 + 5 == result);
尽管函数对象也可以存储下来,但它无法与函数指针一起以多态形式存储。
成员函数可存储在 pointer-to-member-function 中:
Adder adder;

typedef int (Adder::*MemberFunctionPointerType)(int x, int y);
MemberFunctionPointerType mfp = &Adder::Add;
但是,pointer-to-member-function 类型与 pointer-to-non-member-function 类型不兼容,因此无法与其非成员函数竞争者一起以多态形式存储。即使可以,成员函数仍需要一个对象来提供成员函数调用的上下文:
int result = (adder.*mfp)(4, 5);
ASSERT(4 + 5 == result);
我想我不必再做解释您也应该明白我的意思了。幸运的是,新的 tr1::function 类模板提供了一个解决方案。tr1::function 类模板为在其模板参数中定义的函数类型保存着一个可调用对象。接下来,我将使用非成员函数对其进行初始化:
function<int (int x, int y)> f = &Add;

int result = f(4, 5);
ASSERT(4 + 5 == result);
使用函数对象来初始化也一样轻松:
function<int (int x, int y)> f = AddFunctor();
您甚至还可以使用新的函数绑定功能通过成员函数对其进行初始化:
function<int (int x, int y)> f = bind(&Adder::Add, &adder, _1, _2);
有关 bind 函数的内容我会在稍后做介绍,但在这里您需要了解的就是现在可将单个函数包装绑定到非成员函数、函数对象甚至成员函数中。可将其存储下来并在今后随时调用,所有这一切都是以多态形式执行的。
函数包装也是可以重新绑定的,并且可以像普通的函数指针一样设置为空值:
function<int (int x, int y)> f;
ASSERT(0 == f);

f = &Add;
ASSERT(0 != f);

f = bind(&Adder::Add, &adder, _1, _2);
bind 函数模板的功能要比标准 C++ 库中的函数对象适配器强大得多——尤其是 std::bind1st() 和 std::bind2nd()。在这个示例中,bind 的第一个参数是成员函数的地址。第二个参数是对象的地址,届时将在此对象中调用成员。此示例中的最后两个参数定义了调用函数时将要解析的占位符。
当然,bind 并不仅限于成员函数。您可通过绑定标准 C++ 库的 multiplies 函数对象来创建一个平方函数,利用此函数可生成一个能得出参数平方结果的单参数函数:
function<int (int)> square = bind(multiplies<int>(), _1, _1);

int result = square(3);
ASSERT(9 == result);
请注意,tr1::function 类模板非常适合与标准 C++ 库算法一起使用。给定一个整数容器,就可以使用成员函数生成所有值的总和,如下所示:
function<int (int x, int y)> f = // initialize

int result = accumulate(numbers.begin(),
numbers.end(),
0, // initial value
f);
请记住,tr1::function 类模板可能会禁止编译器优化(如内联),但如果您只是直接使用函数指针或函数对象则可能不会出现这一问题。因此,请仅在必要时才使用 tr1::function 类模板,例如当使用可能会被重复调用的累积算法时。如果可能,应直接将函数指针、成员函数指针(使用 TR1 的 mem_fn 改写过的)以及函数对象(如 bind 所返回的)传递给标准 C++ 库算法和其他模板化的算法。
让我们接着往下看。接下来还有个更有趣的问题。假设有个 Surface 类代表一些绘图表面,还有个 Shape 类,它可以将其自身绘制到表面上:
class Surface
{
//...
};

class Shape
{
public:
void Draw(Surface& surface) const;
};
现在考虑一下怎样才能够将容器中的每个形状都绘制到给定表面上。您可能会考虑使用 for_each 算法,如下所示:
Surface surface = // initialize

for_each(shapes.begin(),
shapes.end(),
bind(&Shape::Draw, _1, surface)); // wrong
在这里,我打算利用 bind 函数模板来针对形状容器的每个元素调用成员函数,从而将表面作为参数绑定到 Draw 成员函数。但遗憾的是,这要取决于 Surface 的定义方式,有时可能无法按预期的那样运行或编译。之所以出现这个问题,是因为当您实际需要的是一个引用时,bind 函数模板却试图生成表面副本。值得庆幸的是,TR1 还引入了 reference_wrapper 类模板,它允许您将引用视为一个可随意复制的值。由于类型推断功能的存在,ref 和 cref 函数模板可简化 reference_wrapper 对象的创建过程。
借助于 reference_wrapper,for_each 算法现在可以简单有效地将形状成功绘制到表面上:
for_each(shapes.begin(),
shapes.end(),
bind(&Shape::Draw, _1, ref(surface)));
正如您所设想的,对于新的函数包装、绑定功能和引用包装,可以通过多种方式来组合它们,从而灵活地解决各种问题。

智能指针
智能指针对 C++ 开发人员而言是不可或缺的工具。我通常使用 ATL 的 CComPtr 来处理 COM 接口指针,使用标准 C++ 库的 auto_ptr 来处理原始 C++ 指针。后者非常适合需要动态创建 C++ 对象并能够确保当 auto_ptr 对象超出范围时可以将对象安全删除的情形。
智能指针像 auto_ptr 一样都非常有用,但它只能安全地用在少数情形下。这主要是因为它所实现的所有权转移语义。即,如果复制或分配 auto_ptr 对象,则基础资源的所有权将被转移,原始 auto_ptr 对象将失去它。只有当您对资源分配拥有精细的控制权时它才会有明显作用,但很多情况下您可能需要共享对象,这时实现共享所有权语义的智能指针将会非常有用。更为重要的是,auto_ptr 无法与标准 C++ 库容器一起使用。
TR1 引入了两个新的智能指针,它们协同工作来提供多种用途。shared_ptr 类模板的工作方式与 auto_ptr 十分相似,但它不能转移资源的所有权,它只是增加资源的引用计数。如果用来保存对象引用信息的最后一个 shared_ptr 对象被破坏或重置,资源将被自动删除。通过 weak_ptr 类模板与 shared_ptr 的协同工作,调用方可以在不影响引用计数的情况下引用资源。如果在对象模型中有循环关系或打算实现缓存服务,则这将非常有用。它还非常适合与标准 C++ 库容器一起使用!
作为对比,请看一看以下的 auto_ptr 用法:
auto_ptr<int> ap(new int(123));

ASSERT(0 != ap.get());

// transfer ownership from ap to ap2
auto_ptr<int> ap2(ap);

ASSERT(0 != ap2.get());
ASSERT(0 == ap.get());
auto_ptr 复制构造函数将所有权从 ap 传输到 ap2。shared_ptr 的行为同样是可预测的:
shared_ptr<int> sp(new int(123));

ASSERT(0 != sp);

// increase reference count of shared object
shared_ptr<int> sp2(sp);

ASSERT(0 != sp2);
ASSERT(0 != sp);
从内部来说,引用相同资源的所有 shared_ptr 对象都共享一个控制块,此控制块将跟踪共同拥有资源的 shared_ptr 对象的数量以及引用此资源的 weak_ptr 对象的数量。稍后我将展示如何使用 weak_ptr 类模板。
与 auto_ptr 类似的成员函数由 shared_ptr 提供。其中包括解引用操作符和箭头操作符、用来替换资源的 reset 成员函数以及返回资源地址的 get 成员函数。此外还提供一些特有的成员函数(其中包括一个恰好也以 unique 命名的函数)。unique 成员函数将测试 shared_ptr 对象是否为保存着资源引用信息的唯一智能指针。示例如下:
shared_ptr<int> sp(new int(123));
ASSERT(sp.unique());

shared_ptr<int> sp2(sp);
ASSERT(!sp.unique());
ASSERT(!sp2.unique());
也可以使用 use_count 成员函数来获取拥有资源的 shared_ptr 对象的数量:
shared_ptr<int> sp;
ASSERT(0 == sp.use_count());

sp.reset(new int(123));
ASSERT(1 == sp.use_count());

shared_ptr<int> sp2(sp);
ASSERT(2 == sp.use_count());
ASSERT(2 == sp2.use_count());
但是,应将 use_count 的用途仅限于进行调试,因为无法保证它在所有实现中都是一个恒定的时间操作。请注意,可借助提供的操作符 unspecified-bool-type 来确定 shared_ptr 是否拥有资源,并且可使用 unique 函数来确定 shared_ptr 是否为某个资源的唯一拥有者。
weak_ptr 类模板存储着对 shared_ptr 对象所拥有的资源的弱引用。如果拥有资源的所有 shared_ptr 对象都被破坏或重置,资源将被删除,无论是否有 weak_ptr 对象正在引用它。为确保不使用仅被 weak_ptr 对象引用的资源,weak_ptr 类模板不会提供熟悉的 get 成员函数来返回资源的地址或成员访问操作符。相反,首先必须将弱引用转换为强引用才能访问资源。lock 成员函数提供了此功能,如图 8 所示。
Surface surface;

shared_ptr<Shape> sp(new Shape);
ASSERT(1 == sp.use_count());

weak_ptr<Shape> wp(sp);
ASSERT(1 == sp.use_count()); // still 1

// arbitrary application logic...

if (shared_ptr<Shape> sp2 = wp.lock())
{
sp2->Draw(surface);
}

如果资源中途被释放,weak_ptr 对象的 lock 成员函数会返回一个并不拥有资源的 shared_ptr 对象。可以想象,shared_ptr 和 weak_ptr 必将会极大地简化许多应用程序中的资源管理工作。
毋庸置疑,Visual C++ 2008 功能包是一个备受欢迎的 Visual C++ 库升级程序,肯定迟早会派上用场!值得研究的内容不胜枚举,但我希望通过本文的介绍能激起您的兴趣,使您能够花些时间亲自深入研究一下。

Kenny Kerr 是一名专门从事 Windows 软件开发的软件专家。他热衷于撰写有关编程和软件设计的文章,并向开发人员讲授与此有关的知识。您可通过他的博客 weblogs.asp.net/kennykerr 与 Kenny 联系。

分享到:
评论

相关推荐

    Microsoft Visual C++ 2015 Redistributable Update 3 RC

    这些组件是运行使用 Visual Studio 2015(含 Update 3 RC)开发的 C++ 应用程序所必需的,并与 Visual C+ 库动态链接。这些包可用于在计算机上运行此类应用程序,即使没有安装 Visual Studio 2015。这些包还安装以下...

    Visual C++ 2012入门经典(第6版)中文版、英文版和源代码

    通过学习本书, 您可以使用Microsoft公司最新的应用程序开发系统Visual Studio 2012,成为优秀的C++程序员。本书旨在讲述C++程序设计语言,然后讲述如何运用C++语言开发 自己的Windows应用程序。在此过程中,读者...

    Visual C++教程包含doc版Pdf版本。

    本光盘教学软件的目的就是为了让你学会在Visual C++环境下,利用微软的基本类库MFC开发出功能强大的Windows应用程序。在本章节的内容当中,我们将向您介绍使用VC开发软件需要用到的一些基本概念,使用MFC进行编程的...

    使用Visual C++编写的聊天程序

    使用Visual C++编写的聊天程序,可实现私聊和传送文件等功能,有利于学习,可直接使用。包含客户端和服务端。

    Microsoft Visual C++运行库合集包完整版2021年8月版_x86_x64_v52.exe

    Microsoft Visual C++ Redistributable(系统运行库,简称MSVC,VB/VC)是Windows操作系统应用程序的基础类型库组件。此版Visual C 运行库合集包(VisualCppRedist),是由国外My Digital Life论坛@abbodi1406改装而...

    Microsoft Visual C++ Redistributable Package 2019 14.28.29812.0 下载.zip

    如果您使用同一版本的 Microsoft Visual C ++ 创建多个应用程序,则所有使用同一版本的 Microsoft Visual C ++ 创建的应用程序都需要一个可再发行组件包。 当您需要 Microsoft Visual C ++ 可再发行组件时 如果您有...

    Visual C++范例大全

    且以相关知识点命名,便于速查和学习实例均给出了核心代码,各功能实现尽可能采用独立函数,可移植性强《VisualC++范例大全》内容概述:VisualC++与C++开发基础(21个实例);消息映射与处理(17个实例);Windows...

    WINDOWS 7安装visual c++ 2005 2008

    MindManager8安装时提示要安装visual c++ 2005 sp1,但在win7下安装visual c++ 2005 sp1老是失败,安装wisual c++ 2005 时提示:无法访问您试图使用的功能所在的网络位置。 手动指定相应vcredist.msi文件,却提示...

    Microsoft Visual C++ 1.52c.7z DOS终结者完整ISO镜像

    直到今天,它仍然具有强大的生命力,一些为MSDOS、Windows 3.1编写16位应用程序的程序员还在使用这个版本,我个人也很喜欢这样小巧的开发工具,因为我只需库文件和编译工具而已,完成编码后剩下的工作就交给make工具...

    学生成绩管理(Visual C++ )

    如果应用程序在共享 DLL 中使用 MFC,且应用程序使用的语言不是 操作系统的当前语言,则需要从 Microsoft Visual C++ 光盘上 WinSystem 目录下将相应的本地化资源 MFC70XXX.DLL 复制到计算机的 system 或 ...

    Visual C++2010开发权威指南.part09

    第20章 Visual C++ 2010应用程序 第20章 部署 721 20.1 Windows Installer介绍 722 20.2 一个简单的Windows应用 20.2 程序 723 20.3 使用安装向导快速创建安 20.3 装包 726 20.4 手动创建安装程序 728 20.5 使用安装...

    Microsoft Visual C++(VC运行库)运行库

    这些文件包含了一系列常用的函数和工具,如内存管理、输入输出、字符串处理等,旨在为开发人员提供便利,以便他们能够更快速地开发各种应用程序。运行库的角色类似于共享工具箱,允许多个程序共用相同的功能模块,...

    《Visual C++网络通信开发入门与编程实践》书中的源代码

    《Visual C++网络通信开发入门与编程实践》的特点是:以通信协议层的技术为主线,在此基础上讲解各技术的应用范围,再通过众多孝中、大型实例来全面而生动地讲解VC++的知识,既能让读者全面地学习VC++技术,又能让...

    Visual C++开发实用编程200例

    Visual C++作为Visual Studio产品的一部分,是一个功能很强大的综合性集成软件开发环境,对众多应用领域(网络、多媒体、数据库等)都有很好的支持,Visual C++是广大软件开发人员特别是系统程序员做开发的首选平台。...

    第2章 MFC基础 本章先介绍Visual C++和MFC的基本情况,然后讨论Windows API和MFC编程的基本方法。

    MFC(Microsoft Foundation Class Library,微软基础类库)是Windows API的C++类封装,是一种基于Visual C++的应用程序框架,提供了文档/视图结构和新消息映射机制,简化了Windows应用程序的编写。 本章先介绍Visual...

    Visual C++ 数据库系统开发完全手册.part1

    4.1.2 Visual C++ 6.0中的应用程序向导 4.1.3 MFC应用程序的开发流程 4.2 应用程序向导生成文件 4.2.1 文件类型 4.2.2 典型文件举例 4.3 ClassWizard(类向导) 4.3.1 ClassWizard的功能 4.3.2 ClassWizard的简单...

    Visual_C++从入门到实践

    Visual C++ 6.0提供了功能强大的向导工具,包 括MFC AppWizard、ClassWizard和MFC ActiveX ControlWizard等。Visual C++ 6.0同时提供了对数据库 开发和Internet的强大支持。本章讲解的知识点包括: Visual C++概述...

    Visual C++开发GIS系统:开发实例剖析

    本书是GIS系统开发技术和Visual C++编程技术的集合体,全面介绍和分析了MFC程序的框架结构和各种MFC辅助工具的使用方法,实例剖析了MFC中与图形和数据库应用程序设计有关的类。介绍了GIS系统的开发方法和思路,结合...

    SkinMagic SDK for Visual C++

    SkinMagic SDK是为C/C++程序员设计的函数库,通过使用该库提供的API可以很方便的为你 的应用程序增加最流行的换皮肤的功能。SkinMagic是基于图片的皮肤系统,只要简单的几个 函数调用你的应用程序就具有了换皮肤的...

Global site tag (gtag.js) - Google Analytics