网络编程Tcp的客户端和服务器端

本文首先对基础知识做一下介绍,后面会给出示例程序,便于分析理解。大部分资料来源于网络,自己只是对其做了个整理。

 

    首先我们先来回顾下网络模型

    1. ISO(internet Standard Organization 国际标准组织)对OSI(Open SYstem Interconnect开放互联

模型) 七层网络模型的定义:
物理层(Physical layer) 链路层(Data link layer) 网络层(Network layer) 
传输层(Transport layer) 会话层(Session layer) 表示层(Presentation layer) 

应用层(Application layer)
    2. TCP/IP模型分为四层:
  网络接口层(Network access layer), 互联网层(internet layer),传输层((Transport layer),

  和应用层(Application layer).

 

   实际中我们应用的是TCP/IP网络模型,而接下来讨论的windows socket 也是基于本模型的。所以有必要

先了解下TCP/IP 体系结构与特点。

 

TCP/IP 体系结构与特点

1、TCP/IP体系结构


TCP/IP协议实际上就是在物理网上的一组完整的网络协议。其中TCP是提供传输层服务,而IP则是提供网络层服务。TCP/IP包括以下协议:(结构如图1.1)
[网络编程] <wbr>socket <wbr>编程浅谈(一)
(图1.1)

IP: 网间协议(Internet Protocol) 负责主机间数据的路由和网络上数据的存储。同时为ICMP,TCP,UDP提供分组发送服务。用户进程通常不需要涉及这一层。
ARP: 地址解析协议(Address Resolution Protocol)
此协议将网络地址映射到硬件地址。
RARP: 反向地址解析协议(Reverse Address Resolution Protocol)
此协议将硬件地址映射到网络地址
ICMP: 网间报文控制协议(Internet Control Message Protocol)
此协议处理信关和主机的差错和传送控制。
TCP: 传送控制协议(Transmission Control Protocol)
这是一种提供给用户进程的可靠的全双工字节流面向连接的协议。它要为用户进程提供虚电路服务,并为数据可靠传输建立检查。(注:大多数网络用户程序使用TCP)
UDP: 用户数据报协议(User Datagram Protocol)
这是提供给用户进程的无连接协议,用于传送数据而不执行正确性检查。
FTP: 文件传输协议(File Transfer Protocol)
允许用户以文件操作的方式(文件的增、删、改、查、传送等)与另一主机相互通信。
SMTP: 简单邮件传送协议(Simple Mail Transfer Protocol)
SMTP协议为系统之间传送电子邮件。
TELNET:终端协议(Telnet Terminal Procotol)
允许用户以虚终端方式访问远程主机
HTTP: 超文本传输协议(Hypertext Transfer Procotol)
TFTP: 简单文件传输协议(Trivial File Transfer Protocol)

2、TCP/IP特点
TCP/IP协议的核心部分是传输层协议(TCP、UDP),网络层协议(IP)和物理接口层,这三层通常是在操作系统内核中实现。因此用户一般不涉及。编程时,编程界面有两种形式:一、是由内核心直接提供的系统调用;二、使用以库函数方式提供的各种函数。前者为核内实现,后者为核外实现。用户服务要通过核外的应用程序才能实现,所以要使用套接字(socket)来实现。
图1.2是TCP/IP协议核心与应用程序关系图。
[网络编程] <wbr>socket <wbr>编程浅谈(一)
(图1.2)

 

涉及到的专用术语
1、套接字
它是网络的基本构件。它是可以被命名和寻址的通信端点,使用中的每一个套接字都有其类型和一个与之相连听进程。套接字存在通信区域(通信区域又称地址簇)中。套接字只与同一区域中的套接字交换数据(跨区域时,需要执行某和转换进程才能实现)。WINDOWS 中的套接字只支持一个域——网际域。套接字具有类型。
WINDOWS SOCKET 1.1 版本支持两种套接字:流套接字(SOCK_STREAM)和数据报套接字(SOCK_DGRAM)

2、WINDOWS SOCKETS 实现
一个WINDOWS SOCKETS 实现是指实现了WINDOWS SOCKETS规范所描述的全部功能的一套软件。一般通过DLL文件来实现

3、阻塞处理例程
阻塞处理例程(blocking hook,阻塞钩子)是WINDOWS SOCKETS实现为了支持阻塞套接字函数调用而提供的一种机制。

4、多址广播(multicast,多点传送或组播)
是一种一对多的传输方式,传输发起者通过一次传输就将信息传送到一组接收者,与单点传送
(unicast)和广播(Broadcast)相对应。

 

客户机/服务器模式
在TCP/IP网络中两个进程间的相互作用的主机模式是客户机/服务器模式(Client/Server model)。该模式的建立基于以下两点:1、非对等作用;2、通信完全是异步的。客户机/服务器模式在操作过程中采取的是主动请示方式:

首先服务器方要先启动,并根据请示提供相应服务:(过程如下)
1、打开一通信通道并告知本地主机,它愿意在某一个公认地址上接收客户请求。
2、等待客户请求到达该端口。
3、接收到重复服务请求,处理该请求并发送应答信号。
4、返回第二步,等待另一客户请求
5、关闭服务器。
客户方:
1、打开一通信通道,并连接到服务器所在主机的特定端口。
2、向服务器发送服务请求报文,等待并接收应答;继续提出请求……
3、请求结束后关闭通信通道并终止。

 

基本套接字函数

1、创建套接字——socket()
功能:使用前创建一个新的套接字
格式:SOCKET PASCAL FAR socket(int af,int type,int procotol);
参数:af: 通信发生的区域
type: 要建立的套接字类型
procotol: 使用的特定协议

2、指定本地地址——bind()
功能:将套接字地址与所创建的套接字号联系起来。
格式:int PASCAL FAR bind(SOCKET s,const struct sockaddr FAR * name,int namelen);
参数:s: 是由socket()调用返回的并且未作连接的套接字描述符(套接字号)。
其它:没有错误,bind()返回0,否则SOCKET_ERROR
地址结构说明:
struct sockaddr_in
{
short sin_family;//AF_INET
u_short sin_port;//16位端口号,网络字节顺序
struct in_addr sin_addr;//32位IP地址,网络字节顺序
char sin_zero[8];//保留
}

3、建立套接字连接——connect()和accept()
功能:共同完成连接工作
格式:int PASCAL FAR connect(SOCKET s,const struct sockaddr FAR * name,int namelen);
SOCKET PASCAL FAR accept(SOCKET s,struct sockaddr FAR * name,int FAR * addrlen);
参数:同上

4、监听连接——listen()
功能:用于面向连接服务器,表明它愿意接收连接。
格式:int PASCAL FAR listen(SOCKET s, int backlog);

5、数据传输——send()与recv()
功能:数据的发送与接收
格式:int PASCAL FAR send(SOCKET s,const char FAR * buf,int len,int flags);
int PASCAL FAR recv(SOCKET s,const char FAR * buf,int len,int flags);
参数:buf:指向存有传输数据的缓冲区的指针。

6、多路复用——select()
功能:用来检测一个或多个套接字状态。
格式:int PASCAL FAR select(int nfds,fd_set FAR * readfds,fd_set FAR * writefds,
fd_set FAR * exceptfds,const struct tim FAR * timeout);
参数:readfds:指向要做读检测的指针
writefds:指向要做写检测的指针
exceptfds:指向要检测是否出错的指针
timeout:最大等待时间

7、关闭套接字——closesocket()
功能:关闭套接字s
格式:BOOL PASCAL FAR closesocket(SOCKET s);

 

典型过程图
2.1 面向连接的套接字的系统调用时序图
[网络编程] <wbr>socket <wbr>编程浅谈(一)


2.2 无连接协议的套接字调用时序图
[网络编程] <wbr>socket <wbr>编程浅谈(一)


2.3 面向连接的应用程序流程图
[网络编程] <wbr>socket <wbr>编程浅谈(一)

好了接下来我们编写一个简单的程序来看看运作过程

 

例子程序 

Visual C++ 2005 调试通过,请在工程的库路径里包含文件Ws2_32.lib

1.TCP客户端

client.cpp

-----------------------------------------------------------

#include <vld.h>
#include <stdio.h>
//#include <sys/socket.h>
//#include <netinet/in.h>
//#include <arpa/inet.h>
//#include <netdb.h>
#include <WinSock2.h>
#include <string.h>
#include <stdlib.h>
char * host_name="127.0.0.1";
int port=10001;
void main(int argc,char *argv[])
{
    char buf[4096];
    char message[1024];
    int socket_descriptor;
    struct sockaddr_in pin;
    struct hostent *server_host_name;
    char *message_str="hello,network\0";
    WSADATA wsaData;

    if(argc<2) {
        printf("send default srting.\n");
    } else {
        message_str=argv[1];
    }


    if(WSAStartup(MAKEWORD(2,1),&wsaData)) {
        printf("Winsock initializing fail\n");
        WSACleanup();
        return ;
    }

    if((server_host_name=gethostbyname(host_name))==0)
    {
        perror("cant resolving localhost\n");
        WSACleanup();
        exit(1);
    }

    // bzero(&pin,sizeof(pin));
    memset(&pin, 0, sizeof(pin));
    memset(buf, 0, sizeof(buf));
    pin.sin_family=AF_INET;
    pin.sin_addr.s_addr=htonl(INADDR_ANY);
    pin.sin_addr.s_addr=((struct in_addr *)(server_host_name->h_addr))->s_addr;
    pin.sin_port=htons(port);

    if((socket_descriptor=socket(AF_INET,SOCK_STREAM,0))==-1) {
        perror("Error opening socket\n");
        WSACleanup();
        exit(1);
    }

    if(connect(socket_descriptor,(const sockaddr *)&pin,sizeof(pin))==-1)
    {
        perror("cant connecting to server");
        closesocket(socket_descriptor);
        WSACleanup();
        exit(1);
    }

    printf("send message %s to server ...\n",message_str);
    if (send(socket_descriptor,message_str,strlen(message_str),0)==-1) {
        perror("cant send message\n");
        closesocket(socket_descriptor);
        WSACleanup();
        exit(1);
    }

    printf("waiting for response from server\n");
    if (recv(socket_descriptor,buf,sizeof(buf),0)==-1) {
        perror("cant receive response\n");
        closesocket(socket_descriptor);
        WSACleanup();
        exit(1);
    }

    printf("\nResponse from server:\n\n%s\n",buf);
   
    closesocket(socket_descriptor);
    WSACleanup();
   
    return ;
}

-----------------------------------------------------------

 

1.TCP服务端

server.cpp

-----------------------------------------------------------

#include <vld.h>
#include <stdio.h>
//#include <sys/socket.h>
#include <winsock2.h>
//#include <netinet/in.h>
//#include <arpa/inet.h>
//#include <netdb.h>
#include <stdlib.h>
#include <string.h>
int port = 10001;
int main ()
{
    struct sockaddr_in sin;
    struct sockaddr_in pin;
    int sock_descriptor;
    int temp_sock_descriptor;
    int size_of_addr;
    size_of_addr=sizeof(pin);
    char buf[8192];
    int i, lenth;
    int ret;

    WSADATA wsaData;

    if (WSAStartup(MAKEWORD(2,1),&wsaData)){
        printf("Winsock initializing fail\n");
        WSACleanup();
        return 0;
    }

    sock_descriptor = socket(AF_INET, SOCK_STREAM, 0);
    //bzero(&sin, sizeof (sin));
    memset(&sin, 0, sizeof(sin));
    memset(buf, 0, sizeof(buf));
    sin.sin_family = AF_INET;
    sin.sin_addr.s_addr = htonl(INADDR_ANY);
    sin.sin_port = htons (port);

    if (bind (sock_descriptor, (struct sockaddr *) &sin, sizeof (sin)) == -1) {
        perror ("bind!");
        closesocket(sock_descriptor);
        WSACleanup();
        exit (1);
    }

    if (listen (sock_descriptor, 20) == -1) {
        perror ("listen!");
        closesocket(sock_descriptor);
        WSACleanup();
        exit (1);
    }

    printf ("waiting for accepting connections from client\n");
    while(1) {
        temp_sock_descriptor = accept(sock_descriptor,(struct sockaddr *)&pin,&size_of_addr);
        if (temp_sock_descriptor == -1) {
            perror ("call to accept");
        }
       
        if (recv (temp_sock_descriptor, buf, sizeof (buf), 0) == -1) {
            perror ("recv!");
            closesocket(sock_descriptor);
            WSACleanup();
            exit (1);
        }

        printf ("received:%s\n", buf);
        lenth = strlen (buf);
        for (i = 0; i <= lenth; i++) {
            buf[i] = toupper (buf[i]);

        }

        if (send (temp_sock_descriptor, buf, lenth, 0) == -1) {
            perror ("send!");
            closesocket(sock_descriptor);
            WSACleanup();
            exit (1);
        }

    }
   
    closesocket (temp_sock_descriptor);
    WSACleanup();
   
    return 0;
}

-----------------------------------------------------------

程序运行截图:



  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值