套接字(socket)当今已成为最流行的网络通信应用程序接口。套接字最初是由加利福尼亚大学Berkeley分校为Unix操作系统开发的网络通信接口,后来它又被移植到DOS与Windows系统,特别是近几年来互联网络在全世界范围内被广泛普及并且增长迅猛,进一步奠定了它在网络通信程序开发领域的主宰地位。跨系统网络互联的呼声在我国也越来越高。近来,在开发的银行与证券联网系统中,我们尝试做了由Unix Sockets到Windows Sockets通信应用程序的移植,有所体会,望能将移植过程中应该注意的几点问题与热心于这方面工作的人们共同作以探讨,以期能起到抛砖引玉的作用。

1. 修改头文件的定义

Unix Sockets应用程序中包含头文件的语句为#include<sys/socket.h>,而在Windows Sockets应用程序中相应语句应为 #include<winsock.h>#include<winsock2.h>

Windows Sockets的实现由两部分组成,即开发组件与运行组件。开发组件是供程序员开发Windows Sockets 应用程序使用的,它包括Windows Sockets应用程序接口函数库、头文件以及一些介绍Windows Sockets实现的文档,其中头文件winsock.h中包含了Windows Sockets 实现所定义的宏、常数值、数据结构和函数调用接口原型。winsock.h是编写Windows Sockets应用程序必须包含的头文件。

那么winsock和winsock2有什么区别呢?如果当winsock.h在winsock2.h之前包含时就会出错,因为两个头文件不能很好的共存;但如果winsock2.h在winsock.h之前包含,那么就不会出错,因为winsock2.h定义了_WINSOCKAPI_,阻止了编译器去处理后面的winsock.h。

2. WinSockets DLL初始化与资源释放

在Windows Sockets的接口函数中,提供了WSAStartup()WSACleanup()两个函数,用于在程序开始时初始化Windows Sockets DLL和程序结束时释放Windows Sockets DLL资源,它们是编写Windows Sockets应用程序时必须使用的两个函数。

3. 将套接字的类型由int改为SOCKET

在Unix系统中,套接字类型定义为int,而在Windows系统中,套接字类型被定义为SOCKET,即unsigned int型。

4. 错误码的获取与设置

在Unix系统中,获取、设置错误码使用全局变量errno,而在Windows系统中则应将其改为用函数WSAGetLastError()WSASetLastError()。Windows Sockets 为了与多线程环境兼容,提供了两个出错处理函数WSAGetLastError()与WSASetLastError()来获取与设置当前线程的最近错误号,而不再使用Unix系统中的全局变量errno和h_errno。

5. 关闭套接字

在Unix系统中使用close()函数来关闭套接字;

而在Windows系统中则用closesocket()函数。

6. 对套接字的控制

在Unix系统中,使用ioctl()函数fcntl()函数实现对套接字的控制,而在Windows系统中则应使用ioctlsocket()函数

7. getsockopt()和setsockopt()函数的处理

用来设置套接字选项。在Unix Sockets和Windows Sockets中对这两个函数提供的支持不同。在Windows Sockets中getsockopt()setsockopt()不支持的Berkeley Sockets选项有:

l SO_RCVLOWAT  接受低潮标志

l SO_RCVTIMEO  接受超时

l SO_SNDLOWAT  发送低潮标志

l SO_SNDTIMEO  发送超时

l IP_OPTIONS  取得IP头中的选项

l TCP_MAXSEG  取得TCP最大尺寸

l SO_ACCEPTCONN  套接字正在监听

l SO_ERROR  取错误状态并清除

l SO_TYPE  套接字类型

使用不支持的选项将返回错误码WSAENOPROTOOPT,它由WSAGetLastError()函数返回。由于提供的支持不同,进行移植时若有必要应修改相应的源程序。

8. 使用宏FD_XXX来控制fd_set结构

由于Windows Sockets 某些函数在接口上虽然与Unix Sockets一致,但是它们的内部实现却不一样,例如,在函数select()的参数中,Unix Sockets实现套接字集合使用的是位掩码,但在Windows Sockets中却是使用一个SOCKET的数组。虽然套接字的集合仍由fd_set类型表示,但在Unix Sockets 源文件中直接修改fd_set结构的代码在Windows Sockets环境下将不能正常工作。因此,在进行移植时应将源程序中对结构fd_set的直接修改改为通过使用FD_XXX宏来修改。

9. 应在WinSockets程序中尽量使用WSA宏

在Windows Sockets的头文件中定义了许多以WSA开头的宏,在程序中使用这些宏,可以大大地增加程序的可读性。例如:

errno = WSAGetLastError();

if (errno = WSAEWOULDBLOCK){

……;

}

这些宏是Windows Sockets就应用程序实现过程中可能会出现的错误,为便于程序的读写而专门做出的扩充,它是Windows Sockets应用程序编写人员开发程序时共同遵循的规范,因此应尽量使用这些宏,以增加程序的可读性。

Windows Sockets就应用程序实现过程中可能会出现的错误,为便于程序的读写而专门做出的扩充,它是Windows Sockets应用程序编写人员开发程序时共同遵循的规范,因此应尽量使用这些宏,以增加程序的可读性。

10. 阻塞调用的处理

Windows是非抢先多任务环境,各任务之间的切换是通过消息驱动的,如果一个应用程序不能主动放弃其控制权,别的应用程序就不能够执行,这一点与Unix操作系统有着本质的区别。对于从Unix Sockets环境中移植来的应用程序来说,阻塞问题必须考虑。

为解决阻塞问题,Windows Sockets特增设了如下几个阻塞处理函数:

是非抢先多任务环境,各任务之间的切换是通过消息驱动的,如果一个应用程序不能主动放弃其控制权,别的应用程序就不能够执行,这一点与Unix操作系统有着本质的区别。对于从Unix Sockets环境中移植来的应用程序来说,阻塞问题必须考虑。

为解决阻塞问题,Windows Sockets特增设了如下几个阻塞处理函数:

l WSAIsBlocking() 检测阻塞调用是否正在进行

l WSACancelBlockingCall() 取消一个正在进行的阻塞调用

l WSASetBlocking() 设置自己的阻塞处理例程

l WSAUnhookBlockingHook() 恢复默认的阻塞处理例程

为不影响原来环境中的阻塞处理例程,在安装自己的阻塞处理例程时,应注意保存返回的先前安装的阻塞处理例程的程序实例指针,并在处理结束后恢复。

11. 尽量将阻塞调用改为基于消息的异步操作

在Windows Sockets网络程序设计中,尽管它允许阻塞操作,但是一个阻塞可能阻塞整个Windows环境,而在Unix Sockets程序中,套接字的默认操作模式却是阻塞的。Windows Sockets为了支持Windows消息驱动机制,对网络事件采用了基于消息的异步存取策略,较好的解决了阻塞问题。为此,建议最好将源程序中的阻塞调用改为基于消息的异步操作。

Windows Sockets为实现异步存取操作增设了如下的有关函数:

l WSAAsyncSelect() 标准Berkeley函数select()的异步版本

l WSAAsyncGetXByY() 标准Berkeley函数getXbyY()的异步版本

l WSACancelAsyncRequest() 取消函数WSAAsyncGetXByY()执行中的实例

由Unix Sockets应用程序向Windows Sockets的移植是一件比较繁琐的工作,在移植中除了以上Sockets具体实现直接相关的几点外,还应注意程序开发环境的的差异,这时程序上也应作一些相应的调整,例如,在Windows应用程序中有关的一些指针都应设为FAR类型等,这里就不再一一赘述。

将Socket应用程序从Unix向Windows移植中应注意的几点问题的更多相关文章

  1. 用java写的一个程序,可以调用windows系统中自带的各种工具,例如截图,便签等

    由于图片资源没有上传,所以运行后不会有图片,感兴趣的同学可以@我,我打包上传, package SmallPrograme; import java.awt.*; import java.awt.ev ...

  2. C/C++ Windows移植到Linux

    近期写了有关Socket的程序,需要从windows移植到linux.现把有用的东东收集整理记录下来. 1.头文件windows下winsock.h或winsock2.h:linux下netinet/ ...

  3. 将WCF寄宿在托管的Windows服务中

    在我之前的一篇博客中我介绍了如何发布WCF服务并将该服务寄宿于IIS上,今天我再来介绍一种方式,就是将WCF服务寄宿在Windows服务中,这样做有什么好处呢?当然可以省去部署IIS等一系列的问题,能 ...

  4. Socket程序从windows移植到linux下需要注意的

    )头文件 windows下winsock.h或winsock2.h linux下netinet/in.h(大部分都在这儿),unistd.h(close函数在这儿),sys/socket.h(在in. ...

  5. 操作系统Unix、Windows、Mac OS、Linux的故事

    电脑,计算机已经成为我们生活中必不可少的一部分.无论是大型的超级计算机,还是手机般小巧的终端设备,都跑着一个操作系统.正是这些操作系统,让那些硬件和芯片得意组合起来,让那些软件得以运行,让我们的世界在 ...

  6. 汉高澳大利亚sinox为什么不能下载源代码,因为sinox执行unix/linux/windows规划

    中国用户下载真正的澳大利亚sinox说完后sinox没有下载源代码. 这意味着,类似linux如下载linux 开源安装. 要知道.sinox并非linux. 首先,sinox是商业操作系统,就像 w ...

  7. [Oracle] UNIX与Windows 2000上Oracle的差异(III)

    作者:Ian Adam & David Stien, SAIC Ltd 日期:19-Dec-2003 出处:http://www.dbanotes.net翻译:Fenng ORACLE 的安装 ...

  8. [Oracle]UNIX与Windows 2000上Oracle的差异(I)

    作者:Ian Adam & David Stien, SAIC Ltd 日期:19-Dec-2003 出处:http://www.dbanotes.net翻译:Fenng ---------- ...

  9. 操作系统 - unix和windows下进程异同

    在UNIX系统中,只有一个系统调用可以用来创建新进程:fork.这个系统调用会创建一个与调用进程相同的副本.在调用了fork之后,这两个进程(父进程和子进程)拥有相同的存储映像.同样的环境字符串和同样 ...

随机推荐

  1. Winfrom 开源组件Control.FirefoxDialog使用

    1. 如果窗体是以模式窗体方式打开的,会出现点了应用,窗体就立马关闭.此时可能别的设置需要一块设置,这种就存在问题. var form1 = new Form1(); form1.ShowDialog ...

  2. 利用spring的ApplicationListener实现springmvc容器的初始化加载

    1.我们在使用springmvc进行配置的时候一般初始化都是在web.xml里面进行的,但是自己在使用的时候经常会测试一些数据,这样就只有加载spring-mvc.xml的配置文件来实现.为了更方便的 ...

  3. UE4关于Oculus Rift (VR)开发忠告

    转自:http://blog.csdn.net/cartzhang/article/details/42493843 UE4虚拟现实 实现的注意事项 https://docs.unrealengine ...

  4. Selenium2+python自动化63-简易项目搭建

    前言 到unittest这里基本上可以搭建一个简易的项目框架了,我们可以用一条run_main.py脚本去控制执行所有的用例,并生成报告,发送邮件一系列的动作 一.新建工程 1.打开pycharm左上 ...

  5. 并发服务器和HTTP协议

    单进程服务器 1. 完成一个简单的TCP服务器 from socket import * serSocket = socket(AF_INET, SOCK_STREAM) # 重复使用绑定的信息 se ...

  6. Input设置只读属性

    input设置为只读一般用于查看详情: 格式为: (1)单纯的input框 <div class="form-group"> <label for="c ...

  7. IOS上架截屏 屏幕快照

    IOS上架截屏,屏幕快照,4种屏幕尺寸,每种尺寸5张软件功能截图. 大小等于对应设备的屏幕的像素大小.使用模拟器,command +s截图就可以了虚拟机里的手机截屏就保存在mac 桌面上了.jpg,p ...

  8. MyBatis ehcache二级缓存

    ehcache二级缓存的开启步骤: 1.导入jar 2.在映射文件中指定用的哪个缓存 3.加一个配置文件,这个配置文件在ehcache jar包中就有 使增删改对二级缓存不刷新: 对一级缓存没有用的, ...

  9. 手工kill掉VNC进程的故障处理

    1.模拟Kill掉已经启动的VNC服务 1)启动桌面1的服务 [root@testdb ~]# vncserver :1 New 'testdb:1 (root)' desktop is testdb ...

  10. 6-关于#include<bits/stdc++.h>

    万能头文件#include<bits/stdc++.h> (转载)   最近在打cf时赛后翻阅别人的代码总是会发现一个陌生而奇怪的头文件#include<bits/stdc++.h& ...