作者 : 卿笃军

题目说明:

仿照例题,在其基础上实现下面功能:

1)使用C++语言实现进程内组件,组件提供复数的加、减、乘、除等计算服务;client部分包含录入(实部和虚部分开录入)和查询部分。

2)在VC++环境上利用ATL向导进行开发。

3)至少实现三种执行模式中的两种。

4) 程序设计风格良好,有文档凝视,方法凝视,语句凝视。并附带说明文档。

5)下周六晚10点前统一发至我邮箱。

开工、首先编写server端利用ATL COM AppWizard开发出 CalcSvr.dll

第一步:打开VC++6.0选择 ATL COM AppWizard ,然后在project名处输入 CalcSvr  文件夹我们这里选择  E:\Ock

点击确定。进入下一步后,所有默认,点击完毕!

第二步:開始创建我们的ATL类了,其作用就是实现.dll的功能。

点击插入->类    例如以下图:

然后,開始创建ATL类,类的类型选择 ATL Class,类名称我这里输入CCalcMath。其他不要动,点击确定。

第三步:加入类方法,看到刚才创建的那个CCalcMath类了吧?右击ICalcMath然后选择Add Method加入方法。

第四步:输入类名和类參数,注意这里类參数直接用逗号分开即可了,也不须要写成例如以下形式:

[in] int Num1, [in] int Num2, [out] int *result

这个[in],[out]是VS里面的。VS里面是设计成了复选框,直接点击的。

自己仿照上面的加法操作。将减法,乘法。除法,素数~~所有加入好。

须要注意的是,类參数时依据你自己的须要设定的。比方,你推断一个数字是不是素数,这时候仅仅须要两个參数,一个传进来,一个表示传出结果。

第五步:实现类功能。类已经所有加入好了,我们如今实现一下类功能就OK了。

点击InterfaceSupportsErrorInfo能够进入实现页面,当然,你也能够直接点击CCalcMath.cpp

第六步:分别实现 Add(), Sub(),Mul(), Div(),Sushu() 加,减,乘,除,素数这4个函数的功能:

以下是所有实现代码,事实上就加入了几行代码~~~你仅仅须要看着四个函数里面加入的内容即可了。

由于以下的代码,非常多是类向导自己主动生成的。

// CalcMath.cpp : Implementation of CCalcSvrApp and DLL registration.
#include "stdafx.h"
#include "CalcSvr.h"
#include "CalcMath.h"
#include <math.h> /////////////////////////////////////////////////////////////////////////////
// STDMETHODIMP CCalcMath::InterfaceSupportsErrorInfo(REFIID riid)
{
static const IID* arr[] =
{
&IID_ICalcMath,
}; for (int i=0;i<sizeof(arr)/sizeof(arr[0]);i++)
{
if (InlineIsEqualGUID(*arr[i],riid))
return S_OK;
}
return S_FALSE;
} STDMETHODIMP CCalcMath::Add(double Num1, double Num2, double *result)
{
// TODO: Add your implementation code here
*result = Num1 + Num2;
return S_OK;
} STDMETHODIMP CCalcMath::Sub(double Num1, double Num2, double *result)
{
// TODO: Add your implementation code here
*result = Num1 - Num2;
return S_OK;
} STDMETHODIMP CCalcMath::Mul(double Num1, double Num2, double *result)
{
// TODO: Add your implementation code here
*result = Num1 * Num2;
return S_OK;
} STDMETHODIMP CCalcMath::Div(double Num1, double Num2, double *result)
{
// TODO: Add your implementation code here
if (Num2 != 0)
*result = Num1 / Num2;
return S_OK;
} STDMETHODIMP CCalcMath::Sushu(int Num, int *result)
{
// TODO: Add your implementation code here
int i = 0; for(i = 2; i <= sqrt(Num); i++) //记得加上 #include <math.h>
{
if((Num % i) == 0)
{
*result = 0; //0表示非素数
break;
}
}
if (i > sqrt(Num))
*result = 1; //1表示是素数 return S_OK;
}

第七步:编译->链接。好了,我们的server端已经编写好了。

这时候,你在project文件夹以下的Debug里面会看到CalcSvr.dll,同一时候注意上面的路径。由于你如今要手动注冊CalcSvr.dll

第八步:注冊CalcSvr.dll。

调出dos窗体。你能够按住win+R呼出执行,或者直接在開始栏里面点击执行,然后输入 cmd

运行注冊的时候,须要注意.dll的路径

首先。进入dll所在的那个磁盘。如我的在E盘,则输入例如以下:

E:

然后进入对应的目录,cd表示进入该目录~,输入例如以下命令:

cd Ock\CalcSvr\Debug

server端。我们已经编写好了,以下我们来编写client。

这里就不用MFC了,为了简洁明了起见,直接创建一个win32控制台应用程序。win32控制台应用程序详细创建不说了。仅仅要写过Hello World的都会。

以下编写client应用程序。

第一步:win32控制台应用程序 project命名为 CalcClient,然后创建C++ Source File的时候也命名为:CalcClient.cpp。

第二步:将開始我们编写server端的时候,project文件夹下的CalcSvr.h 和 CalcSvr_i.c两个文件拷贝过来,放在clientproject文件夹以下。

第三步:在CalcClient.cpp里面。编写client详细实现代码。

#include <windows.h>
#include <iostream>
#include "CalcSvr.h"
#include "CalcSvr_i.c"
using namespace std; template< class T > //函数模板
void show( T t )
{
cout << t << endl;
} void DCOM()
{
double data = 0.0;
int flag = 0;
double a = 0.0, b = 0.0;
//初始化COM库
HRESULT hr=::CoInitialize(NULL); //用来告诉Windows以单线程的方式创建COM对象
if( !SUCCEEDED( hr ) )
{
show( "init right failed!" );
}
ICalcMath *plRF = NULL;
hr=::CoCreateInstance(CLSID_CalcMath,NULL,CLSCTX_INPROC_SERVER,IID_ICalcMath,(LPVOID *)&plRF);//用指定的类标识符在本地创建一个未初始化的COM对象
if(FAILED(hr))
{
cout<<"创建组件实例失败"<<endl;
}
//调用方法
cout<<"请选择:1.加;2.减。3.乘;4.除;5.推断素数(注:1表示素数;0表示非素数);0.退出\n__\b\b";
cin>>flag;
while(flag!=0)
{
switch(flag)
{
case 1:
{
cout<<"请输入要相加的两个数字:" ;
cin>>a>>b;
plRF->Add(a,b,&data);
cout<<"计算结果为:";
show( data );
break;
}
case 2:
{
cout<<"请输入要相减的两个数字:" ;
cin>>a>>b;
plRF->Sub(a,b,&data);
cout<<"计算结果为:";
show( data );
break;
}
case 3:
{
cout<<"请输入要相乘的两个数字:" ;
cin>>a>>b;
plRF->Mul(a,b,&data);
cout<<"计算结果为:";
show( data );
break;
}
case 4:
{
cout<<"请输入要相除的两个数字:" ;
cin>>a>>b;
if (b-0<1e-5 || 0-b>1e-5)
{
cout<<"除数不能为0\n";
break;
}
plRF->Div(a,b,&data);
printf("计算结果为:");
show( data );
break;
}
case 5:
{
cout<<"请输入要进行推断的数字:" ;
cin>>a;
plRF->Sushu((int)a,(int *)&data);
cout<<"推断结果为:";
show( data );
break;
}
case 0:break;
default:break;
}
system("pause");
system("cls");
cout<<"请选择:1.加;2.减;3.乘。4.除;5.推断素数(注:1表示素数;0表示非素数);0.退出\n__\b\b";
cin>>flag;
}
plRF->Release();
CoUninitialize(); //CoUninitialize关闭当前线程的COM库,卸载线程载入的全部dll,释放不论什么其它的资源,关闭在线程上维护全部的RPC连接。
}
int main()
{
DCOM(); return 0;
}
/*
ps:
//#define _WIN32_DCOM //#if (_WIN32_WINNT >= 0x0400 ) || defined(_WIN32_DCOM) // DCOM
*/

第四步:编译->链接->执行。

附带说明一下关于.dll的自己主动注冊。

问:既然,能够进入CMD手动注冊dll,那么为什么我们不实现自己主动注冊呢?

我的.dll被我复制到例如以下文件夹里面了~~以下的说明所有是针对这一文件夹路径的。

答:事实上自己主动注冊很easy,在主函数里面调用一条例如以下DOS命令即可了:

 system("regsvr32 E:\\Ock\\CalcClient\\CalcSvr.dll /s"); 
说明:上面的红色字体是路径(注意'\\',双斜杠)。 /s表示不弹出注冊成功与否的提示。

(曾经手动注冊的时候,注冊成功不是还弹出一个对话框吗?/s就是屏蔽这个的)。

问:可是,这里又遇到一个问题了,将程序移植到别人的电脑上面,路径就不同了~~~
答:这个问题不难解决呀,曾经不是学过int main()函数传參吗?还记得吧?看看以下这个能想起来么?
int main(int argc, char *argv[])
{
puts(argv[0]); return 0;
}

对了。argv[0]打印出来的就是.exe的运行路径了。那这样就好办了。


以下就是改好后的代码了~~~实现了自己主动注冊.dll
void regsvr32(char *path)
{
int i = 0, count = 0;
char buf[100], regPath[100]; //找到倒数第二个 \ 反斜杠的位置
for (i = strlen(path); i >= 0; --i)
{
if (path[i] == '\\')
{
++count;
if (count == 2)
break;
}
}
strncpy(buf,path,i); //拷贝出 路径的前 i 位
buf[i] = '\0';
sprintf(regPath,"regsvr32 %s\\CalcSvr.dll /s",buf); //格式化路径
system(regPath); //运行注冊 .dll
} int main(int argc, char *argv[])
{ regsvr32(argv[0]); //自己主动注冊
DCOM(); return 0;
}

假设不清楚main參数问题,我的另外一篇博客:http://blog.csdn.net/qingdujun/article/details/24844597 第10个问题中有说明~~~

參考文献:qipanlimingCSDN博客的专栏。COM/DCOM开发练习之进程内组件实例。http://blog.csdn.net/lican19911221/article/details/7459983。2014年5月13日

COM/DCOM开发练习之进程内组件实例的更多相关文章

  1. 【转载】进程内COM与进程外COM

    原文:http://www.cnblogs.com/jyz/archive/2009/03/08/1406229.html 1.进程内和进程外Com COM/DCOM 组件可以在DLL 或EXE 文档 ...

  2. 进程内COM与进程外COM

    1.进程内和进程外Com COM/DCOM 组件可以在DLL 或EXE 文档中实现. 其中在 DLL 中实现的COM/DCOM组件称为 In-Process Server,因为这些组件是加载到使用它们 ...

  3. 在VC6.0下如何调用Delphi5.0开发的进程内COM

    因为本人的语言水平很差,考大学时150的总分,我考了个60分.外语也是,初中及格过一次,会考及格过一次.其它的时间好像从没有及格过.所以我不写文章,因我一百字的文章给我写,至少要出八九个错别字.哈哈… ...

  4. Android开发-API指南-进程与线程

    Processes and Threads 英文原文:http://developer.android.com/guide/components/processes-and-threads.html ...

  5. android开发3:四大基本组件的介绍与生命周期

    android开发3:四大基本组件的介绍与生命周期 Android四大基本组件分别是Activity,Service服务,Content Provider内容提供者,BroadcastReceiver ...

  6. 自己实现一个Electron跨进程消息组件

    我们知道开发Electron应用,难免要涉及到跨进程通信,以前Electron内置了remote模块,极大的简化了跨进程通信的开发工作,但这也带来了很多问题,具体的细节请参与我之前写的文章: http ...

  7. Android源码——Activity进程内启动

    进程内启动Activity MainActivity组件向ActivityManagerService发送一个启动SubActivityInProcess组件的进程间通信请求: ActivityMan ...

  8. [原创]上海好买基金招高级Java技术经理/运维主管/高级无线客户端开发等职位(内推)

    [原创]上海好买基金招高级Java技术经理/运维主管/高级无线客户端开发等职位(内推) 内部推荐职位 高级JAVA技术经理: 岗位职责: 负责项目管理(技术方向),按照产品开发流 ,带领研发团队,制定 ...

  9. .NET跨平台实践:再谈用C#开发Linux守护进程

    Linux守护进程是Linux的后台服务进程,相当于Windows服务进程,对于为Linux开发服务程序的朋友来说,Linux守护进程相关技术是必不可少的,因为这个技术不仅仅是开发守护进程,还可以拓展 ...

随机推荐

  1. Python函数:对变量赋值,变量即局部

    b = 6 def f2(a): print(a) print(b) b = 9 UnboundLocalError: local variable 'b' referenced before ass ...

  2. SQL存储过程相关信息查看

    --1.查看所有存储过程与函数      exec sp_stored_procedures     或者      select * from dbo.sysobjects where OBJECT ...

  3. .NetCore读取配置Json文件到类中并在程序使用

    ConfigurationBuilder 这个类提供了配置绑定,在dnc中 Program中WebHost提供了默认的绑定(appsettings文件) 如果我们需要加载我们自己的json配置文件怎么 ...

  4. mongo学习链接

    http://www.runoob.com/mongodb/mongodb-create-database.html

  5. Redis(一)Redis简述

    一 Redis 概念 Redis是一个开源的使用ANSI C语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value数据库,和Memcached类似,它支持存储的value类型相对更多,包 ...

  6. day1作业二:多级菜单操作

    作业二:多级菜单 (1)三级菜单 (2)可以次选择进入各子菜单 (3)所需新知识点:列表.字典 要求:输入back返回上一层,输入quit退出整个程序 思路: (1)首先定义好三级菜单字典: (2)提 ...

  7. js 格式化时间,可定义格式

    var format = function (time, format) { var t = new Date(time); var tf = function (i) { return (i < ...

  8. thinkphp _complex 复合查询 where多个子组实现

    SELECT * FROM `user` WHERE ( `mobile` = '13824653465' OR `nickname` = 'evan' OR `openid` = '14545-fd ...

  9. 再议js的传递和深复制

    病理 基本类型的传递就是按值传递,比如说 var a = 1; var b = a; b = 3; console.log(a,b);//1,3 很明显,a的值并未因为b的值改变而变化,这是因为a只是 ...

  10. 浅谈Comparable与Comparator的区别

    平时进行自定义排序一直使用实现Comparable接口,一段时间后操作的时候居然发现有了个Comparator接口 上网差了些资料,总结笔记一下. 基本原理就是比较,底层是二叉树 比如是3,6,5,1 ...