【linux高级程序设计】(第十三章)Linux Socket网络编程基础 3
使用之前的函数实现的简单聊天程序
TCP协议
双方实时发送/接收消息
实现后的问题:
可能是我虚拟机的IP地址配得有问题吧。在一台电脑上面开两个终端,用127.0.0.1的IP收发可以互通。但是两个虚拟机就不行了,用192.168的IP段,能够ping通但是代码接收不到消息。
还有,两个进程都是接收到消息后,需要我自己按一下回车才能发送消息。
服务器端代码:
#include<stdio.h>
#include<string.h>
#include<errno.h>
#include<sys/socket.h>
#include<resolv.h>
#include<stdlib.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#define MAXBUF 1024
int main(int argc, char *argv[])
{
int pid;
int sockfd, new_fd;
socklen_t len;
struct sockaddr_in my_addr, their_addr;
unsigned int myport, lisnum;
char buf[MAXBUF + ];
if(argv[])
myport = atoi(argv[]); //命令行字符串转为整数,端口
else
myport = ; //默认端口 if(argv[])
lisnum = atoi(argv[]); //监听队列的大小
else
lisnum = ; //创建socket对象, IPv4, TCP, 默认协议
if((sockfd = socket(AF_INET, SOCK_STREAM, )) == -) //创建socket对象
{
perror("socket");
exit(EXIT_FAILURE);
} bzero(&my_addr, sizeof(my_addr));
my_addr.sin_family = AF_INET; //地址协议
my_addr.sin_port = htons(myport); //地址端口
if(argv[])
my_addr.sin_addr.s_addr = inet_addr(argv[]); //指定IP地址 从点分十进制字符串转为32位二进制
else
my_addr.sin_addr.s_addr = INADDR_ANY; //否则设置为本机任意地址 char mybuf[];
inet_ntop(AF_INET, &my_addr.sin_addr.s_addr, mybuf, );
printf("the ip is '%s'\n", mybuf);
//绑定地址信息
if(bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == -)
{
perror("bind");
exit(EXIT_FAILURE);
} //监听网络
if(listen(sockfd, lisnum) == -)
{
perror("listen");
exit(EXIT_FAILURE);
}
printf("wait for connect\n");
len = sizeof(struct sockaddr); //阻塞等待连接
if((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &len)) == -)
{
perror("accept");
exit(EXIT_FAILURE);
}
else //打印接收到的信息
{
printf("server: got connection from %s, port %d, socket %d\n", inet_ntoa(their_addr.sin_addr), ntohs(their_addr.sin_port), new_fd);
} //创建新进程
if(- == (pid = fork()))
{
perror("fork");
exit(EXIT_FAILURE);
}
else if( == pid) //子进程用于发送消息
{
while()
{
bzero(buf, MAXBUF + );
printf("input the message to send:");
fgets(buf, MAXBUF, stdin);
if(!strncasecmp(buf, "quit", ))
{
printf("i will close the connect!\n");
break;
}
len = send(new_fd, buf, strlen(buf) - , );
if(len < )
{
printf("message '%s' send failure! errno code is %d, errno message is '%s'\n", buf, errno, strerror(errno));
break;
}
}
}
else //父进程用于接收消息
{
while()
{
bzero(buf, MAXBUF + );
len = recv(new_fd, buf, MAXBUF, );
if(len > )
{
printf("message recv successful : '%s', %dByte recv\n", buf, len);
}
else if(len < )
{
printf("recv failure! errno code is %d, errno message is '%s'\n", errno, strerror(errno));
break;
}
else
{
printf("the other one close quit\n");
break;
}
}
} close(new_fd);
close(sockfd);
return ;
}
客户端代码:
#include<stdio.h>
#include<string.h>
#include<errno.h>
#include<sys/socket.h>
#include<resolv.h>
#include<stdlib.h>
#include<netinet/in.h>
#include<arpa/inet.h>
#include<unistd.h>
#define MAXBUF 1024
int main(int argc, char **argv)
{
int sockfd, len;
struct sockaddr_in dest;
char buffer[MAXBUF + ];
if(argc != )
{
printf(" error format, it must be :\n \t\t%s IP port\n", argv[]);
exit(EXIT_FAILURE);
} //创建socket对象
if((sockfd = socket(AF_INET, SOCK_STREAM, )) < )
{
perror("Socket");
exit(errno);
}
printf("socket created\n");
bzero(&dest, sizeof(dest));
dest.sin_family = AF_INET; //地址协议
dest.sin_port = htons(atoi(argv[])); //对方端口
//对方IP地址
if(inet_aton(argv[], (struct in_addr *)&dest.sin_addr.s_addr) == )
{
perror("argv[1]");
exit(errno);
}
//发起连接
if(connect(sockfd, (struct sockaddr *)&dest, sizeof(dest)) == -)
{
perror("Connect");
exit(errno);
}
printf("server connected\n");
pid_t pid; //创建子进程
if(- == (pid = fork()))
{
perror("fork");
exit(errno);
}
else if(pid == ) //子进程用于数据接收
{
while()
{
bzero(buffer, MAXBUF + );
len = recv(sockfd, buffer, MAXBUF, );
if(len > )
{
printf("recv successful:'%s', %d byte recv\n", buffer, len);
}
else if(len < )
{
perror("recv");
break;
}
else
{
printf("the other one close, quit\n");
break;
}
}
}
else //父进程用于数据发送
{
while()
{
bzero(buffer, MAXBUF + );
printf("input the message to send:");
fgets(buffer, MAXBUF, stdin);
if(!strncasecmp(buffer, "quit", ))
{
printf("i will close the connect!\n");
break;
}
len = send(sockfd, buffer, strlen(buffer) - , );
if(len < )
{
printf("message '%s' send failure! errno code is %d, errno message is '%s'\n", buffer, errno, strerror(errno));
break;
}
}
}
close(sockfd);
return ;
}
【linux高级程序设计】(第十三章)Linux Socket网络编程基础 3的更多相关文章
- 【linux高级程序设计】(第十三章)Linux Socket网络编程基础 2
BSD Socket网络编程API 创建socket对象 int socket (int __domain, int __type, int __protocol) :成功返回socket文件描述符, ...
- Socket网络编程-基础篇
Socket网络编程 网络通讯三要素: IP地址[主机名] 网络中设备的标识 本地回环地址:127.0.0.1 主机名:localhost 端口号 用于标识进程的逻辑地址 有效端口:0~65535 其 ...
- python全栈开发从入门到放弃之socket网络编程基础
网络编程基础 一 客户端/服务器架构 1.硬件C/S架构(打印机) 2.软件C/S架构 互联网中处处是C/S架构 如黄色网站是服务端,你的浏览器是客户端(B/S架构也是C/S架构的一种) 腾讯作为服务 ...
- 第九章:Python の 网络编程基础(一)
本課主題 何为TCP/IP协议 初认识什么是网络编程 网络编程中的 "粘包" 自定义 MySocket 类 本周作业 何为TCP/IP 协议 TCP/IP协议是主机接入互网以及接入 ...
- 【linux高级程序设计】(第十三章)Linux Socket网络编程基础 4
网络调试工具 tcpdump 功能:打印指定网络接口中与布尔表达式匹配的报头信息 关键字: ①类型:host(默认).net.port host 210.27.48.2 //指明是一台主机 net 2 ...
- 【linux高级程序设计】(第十三章)Linux Socket网络编程基础
IP地址定义: struct in_addr{ __u32 s_addr; }; in_addr_t inet_addr (__const char * __cp) :把点分十进制IP地址字符串转换 ...
- 第十三章:Python の 网络编程进阶(二)
本課主題 SQLAlchemy - Core SQLAlchemy - ORM Paramiko 介紹和操作 上下文操作应用 初探堡垒机 SQLAlchemy - Core 连接 URL 通过 cre ...
- 第十三篇:socket网络编程
本篇主要介绍网络编程的基础,以及UDP/TCP网络的socket编程,关于UDP套接字聊天器的实现.以及基于TCP套接字的服务器/客户端的实现上传下载功能. 一.网络通信 关于网络通信即通过网络(介质 ...
- socket网络编程基础小记
"一切皆Socket!" 话虽些许夸张.可是事实也是,如今的网络编程差点儿都是用的socket. --有感于实际编程和开源项目研究. 我们深谙信息交流的价值,那网络中进程之间怎样通 ...
- java架构《Socket网络编程基础篇》
本章主要介绍Socket的基本概念,传统的同步阻塞式I/O编程,伪异步IO实现,学习NIO的同步非阻塞编程和NIO2.0(AIO)异步非阻塞编程. 目前为止,Java共支持3种网络编程模型:BIO.N ...
随机推荐
- 关于我的Android 博客
我是曹新雨,我为自己代言.现在的菜鸟,3年以后我就是大神.为自己加油.微信:aycaoxinyu 关于我的Android博客,都是我当初遇到困难,克服之后,写上去的.后来,有人加我微信,问我一些问题, ...
- 调用startActivityForResult后直接调用onActivityResult
人员都知道,可以经由过程应用 startActivityForResult() 和 onActivityResult() 办法来传递或接管参数. 然而在"轻听"项目中,还没比及被调 ...
- HDFS写数据和读数据流程
HDFS数据存储 HDFS client上传数据到HDFS时,首先,在本地缓存数据,当数据达到一个block大小时.请求NameNode分配一个block. NameNode会把block所在的Dat ...
- Android学习记录(1)—Android中XML文件的序列化生成与解析
xml文件是非常常用的,在android中json和xml是非常常用的两种封装数据的形式,从服务器中获取数据也经常是这两种形式的,所以学会生成和解析xml和json是非常有用的,json相对来说是比较 ...
- 剑指Offer - 九度1513 - 二进制中1的个数
剑指Offer - 九度1513 - 二进制中1的个数2013-11-29 23:35 题目描述: 输入一个整数,输出该数二进制表示中1的个数.其中负数用补码表示. 输入: 输入可能包含多个测试样例. ...
- appium+python的APP自动化(1)
写这个东西也是自己喜欢研究些自动化的东西,以下全是自己的经验所得,由于开源的软件对于各版本以及操作系统要求很高,会经常碰到一些不兼容的问题,这个都属于正常的,换版本就对了. 本人的环境搭建都是在win ...
- Opencv3.4.5安装包
这个资源是Opencv3.4.5安装包,包括Windows软件包,Android软件包,IOS软件包,还有opencv的源代码:需要的下载吧. 点击下载
- Linux 文本对比 diff 命令详解(整理)
diff 命令详解 1.概述 windows系统下面就有不错的文本对比工具可以使用,例如常用的Beyond Compare,WinMerge都是图形界面的比较工具而且使用非常方便,如果你仅仅是在win ...
- 积累: .net里有个线程安全的int+1类
Interlocked.Increment(ref id);
- J2EE的十三个技术——EJB之消息驱动JMS
JMS--Java Message Service JAVA的消息服务,消息可实现两端通信. 用于访问面向消息中间件的标准api,他提供与厂商无关的访问方法,以访问消息收发服务. 特点:即使其中一方不 ...