面向连接的echo服务编程实例
以下是echo_serv.c的源码,提供创建服务端,绑定套接字到本机IP的8080端口,当收到客户端发送的字符串就在屏幕上打印出来,并且把字符串发送给客户端
// echo_serv.c – gcc –o s echo_serv.c
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h> #define EHCO_PORT 8080
#define MAX_CLIENT_NUM 10 int main()
{
int sock_fd;
struct sockaddr_in serv_addr;
int clientfd;
struct sockaddr_in clientAdd;
char buff[];
socklen_t len;
int closing =;
int n; /* 创建socket */
sock_fd = socket(AF_INET, SOCK_STREAM, );
if(sock_fd==-) {
perror("create socket error!");
return ;
} else {
printf("Success to create socket %d\n", sock_fd);
} /* 设置server地址结构 */
bzero(&serv_addr, sizeof(serv_addr)); // 初始化结构占用的内存
serv_addr.sin_family = AF_INET; // 设置地址传输层类型
serv_addr.sin_port = htons(EHCO_PORT); // 设置监听端口
//serv_addr.sin_addr.s_addr = htons(INADDR_ANY); // 设置服务器地址 //用这个不能实现在不同主机上的通讯
serv_addr.sin_addr.s_addr = inet_addr("192.168.13.145"); // 设置服务器地址 bzero(&(serv_addr.sin_zero), ); /* 把地址和套接字绑定 */
if(bind(sock_fd, (struct sockaddr*)&serv_addr, sizeof(serv_addr))!= ) {
printf("bind address fail! %d\n", errno);
close(sock_fd);
return ;
} else {
printf("Success to bind address!\n");
} /* 设置套接字监听 */
if(listen(sock_fd ,MAX_CLIENT_NUM) != ) {
perror("listen socket error!\n");
close(sock_fd);
return ;
} else {
printf("Success to listen\n");
} /* 创建新连接对应的套接字 */
len = sizeof(clientAdd);
clientfd = accept(sock_fd, (struct sockaddr*)&clientAdd, &len);
if (clientfd<=) {
perror("accept() error!\n");
close(sock_fd);
return ;
} /* 接收用户发来的数据 */
while((n = recv(clientfd,buff, , )) > ) {
buff[n] = '\0'; // 给字符串加入结束符
printf("number of receive bytes = %d data = %s\n", n, buff); // 打印字符串长度和内容
fflush(stdout);
send(clientfd, buff, n, ); // 发送字符串内容给客户端
if(strncmp(buff, "quit", ) == ) // 判断是否是退出命令
break;
} close(clientfd); // 关闭新建的连接
close(sock_fd); // 关闭服务端监听的socket return ;
}
然后是客户端程序,在和服务器建立连接后发送字符串到服务器端,并且接受服务器发送的字符串显示在屏幕上
// echo_client – gcc –o c echo_client.c
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h> #define EHCO_PORT 8080
#define MAX_COMMAND 5 int main()
{
int sock_fd;
struct sockaddr_in serv_addr; char *buff[MAX_COMMAND] = {"abc", "def", "test", "hello", "quit"};
char tmp_buf[];
socklen_t len;
int n, i; /* 创建socket */
sock_fd = socket(AF_INET, SOCK_STREAM, );
if(sock_fd==-) {
perror("create socket error!");
return ;
} else {
printf("Success to create socket %d\n", sock_fd);
} /* 设置server地址结构 */
bzero(&serv_addr, sizeof(serv_addr)); // 初始化结构占用的内存
serv_addr.sin_family = AF_INET; // 设置地址传输层类型
serv_addr.sin_port = htons(EHCO_PORT); // 设置监听端口
//serv_addr.sin_addr.s_addr = htons(INADDR_ANY); // 设置服务器地址 //用这个不能实现在不同主机上的通讯
serv_addr.sin_addr.s_addr = inet_addr("192.168.13.145"); // 设置服务器地址
bzero(&(serv_addr.sin_zero), ); /* 连接到服务端 */
if (-==connect(sock_fd, (struct sockaddr*)&serv_addr, sizeof(serv_addr))) {
perror("connect() error!\n");
close(sock_fd);
return ;
}
printf("Success connect to server!\n"); /* 发送并接收缓冲的数据 */
for (i=;i<MAX_COMMAND;i++) {
send(sock_fd, buff[i], , ); // 发送数据给服务端
n = recv(sock_fd, tmp_buf, , ); // 从服务端接收数据
tmp_buf[n] = '\0'; // 给字符串添加结束标志
printf("data send: %s receive: %s\n", buff[i], tmp_buf); // 打印字符串
if (==strncmp(tmp_buf, "quit", )) // 判断是否是退出命令
break;
} close(sock_fd); // 关闭套接字 return ;
}
关于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函数可以用来进行网络字节和主机字节的转换.
说白了其实不过只是大端方式小端方式的转换
面向连接的echo服务编程实例的更多相关文章
- Python网络编程--Echo服务
Python网络编程--Echo服务 学习网络编程必须要练习的三个小项目就是Echo服务,Chat服务和Proxy服务.在接下来的几篇文章会详细介绍. 今天就来介绍Echo服务,Echo服务是最基本的 ...
- PHP多进程编程实例
这篇文章主要介绍了PHP多进程编程实例,本文讲解的是在Linux下实现PHP多进程编程,需要的朋友可以参考下 羡慕火影忍者里鸣人的影分身么?没错,PHP程序是可以开动影分身的!想完成任务,又觉得一个进 ...
- Web 服务编程,REST 与 SOAP(转)
原文地址:Web 服务编程,REST 与 SOAP REST 简介 在开始我们的正式讨论之前,让我们简单看一下 REST 的定义. REST(Representational State Transf ...
- JAX-RS 2.0 REST客户端编程实例
JAX-RS 2.0 REST客户端编程实例 2014/01/28 | 分类: 基础技术, 教程 | 0 条评论 | 标签: JAX-RS, RESTFUL 分享到:3 本文由 ImportNew - ...
- Java学习之网络编程实例
转自:http://www.cnblogs.com/springcsc/archive/2009/12/03/1616413.html 多谢分享 网络编程 网络编程对于很多的初学者来说,都是很向往的一 ...
- 内核编程实例,多文件的Makefile
内核编程实例,多文件的Makefile 经典的hello word测试 ////# cat hello.c #include <linux/module.h> #include <l ...
- NET Socket服务编程
smark https://github.com/IKende/ .NET Socket服务编程之-高效连接接入编 在.NET上编写网络服务深入都有2,3年了,而这些时间时如何在.NET里实现网络服务 ...
- NIO Socket编程实例
1.阻塞模式实例 NIOUtil类,用来通过SOcket获取BufferedReader和PrintWriter. package IO; import java.io.BufferedReader; ...
- linux下socket编程实例
linux下socket编程实例一.基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的.网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.s ...
随机推荐
- Spring 相关目录
Spring 相关目录 学习笔记 Spring 学习笔记 IoC 基础 Spring 学习笔记 Resource 资源 Spring 学习笔记 数据绑定,校验,BeanWrapper 与属性编辑器 源 ...
- Windows10安装好Visual Studio2017后,找不到MFC向导
前段时候在Windows10中安装好Visual Studio2017后,想创建一个基于MFC的对话框应用,发现无法找到MFC开发向导选项,很是奇怪,以前使用VC6.0或者Visual Studio2 ...
- 对 HTTP HTTPS的认识
1.HTTP:超文本传输协议 -以明文的形式传输 -效率更高,但是不安全 2.HTTPS:HTTP+SSL -传输之前数据先加密,之后在揭秘 -效率低,但是安全 3.get请求和post请求的区别 - ...
- 轻松理解https,So easy!
Java技术栈 www.javastack.cn 优秀的Java技术公众号 作者:翟志军 https://showme.codes/2017-02-20/understand-https/ 本文尝试一 ...
- 数据持久化之嵌入式数据库 SQLite(三)
阿里P7Android高级架构进阶视频免费学习请点击:https://space.bilibili.com/474380680 SQLite 是 D. Richard Hipp 用 C 语言编写的开源 ...
- Excel_PoweQuery——条件计数、条件求和
岁月不居,时节如流. 时光荏苒,岁月如梭. 前面两段充分体现了博主深厚的文学素养,别和博主争,博主说啥就是啥. 其实,对于大量数据的处理,这几年微软Office做的不单单是2007的时候把Excel的 ...
- 浅析php-fpm和fastcgi的关系
先讲讲CGI吧 浏览器向web server发起请求的时候,要有url,header,params等等吧,为什么有这些数据呢,这就是CGI的事了,CGI就规定了,传哪些数据,用什么样的格式传输 web ...
- 视频专家之路【二】:ffmpeg工具的使用
本文是听了雷宵骅大神的课之后的总结,部分内容借用了其PPT的内容,如有侵权请告知删除. 雷宵骅大神的博客为:https://blog.csdn.net/leixiaohua1020 要学习汽修,那么首 ...
- 71 Serializable(序列化和反序列化)
对象的输出流:ObjectOutputStream 把对象输出到文件存储起来,我们称作为序列化对象的输入流:ObjectInputStream 把对象从文件中读取出来,我们称作为反序列化 Obj ...
- macOS免费的NTFS读写软件
Mounty for Mac brew cask install mounty