两个程序同时运行的情况下?
请具体一点
请具体一点
解决方案 »
- 求教 怎么让程序使用socks5代理连接网络
- Visual studio 制作安装包的问题
- 请教远生视频会议系统问题(有源码)
- 帮我解释一句话
- ftp client 出现425 :can't open data connnecion ,请问 造成错误的原因是什么,我用的是socket api 组塞式的数据连接
- 200分,求救在线等待。。。关于如何关闭不是自己程序打开的RAS连接
- 哪里有boundschecker 6.5 for vc下载
- 除了OnTime()外还有什么控制程序运行时间的函数.
- 一个关于BMP的问题,大虾请进:
- 帮帮忙!能给我一个IPX/SPX的控件或是其它工具吗?
- 如何把数字显示在CEdit中?
- 请问怎么得到一个Sheet的指针?
* 函数介绍:打开实时库
* 输入参数:无
* 输出参数:无
* 返回值 :成功:TRUE,失败:FALSE
* 备注:一个应用程序应该只调用一次,在app类的InitInstance中调用
*/REALDB_API BOOL realdb_Open(BOOL bInitRealDB)
{
dbinit(); if (g_bInitialized)
{
UnmapViewOfFile((void*)g_pRealDB);
CloseHandle(g_hFileMapping);
} g_hFileMapping = CreateFileMapping((HANDLE)0xffffffff,NULL,
PAGE_READWRITE,0,sizeof(REALDB),"sms SCADA Real Time DB");
if (g_hFileMapping == NULL)
{
//TRACE("Error: 创建用于实时数据库的内存映射文件对象失败!");
return FALSE;
}
else
{
g_pRealDB = (REALDB*)MapViewOfFile(g_hFileMapping,
FILE_MAP_WRITE,0,0,0);
}
g_hLogFile = CreateFile("scada.log",GENERIC_WRITE | GENERIC_READ,
FILE_SHARE_WRITE | FILE_SHARE_READ,
NULL,OPEN_ALWAYS,FILE_FLAG_WRITE_THROUGH,NULL); if (INVALID_HANDLE_VALUE == g_hLogFile)
{
//TRACE ("Error: 建立log文件失败");
return FALSE;
} InitCritical(0); const char APP_NAME[] = "DBSetttins";
GetPrivateProfileString(APP_NAME,"DBType","",
g_pRealDB->m_reserved.m_pszDBType,NAME_MAX_LEN,FILE_INI);
GetPrivateProfileString(APP_NAME,"DBSrvName","",
g_pRealDB->m_reserved.m_pszDBSrvName ,NAME_MAX_LEN,FILE_INI);
GetPrivateProfileString(APP_NAME,"DBName","",
g_pRealDB->m_reserved.m_pszParaDBName ,NAME_MAX_LEN,FILE_INI);
GetPrivateProfileString(APP_NAME,"DBLogin","",
g_pRealDB->m_reserved.m_pszDBLogin ,NAME_MAX_LEN,FILE_INI);
GetPrivateProfileString(APP_NAME,"DBPassword","",
g_pRealDB->m_reserved.m_pszDBPassword ,NAME_MAX_LEN,FILE_INI); g_bInitialized = TRUE; if (bInitRealDB)
{
if (stricmp(realdb_GetRealDB()->m_reserved.m_pszDBType,DBTYPE_SQLSERVER) == 0)
{
if (!realdb_InitRealDBFromSQLSrv())
{
LOG ("Error: 从%s初始化实时库中的参数配置失败!",DBTYPE_SQLSERVER);
return FALSE;
}
}
else if (stricmp(realdb_GetRealDB()->m_reserved.m_pszDBType,DBTYPE_ORACLE) == 0)
{
if (!realdb_InitRealDBFromOracle())
{
LOG ("Error: 从%s初始化实时库中的参数配置失败!",DBTYPE_ORACLE);
return FALSE;
}
}
else
{
return FALSE;
}
} return TRUE;
}
DDE还有网络功能。使用过Win311 For Workgroup的人大概都还记得,它自带一个非常吸引人的小程序“Chat”,能使两台计算机通过网络实时交谈,这在当时几乎是一项创举。可是很少有人知道“Chat”使用的是一种特殊的DDE,即NetDDE。它的基本工作原理仍然是DDE,但它能使一台计算机向在同一个网络中的另一台终端发消
息,而不像普通DDE 只能局限在同一台机器上。与其它的数据交换方式相比,DDE已不够先进,而且Microsoft也不再积极支持DDE,所以它的前景不被看好。
3、WM_COPYDATA Windows消息WM_COPYDATA功能强大,知之者却甚少。它的确切定义是:一个应用程序向另一个应用程序传递数据时所发出的消息。众所周知,Windows 在很大程度上依赖于消息机制,那么我们为什么不把数据放在消息中一起发送出去呢?这样,我们只要调用SendMessage(),以对方窗体的句柄作为第一个参数,以含有指向实际数
据的指针结构的地址作为第二个参数,就可以把整个数据块当作消息发向另一个应用程序。也许有的程序员会说,一个自定义的消息同样可以完成这样的工作。他只说对了一半。自定义消息的确可以发送到其它的应用程序上去,但其原始数据所在的内存区只有在发送过程中才是合法的,若在其它模块中存取该内存区就会导致“Access
Violation”。而使用WM_COPYDATA 则不存在这个问题,因为Windows 本身会替你处理这一切。深入钻研下去,你会发现它其实是先创建了一个文件映射的对象,将发送方的原始数据先拷贝至映射文件,然后再在接收方对这个映射文件打开一个“视图”。WM_COPYDATA 的不足之处在于:必须要有一个窗体来接收消息和数据(缺乏灵活性),数据在使用之前先得拷贝到一个映射文件(浪费资源)。 4、NetBIOS 从80年代开始,NetBIOS就是开发Client/Server类程序的标准接口。而当时其它的解决方案几乎都是从UNIX系统上发展而来的。当然,对于今天的用户而言,NetBIOS已不是唯一的选择,他们可以从各种方案中择优选取。在Windows 95和NT平台上,可以通过API函数Netbios()来调用NetBIOS功能。 NetBIOS兼容以下几种低层协议:NetBEUI协议(NBF)、NWLinkNetBIOS(NWNBlink)、NetBIOS over TCP/IP(NetBT)。NetBIOS的优点是速度较快,缺点是可靠性较差。 5、标准的Sockets和WinSock Sockets是15年前在UNIX系统上提出来的,一开始主要是用于本地通讯,但很快就应用在Client/Server体系上了。它的内核很简单:你可以将一个Socket看作是一个双向的节点,一个应用程序可以通过它先与另一个程序建立连接(建立在一个双方都认可的端上,以便于区分同时运行的几个通讯线程),然后就可以彼此交换数据了。 微软公司在其基础上创建了WinSock,专门用于Windows接口,与Socket完全兼容。近年来基于TCP/IP协议的网络大行其道,Socket也随之获得了更加广泛的应用。 如今,Sockets已在Internet上获得了最广泛的应用,主要是因为它的可移植性好: Socket应用程序无论在任何平台间都能互相进行通讯(不管是PC机还是Macintosh,也不管是Windows平台还是UNIX平台)。而最新推出的WinSock 2.0,已不仅仅只基于TCP/IP协议,还可基于其它几种传输协议(如IPX)。 Socket的缺陷是它工作在通讯的低层,所以实现起来较为复杂(如果是Win 95或NT平台,则不存在这个问题,微软公司提供了相应的控件)。但是,如果要在多个平台间互相通讯,则Socket是不二之选。 6、Mailslot和Pipe Mailslot和Pipe有很多相同点,即都是高层的、基于内存的通讯系统。Mailslot由Server端创建,代码如下: myMailslot = CreateMailslot(pSlotName, 0, MAILSLOT_WAIT_FOREVER, NULL); 然后,Client端就可以像打开文件一样打开这个Mailslot,再通过API函数WriteFile()来将数据写入到消息队列中。 与此相类似,Pipe也是由Server端创建的,代码如下: pipe = CreateNamedPipe(pPipeName, PIPE_ACCESS_DUPLEX, PIPE_TYPE_MESSAGE |PIPE_READMODE_MESSAGE | PIPE_WAIT, PIPE_UNLIMITED_INSTANCES, BUFSIZE,BUFSIZE, 20000, NULL); 接着,Client端就可以通过一般的文件API读写数据,代码如下: pipe = CreateFile(pPipeName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING,0,NULL); success = WriteFile(pipe, pMessage, strlen(pMessage) + 1, &bytesWritten, NULL); Pipe分两种,即命名Pipe和匿名Pipe。匿名Pipe是以句柄而不是以名字来进行标识的,因而也就限制了它只能在同一台机器上通讯,而不能应用于网络。命名Pipe则提供了以名字来进行标识,所以能在网络上的其它任何地方打开它。需要注意的是:命名Pipe只能在 Win NT而不是Win 95上创建(即Server端),Client端则可以是任
意平台。 Pipe既可以单向通讯也可以双向通讯,Mailslot则只能单向通讯。Client端可以发送消息给mailslot,但不能接收消息;如果你想要接收消息,则只能创建一个新的mailslot。但mailslot有一个很大的优点:它支持数据广播。也就是说,若Client端发送一条消息,则整个网络中的同名对象都能收到。这是因为mailslot的名字的作用
域只是在本台机器上,所以可在不同机器上创建同名的mailslot,当Client端发来消息时则每一台机上的mailslot都得到了该消息的一份拷贝,并在本机上作出相应的反应。Mailslot的最大缺陷是不可靠,因为它的数据是以数据报格式来传送的,网络错误或负荷过重都会导致数据丢失。 Pipe则较为可靠,但它不能广播。所以,如果你不需要进行广播,则Pipe是更好选择。选择Mailslot,则要对此后遇到的麻烦有充分的准备。 7、剪贴板(Clipboard) 大家对Windows的剪贴板应该不会感到陌生吧?它的出现就是为了实现应用程序间的互相数据交换。Windows提供了一系列的API函数来让应用程序安全地打开剪贴板,读写其中的数据。 剪贴板的缺陷也是显而易见的:当有新的数据放在剪贴板上时,则先前的数据就会被冲掉,而在Windows中用到剪贴板的时候又实在太多了,所以这种方式用于程序间的通讯显然不够安全。 8、COM和DCOM 公共对象模式(COM)是一种协议,它建立了一个软件模块同另一个软件模块之间的连接,然后将其描述出来。当这种连接建立起来之后,则两个模块之间就可以通过称为“接口”的机制来进行通讯。COM可以用不同的语言(VB、VC、Delphi)进行编制,又能被其它语言编写的程序所使用,并且不用管通讯双方实际所处的位置(是
在同一台机上,还是在同一个网络上的不同机上)。事实上,Internet上有大量的COM控件可供人们下载使用,其中有相当一部分就是用于应用程序间的相互通讯(甚至是Internet程序间的通讯),硬盘上能够看到的大量的.ocx文件其实就是一种COM。 COM对象为外部调用提供了一个标准的界面,COM Client通过创建COM Server的一个实例获得指针,转向所需的函数定义处并执行相应的程序。讲得通俗一点,也就是先正确定义好COM对象的属性(Property),再执行相应的方法(Method)。 DCOM(分布式公共对象模式)是COM在网络上的一种扩展,它通过把分布式对象间的通讯变成一个实体来实现通讯。 COM如今被微软公司大力提倡,最著名的有OLE、ActiveX、DirectX和Win95、WinNT的外壳。由于微软公司目前在软件业处于霸主地位,所以COM很有可能成为将来的业界标准,其前途也较为看好。但COM的庞大也会使一些经验丰富的程序员望而却步,他们宁可自己多写一些代码以使整个程序更为简洁、有效而不愿使用COM。 以上介绍的只是现今流行的一些通讯方法,它们各有各的优缺点,没有哪一种是万能的。因此,只有在合适的场合采用合适的方案,才是最好的解决办法。