面向连接的echo服务编程实例

以下是echo_serv.c的源码,提供创建服务端,绑定套接字到本机IP的8080端口,当收到客户端发送的字符串就在屏幕上打印出来,并且把字符串发送给客户端

 1 // echo_serv.c – gcc –o s echo_serv.c
 2 #include <sys/types.h>
 3 #include <sys/socket.h>
 4 #include <netinet/in.h>
 5 #include <arpa/inet.h>
 6 #include <unistd.h>
 7 #include <stdio.h>
 8 #include <errno.h>
 9
10 #define EHCO_PORT 8080
11 #define MAX_CLIENT_NUM 10
12
13 int main()
14 {
15   int sock_fd;
16   struct sockaddr_in serv_addr;
17   int clientfd;
18   struct sockaddr_in clientAdd;
19   char buff[101];
20   socklen_t len;
21   int closing =0;
22   int n;
23
24   /* 创建socket */
25   sock_fd = socket(AF_INET, SOCK_STREAM, 0);
26   if(sock_fd==-1) {
27     perror("create socket error!");
28     return 0;
29   } else {
30     printf("Success to create socket %d\n", sock_fd);
31   }
32
33   /* 设置server地址结构 */
34   bzero(&serv_addr, sizeof(serv_addr));                // 初始化结构占用的内存
35   serv_addr.sin_family = AF_INET;                    // 设置地址传输层类型
36   serv_addr.sin_port = htons(EHCO_PORT);            // 设置监听端口
37   //serv_addr.sin_addr.s_addr = htons(INADDR_ANY);    // 设置服务器地址  //用这个不能实现在不同主机上的通讯
38   serv_addr.sin_addr.s_addr = inet_addr("192.168.13.145");    // 设置服务器地址
39
40   bzero(&(serv_addr.sin_zero), 8);
41
42   /* 把地址和套接字绑定 */
43   if(bind(sock_fd, (struct sockaddr*)&serv_addr, sizeof(serv_addr))!= 0) {
44     printf("bind address fail! %d\n", errno);
45     close(sock_fd);
46     return 0;
47   } else {
48     printf("Success to bind address!\n");
49   }
50
51   /* 设置套接字监听 */
52   if(listen(sock_fd ,MAX_CLIENT_NUM) != 0) {
53     perror("listen socket error!\n");
54     close(sock_fd);
55     return 0;
56   } else {
57     printf("Success to listen\n");
58   }
59
60   /* 创建新连接对应的套接字 */
61   len = sizeof(clientAdd);
62   clientfd = accept(sock_fd, (struct sockaddr*)&clientAdd, &len);
63   if (clientfd<=0) {
64     perror("accept() error!\n");
65     close(sock_fd);
66     return 0;
67   }
68
69   /* 接收用户发来的数据 */
70   while((n = recv(clientfd,buff, 100,0 )) > 0) {
71     buff[n] = ‘\0‘; // 给字符串加入结束符
72     printf("number of receive bytes = %d data = %s\n", n, buff);        // 打印字符串长度和内容
73     fflush(stdout);
74     send(clientfd, buff, n, 0);            // 发送字符串内容给客户端
75     if(strncmp(buff, "quit", 4) == 0)        // 判断是否是退出命令
76       break;
77   }
78
79   close(clientfd);                        // 关闭新建的连接
80   close(sock_fd);                    // 关闭服务端监听的socket
81
82   return 0;
83 }

然后是客户端程序,在和服务器建立连接后发送字符串到服务器端,并且接受服务器发送的字符串显示在屏幕上

 1 // echo_client – gcc –o c echo_client.c
 2 #include <sys/types.h>
 3 #include <sys/socket.h>
 4 #include <netinet/in.h>
 5 #include <arpa/inet.h>
 6 #include <unistd.h>
 7 #include <stdio.h>
 8 #include <errno.h>
 9
10 #define EHCO_PORT 8080
11 #define MAX_COMMAND 5
12
13 int main()
14 {
15   int sock_fd;
16   struct sockaddr_in serv_addr;
17
18   char *buff[MAX_COMMAND] = {"abc", "def", "test", "hello", "quit"};
19   char tmp_buf[100];
20   socklen_t len;
21   int n, i;
22
23   /* 创建socket */
24   sock_fd = socket(AF_INET, SOCK_STREAM, 0);
25   if(sock_fd==-1) {
26     perror("create socket error!");
27     return 0;
28   } else {
29     printf("Success to create socket %d\n", sock_fd);
30   }
31
32   /* 设置server地址结构 */
33   bzero(&serv_addr, sizeof(serv_addr));                // 初始化结构占用的内存
34   serv_addr.sin_family = AF_INET;                    // 设置地址传输层类型
35   serv_addr.sin_port = htons(EHCO_PORT);            // 设置监听端口
36   //serv_addr.sin_addr.s_addr = htons(INADDR_ANY);        // 设置服务器地址  //用这个不能实现在不同主机上的通讯
37   serv_addr.sin_addr.s_addr = inet_addr("192.168.13.145");    // 设置服务器地址
38   bzero(&(serv_addr.sin_zero), 8);
39
40   /* 连接到服务端 */
41   if (-1==connect(sock_fd, (struct sockaddr*)&serv_addr, sizeof(serv_addr))) {
42     perror("connect() error!\n");
43     close(sock_fd);
44     return 0;
45   }
46   printf("Success connect to server!\n");
47
48   /* 发送并接收缓冲的数据 */
49   for (i=0;i<MAX_COMMAND;i++) {
50     send(sock_fd, buff[i], 100, 0);                        // 发送数据给服务端
51     n = recv(sock_fd, tmp_buf, 100, 0);                    // 从服务端接收数据
52     tmp_buf[n] = ‘\0‘;  // 给字符串添加结束标志
53     printf("data send: %s receive: %s\n", buff[i], tmp_buf);        // 打印字符串
54     if (0==strncmp(tmp_buf, "quit", 4))                    // 判断是否是退出命令
55       break;
56   }
57
58   close(sock_fd);                                    // 关闭套接字
59
60   return 0;
61 }

关于htos补充一下,也是摘自其它博主的吧

在C/C++写网络程序的时候,往往会遇到字节的网络顺序和主机顺序的问题。这是就可能用到htons(), ntohl(), ntohs(),htons()这4个函数。
网络字节顺序与本地字节顺序之间的转换函数:

htonl()--"Host to Network Long"
ntohl()--"Network to Host Long"
htons()--"Host to Network Short"
ntohs()--"Network to Host Short"

之所以需要这些函数是因为计算机数据表示存在两种字节顺序:NBO与HBO

网络字节顺序NBO(Network Byte Order): 按从高到低的顺序存储,在网络上使用统一的网络字节顺序,可以避免兼容性问题。

主机字节顺序(HBO,Host Byte Order): 不同的机器HBO不相同,与CPU设计有关,数据的顺序是由cpu决定的,而与操作系统无关。
如 Intel x86结构下, short型数0x1234表示为34 12, int型数0x12345678表示为78 56 34 12  
如 IBM power PC结构下, short型数0x1234表示为12 34, int型数0x12345678表示为12 34 56 78
   
由于这个原因不同体系结构的机器之间无法通信,所以要转换成一种约定的数序,也就是网络字节顺序,其实就是如同power pc那样的顺序. 在PC开发中有ntohl和htonl函数可以用来进行网络字节和主机字节的转换.

说白了其实不过只是大端方式小端方式的转换

时间: 2024-11-02 23:09:28

面向连接的echo服务编程实例的相关文章

Java学习之网络编程实例

转自:http://www.cnblogs.com/springcsc/archive/2009/12/03/1616413.html 多谢分享 网络编程 网络编程对于很多的初学者来说,都是很向往的一种编程技能,但是很多的初学者却因为很长一段时间无法进入网络编程的大门而放弃了对于该部分技术的学习. 在学习网络编程以前,很多初学者可能觉得网络编程是比较复杂的系统工程,需要了解很多和网络相关的基础知识,其实这些都不是很必需的.首先来问一个问题:你 会打手机吗?很多人可能说肯定会啊,不就是按按电话号码

WCF服务编程 读书笔记——第1章 WCF基础(1)

第1章 WCF基础 本章主要介绍WCF的基本概念.构建模块以及WCF体系架构,以指导读者构建一个简单的WCF服务.从本章的内容中,我们可以了解到WCF的基本术语,包括地址(Address).绑定(Binding).契约(Contract)和终结点(Endpoint):了解如何托管服务,如何编写客户端代码:了解WCF的相关主题,诸如进程内托管(In-Proc Hosting)以及可靠性的实现.即使你已经熟知WCF的基本概念,仍然建议你快速浏览本章的内容,它不仅能够巩固你的已有知识,而且本章介绍的一

Linux下shell编程实例

1. 判断一文件是不是块或字符设备文件,如果是将其拷贝到 /dev 目录下 read -p "input a file:" filename if [ -b $filename -o -c $filename ] then cp $filename /dev/ fi 2.编写一个脚本,进行简单的减法运算,要求提示输入变量 #!/bin/bash read -p "input a number:" num1 read -p "input another nu

WCF服务编程 读书笔记——第1章 WCF基础(2)

续:第1章 WCF基础(1) 元数据交换 服务有两种方案可以发布自己的元数据.一种是基于HTTP-GET协议提供元数据, 另一种则是后面将要讨论的使用专门的终结点的方式.WCF能够为服务自动提供基于HTTPGET的元数据,但需要显式地添加服务行为( Behavior)以支持这一功能.本书后面的章节会介绍行为的相关知识.现在,我们只需要知道行为属于服务的本地特性,例如是否需要基于HTTP-GET交换元数据, 就是一种服务行为.我们可以通过编程方式或管理方式添加行为.在例 1 - 10 演示的宿主应

WCF服务编程 读书笔记——第2章 服务契约

操作重载诸如 C++ 和 C# 等编程语言都支持方法重载,即允许具有相同名称的两个方法可以定义不同的参数.例如,如下的 C# 接口就是有效的定义: interface ICalculator { int Add(int arg1,int arg2); double Add(double arg1,double arg2); } 然而,基于 WSDL 的操作却不支持操作重载.因此,在编译如下的契约定义时,装载服务宿主就会抛出 InvalidOperationException异常: // 无效的契

LInux进程间通信之消息队列编程实例

本文主要通过消息队列的编程实例来加深对消息队列的理解. 一.消息队列之创建 创建一个消息队列,需要用到一个函数: #include<sys/types.h> #include<sys/ipc.h> #include<sys/msg.h> int msgget(key_t key,int msgflg); key:需要调用ftok函数来获取. msgflg:IPC_CREAT,不存在则创建,存在则返回已有的qid. IPC_CREAT|IPC_EXCL,不存在则创建,存在

Servlet学习(三)Servlet编程实例-网站登录(修改版-增加数据库-附源码)

我们为刚才的那个示例Servlet学习(二)Servlet编程实例-网站登录添加上数据库,组成一个较为完整的登录功能. 在开始之前我们得首先了解下一些常用的java术语: PO(persistant object) 持久对象 在o/r 映射的时候出现的概念,如果没有o/r映射,就没有这个概念存在了. 通常对应数据模型(数据库),本身还有部分业务逻辑的处理. 可以看成是与数据库中的表相映射的java对象. 最简单的PO就是对应数据库中某个表中的一条记录,多个记录可以用PO的集合. PO中应该不包含

PHP多进程编程实例

这篇文章主要介绍了PHP多进程编程实例,本文讲解的是在Linux下实现PHP多进程编程,需要的朋友可以参考下 羡慕火影忍者里鸣人的影分身么?没错,PHP程序是可以开动影分身的!想完成任务,又觉得一个进程太慢,那么,试试用多进程来搞吧.这篇文章将会介绍一下PHP多进程的基本需求,如何创建多进程以及基本的信号控制,暂时不会告诉你如何进行进程间通信和信息共享. 1. 准备 在动手之前,请确定你用的不是M$ Windows平台(因为我没有Windows).Linux / BSD / Unix应该都是没问

【Java】Web 服务编程技巧与窍门: 在 UDDI 注册中心为 Web 服务注册开发 UDDI Java 应用程序

本技巧建立了一个使用统一描述.发现和集成 (Universal Description, Discovery, and Integration,UDDI) 来注册应用程序级消费的 Web 服务实例.作者提供了详细的代码示例以及基于 Java 的统一描述.发现和集成 (Universal Description, Discovery,and Integration for Java,UDDI4J) API 的扩展 API,通过这些可以使您使用 UDDI 来进行您自己的开发. 0 评论: Andre