QT客户端
//widget.h
#ifndef WIDGET_H
#define WIDGET_H #include <QWidget>
#include <QTcpSocket>
#include <QPushButton>
#include <QLineEdit>
#include <QLabel>
#include <QCloseEvent>
#include <QTextBrowser> class Widget : public QWidget
{
Q_OBJECT public:
Widget(QWidget *parent = );
~Widget();
private:
QTcpSocket *tcpsocket;
QPushButton *btn1,*btn2;
QLineEdit *edit1,*edit2,*edit3;
QLabel *label1,*label2,*label3;
QTextBrowser *textb; void closeEvent(QCloseEvent *event);
private slots:
void btn1_click();
void btn2_click();
void myrecvdata();
}; #endif // WIDGET_H
//widget.cpp
#include "widget.h"
#include <QHBoxLayout>
#include <QVBoxLayout>
#include <QMessageBox>
#include <QHostAddress> Widget::Widget(QWidget *parent)
: QWidget(parent)
{
tcpsocket=new QTcpSocket(this);
btn1=new QPushButton(tr("连接"));
btn2=new QPushButton(tr("发送"));
connect(btn1,SIGNAL(clicked()),this,SLOT(btn1_click()));
connect(btn2,SIGNAL(clicked()),this,SLOT(btn2_click()));
edit1=new QLineEdit();
edit2=new QLineEdit();
edit3=new QLineEdit();
label1=new QLabel(tr("IP地址:"));
label2=new QLabel(tr("端口号:"));
label3=new QLabel(tr("消息内容:"));
textb=new QTextBrowser();
QHBoxLayout *lay1=new QHBoxLayout();
lay1->addWidget(label1);
lay1->addWidget(edit1);
lay1->addWidget(label2);
lay1->addWidget(edit2);
lay1->addWidget(btn1);
QHBoxLayout *lay2=new QHBoxLayout();
lay2->addWidget(label3);
lay2->addWidget(edit3);
lay2->addWidget(btn2);
QHBoxLayout *lay4=new QHBoxLayout();
lay4->addWidget(textb);
QVBoxLayout * lay3=new QVBoxLayout(this);
lay3->addLayout(lay1);
lay3->addLayout(lay2);
lay3->addLayout(lay4);
/*接收数据*/
connect(tcpsocket,SIGNAL(readyRead()),this,SLOT(myrecvdata()));
} //连接
void Widget::btn1_click()
{
/*禁用按钮,值是false*/
btn1->setEnabled(false);
//获取IP地址
QString ipaddr=edit1->text();
edit1->setEnabled(false);
//获取端口号
QString port=edit2->text();
edit2->setEnabled(false);
QHostAddress * serverip=new QHostAddress();
serverip->setAddress(ipaddr);
tcpsocket->connectToHost(*serverip,port.toInt());
//释放socket连接
delete serverip;
} //发送
void Widget::btn2_click()
{
QString strtext=edit3->text();
if(!strtext.isEmpty())
tcpsocket->write(strtext.toStdString().data());
/*清空输入框*/
edit3->clear();
/*设置输入框重新获得焦点*/
edit3->setFocus();
} //接收消息
void Widget::myrecvdata()
{
char buf[]={};
/*bytesAvailable()表示有效数据*/
while(tcpsocket->bytesAvailable()>)
{
memset(buf,,sizeof(buf));
tcpsocket->read(buf,sizeof(buf));
textb->append(buf);
//QMessageBox::information(this,"消息",buf);
}
} //关闭
void Widget::closeEvent(QCloseEvent *event)
{
if(QMessageBox::question(this,"提示","你确定要退出程序吗?",QMessageBox::Yes|QMessageBox::No,QMessageBox::No)==QMessageBox::Yes)
{
event->accept();
}else
{
event->ignore();
}
} Widget::~Widget()
{ }
Linux服务器端
//pub.h
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <errno.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <sys/select.h>
#include <sys/time.h> #define MAXPondNum 100 #ifdef __cplusplus
extern "C"
{
#endif //create socket
int create_socket(int port); //set noblock
int setnonblock(int st); //select
int open_select(int st); #ifdef __cplusplus
extern "C"
}
#endif
//pub.c
#include "pub.h" //create socket
int create_socket(int port)
{
if (port <= )
{
printf("create_socket() param not correct!\n");
return -;
}
int st = socket(AF_INET, SOCK_STREAM, );
if (st < )
{
printf("socket() failed ! error message:%s\n", strerror(errno));
return -;
}
//reuse address
int on = ;
if (setsockopt(st, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on)) < )
{
printf("socket() failed ! error message:%s\n", strerror(errno));
close(st);
return -;
}
//bind
struct sockaddr_in addr;
memset(&addr, , sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(port);
addr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(st, (struct sockaddr *) &addr, sizeof(addr)) < )
{
printf("bind() failed ! error message:%s\n", strerror(errno));
close(st);
return -;
}
//listen
if (listen(st, ) < )
{
printf("listen() failed ! error message:%s\n", strerror(errno));
close(st);
return -;
}
return st;
} //set noblock
int setnonblock(int st)
{
if (st < )
{
printf("setnonblock() param not correct!\n");
return -;
}
int opts = fcntl(st, F_GETFL);
if (opts < )
{
printf("fcntl() failed ! error message:%s\n", strerror(errno));
return -;
}
opts = opts | O_NONBLOCK;
if (fcntl(st, F_SETFL, opts))
{
printf("fcntl() failed ! error message:%s\n", strerror(errno));
return -;
}
return opts;
} //net address to
int sockaddrtoa(const struct sockaddr_in *addr, char *ipaddr)
{
if (addr == NULL || ipaddr == NULL)
{
printf("sockaddrtoa() param not correct!\n");
return -;
}
unsigned char *p = (unsigned char *) &(addr->sin_addr.s_addr);
sprintf(ipaddr, "%u:%u:%u:%u", p[], p[], p[], p[]);
return ;
} //accept clinet
int accept_client(int st)
{
if (st < )
{
printf("accept_client() param not correct!\n");
return -;
}
struct sockaddr_in addr;
memset(&addr, , sizeof(addr));
socklen_t len = sizeof(addr);
int client_st = accept(st, (struct sockaddr *) &addr, &len);
if (client_st < )
{
printf("accept() failed ! error message:%s\n", strerror(errno));
return -;
}
char ipstr[] = { };
sockaddrtoa(&addr, ipstr);
printf("accept by %s\n", ipstr);
return client_st;
} //recv message
int recv_message(int st)
{
if (st < )
{
printf("accept_client() param not correct!\n");
return -;
}
char buf[]={};
int rc=recv(st,buf,sizeof(buf),);
if(rc<)
{
printf("recv() failed ! error message:%s\n",strerror(errno));
return -;
}else if(rc==)
{
/*这里需要判断客户端是不是关闭连接了*/
printf("client is closed!\n");
return -;
}
printf("%s\n",buf);
//发送接收信息
memset(buf,,sizeof(buf));
strcpy(buf,"server have recved !\n");
if(send(st,buf,sizeof(buf),)<)
{
printf("send() failed ! error message:%s\n",strerror(errno));
return -;
}
return ;
} //select
int open_select(int st)
{
if (st < )
{
printf("open_select() param not correct!\n");
return -;
}
int i = ;
/*池子中只存放客户端socket*/
int socketpond[MAXPondNum] = { };
/*
* 注意int socketpond[MAXPondNum] = { -1 };
* 改写法并不能将所有元素初始化为-1,只有0这种唯一特例
* */
for(i=;i<MAXPondNum;i++)
{
socketpond[i]=-;
}
/*定义事件数组中socket最大的值*/
int maxfd = ;
/*准备监听事件数组结构*/
fd_set allsocket;
while ()
{
/*清空有消息的事件数组*/
FD_ZERO(&allsocket);
/*将服务器socket直接放入事件数组中*/
FD_SET(st, &allsocket);
/*假设服务器socket是事件数组中最大的*/
maxfd = st;
/*遍历现有socket池,找出最大的socket,将所有的socket添加到事件数组中*/
for (i = ; i < MAXPondNum; i++)
{
if (socketpond[i] != -)
{
FD_SET(socketpond[i], &allsocket);
if (socketpond[i] > maxfd)
{
maxfd = socketpond[i];
}
}
}
int events = select(maxfd + , &allsocket, NULL, NULL, NULL);
if (events < )
{
printf("select() failed ! error message:%s\n", strerror(errno));
break;
}
/*
* 猜想:当两个客户端同时连接到服务器时,select是如何处理服务器socket事件的呢
* 我认为一个服务器端socket只能处理一个客户端连接请求
* 事件数组中只有一个服务器socket,这个服务器socket只会处理一个客户端连接请求,
* 而此时另外一个客户端请求会在下次的select事件数组中被处理
* */
/*先处理服务器socket特殊处理*/
if (FD_ISSET(st, &allsocket))
{
printf("accept client!\n");
//accept
int client_st = accept_client(st);
if (client_st < )
{
break;
}
//set nonblock
if (setnonblock(client_st) < )
{
break;
}
for (i = ; i < MAXPondNum; i++)
{
if (socketpond[i] == -)
{
socketpond[i] = client_st;
break;
}
}
if (i == MAXPondNum)
{
printf("服务端连接池已满!\n");
close(client_st);
}
}
for (i = ; i < MAXPondNum; i++)
{
if (socketpond[i] < )
{
continue;
}
if (FD_ISSET(socketpond[i], &allsocket))
{
//接收客户端发送消息
if(recv_message(socketpond[i])<)
{
//接收客户端消息出错
//关闭客户端
close(socketpond[i]);
//从服务器端socket池子中将该socket清除
socketpond[i]=-;
}
/*需要处理的事件数减一*/
events--;
}
if(events<)
{
break;
}
}
}
return ;
}
//mserver.c
//服务器端
#include "pub.h" int main(int arg,char *args[])
{
if(arg<)
{
printf("please print one param!\n");
return -;
}
int port=atoi(args[]);
int listen_st=create_socket(port);
if(listen_st<)
return -;
if(setnonblock(listen_st)<)
goto END;
open_select(listen_st);
END:close(listen_st);
return ;
}
.SUFFIXES:.c .o
CC=gcc
SRCS=mserver.c\
pub.c
OBJS=$(SRCS:.c=.o)
EXEC=mser
start:$(OBJS)
$(CC) -o $(EXEC) $(OBJS)
@echo "-------OK--------"
.c.o:
$(CC) -g -Wall -o $@ -c $<
clean:
rm -f $(OBJS)
rm -f $(EXEC)

QT 网络编程三(TCP版)的更多相关文章

  1. c++学习书籍推荐《C++ GUI Qt 4编程(第2版)》下载

    下载地址:点我 百度云及其他网盘下载地址:点我 编辑推荐 <C++ GUI Qt 4编程(第2版)>讲授的大量Qt4编程原理和实践,都可以轻易将其应用于Qt4.4.Qt4.5及后续版本的Q ...

  2. C#网络编程之---TCP协议的同步通信(二)

    上一篇学习日记C#网络编程之--TCP协议(一)中以服务端接受客户端的请求连接结尾既然服务端已经与客户端建立了连接,那么沟通通道已经打通,载满数据的小火车就可以彼此传送和接收了.现在让我们来看看数据的 ...

  3. 嵌入式linux的网络编程(1)--TCP/IP协议概述

    嵌入式linux的网络编程(1)--TCP/IP协议概述 1.OSI参考模型及TCP/IP参考模型 通信协议用于协调不同网络设备之间的信息交换,它们建立了设备之间互相识别的信息机制.大家一定都听说过著 ...

  4. 【Linux网络编程】TCP网络编程中connect()、listen()和accept()三者之间的关系

    [Linux网络编程]TCP网络编程中connect().listen()和accept()三者之间的关系 基于 TCP 的网络编程开发分为服务器端和客户端两部分,常见的核心步骤和流程如下: conn ...

  5. 【网络编程1】网络编程基础-TCP、UDP编程

    网络基础知识 网络模型知识 OSI七层模型:(Open Systems Interconnection Reference Model)开放式通信系统互联参考模型,是国际标准化组织(ISO)提出的一个 ...

  6. python网络编程05 /TCP阻塞机制

    python网络编程05 /TCP阻塞机制 目录 python网络编程05 /TCP阻塞机制 1.什么是拥塞控制 2.拥塞控制要考虑的因素 3.拥塞控制的方法: 1.慢开始和拥塞避免 2.快重传和快恢 ...

  7. Linux网络编程(三)

    Linux网络编程(三) wait()还是waitpid() Linux网络编程(二)存在客户端断开连接后,服务器端存在大量僵尸进程.这是由于服务器子进程终止后,发送SIGCHLD信号给父进程,而父进 ...

  8. Qt网络编程QTcpServer和QTcpSocket的理解

    前一段时间通过调试Qt源码,大致了解了Qt的事件机制.信号槽机制.毕竟能力和时间有限.有些地方理解的并不是很清楚. 开发环境:Linux((fedora 17),Qt版本(qt-everywhere- ...

  9. 【转载】[基础知识]【网络编程】TCP/IP

    转自http://mc.dfrobot.com.cn/forum.php?mod=viewthread&tid=27043 [基础知识][网络编程]TCP/IP iooops  胖友们楼主我又 ...

随机推荐

  1. zh-Hans vs.net 通过 管理nuget程序包下载简体中文语言包 zh-cn

    zh-Hans  vs.net 通过 管理nuget程序包下载简体中文语言包 在搜索中输入:zh-hans

  2. (转) 一步一步学习ASP.NET 5 (三)- 认识新的Web结构

    转发:微软MVP 卢建晖 的文章,希望对大家有帮助.原文:http://blog.csdn.net/kinfey/article/details/44421979 编者语 : 今天微软的两大盛事,早上 ...

  3. 将dll程序集添加到缓存里

    1.点击开始---所有程序---...如下图 并以管理员身份运行. 2.输入命令行 gacutil.exe /i D:\Word\CRS_BPM_Sln\SourceCode\BPM\Referenc ...

  4. Linux:kill 进程

    在使用Linux时,出现端口占用.进程已启动(但处于不可控状态)情况时如何处理? 发现已知端口被占用时,可以使用netstat -apn | grep yourPort 来查看占用该端口的进程的pid ...

  5. PL/SQL远程备份和恢复Oracle数据库

    (转自:http://blog.csdn.net/huchunfu/article/details/25165901) 在客户端远程备份的文件保存在数据库所在主机上,不会直接拷贝到客户端.—————— ...

  6. WEB压力测试

    原文地址:WEB压力测试 作者:鸟哥のlinux webbench最多可以模拟3万个并发连接去测试网站的负载能力,个人感觉要比Apache自带的ab压力测试工具好,安装使用也特别方便. 1.适用系统: ...

  7. SQL Server:服务器角色

    角色 描述 sysadmin 执行SQL Server中的任何动作 serveradmin 配置服务器设置 setupadmin 安装复制和管理扩展过程 securityadmin 管理登录和CREA ...

  8. Windows搭建python开发环境,python入门到精通[一]

    从大学开始玩python到现在参加工作,已经有5年了,现在的公司是一家.net的公司用到python的比较少,最近公司有新项目需要用到python,领导希望我来跟其他同事training,就有了这篇博 ...

  9. android 判断字符串是否为空与比对["=="与equals()的区别]

    if (s == null || s.equals("")) ; } s.equals("")里面是要比对的字符串 声明字符串未赋初始值或值,然后比对就会出错, ...

  10. linux进程间通信-管道

    一 管道的局限性 管道有两个局限性:(1)他是半双工(即数据只能在一个方向上流动).(2)它只能在具有公共祖先的进程之间使用.一个管道由一个进程创建,然后该 进程调用fork,此后父子进程之间就可该管 ...