socket通信
socket
socket 被译为“套接字”,它是计算机之间进行通信的一种约定或一种方式。
通过 socket 约定,不同计算机之间可以进行通信
在linux里一切皆文件,socket是一个表示网络连接的文件,对应网络连接的文件描述符。
在windows里,socket是一个网络连接,有别于文件。
流格式套接字
流格式套接字也叫“面向连接的套接字”(Stream Sockets),在代码中使用 SOCK_STREAM 表示。
SOCK_STREAM 是一种可靠的、双向的通信数据流,
数据可以准确无误地到达另一台计算机,如果损坏或丢失,可以重新发送。
流格式套接字使用了 TCP 协议,TCP 协议会控制你的数据按照顺序到达并且没有错误。
TCP
TCP全称The Transmission Control Protocol,传输控制协议
各个标志位及其意义:SYN(建立连接标志位)、ACK(确认标志位)、FIN(释放连接标志位)
TCP建立客户端与服务端的连接需要经过三次握手
第一次:Client将SYN置1,并产生随机数seq=x,发送数据包,Server接受判断SYN为1
第二次:Server将SYN和ACK置1,ack=x+1,seq=y,发送数据包,Client判断ack为x+1,ACK为1
第三次:Client将ACK置1,ack=y+1,发送数据包,Server接受判断ack为y+1,ACK为1
上述过程完成即建立连接
TCP断开客户端与服务端的连接需要经过四次挥手
第一次挥手:Client将FIN置1,发送数据包
第二次挥手:Server接受判断FIN为1,将ACK置1,发送数据包
第三次挥手:Server做好了释放服连接准备,将FIN置1,发送数据包
第四次挥手:Client接受判断FIN为1,将ACK置1,发送数据包
上述过程完成即断开连接
代码实现
代码实现分为客户端(client)和服务端(server)的实现
C++实现
C语言中文网的例程非常清晰,可以直接参考使用
Windows参考:http://c.biancheng.net/view/2129.html
Linux参考:http://c.biancheng.net/view/2128.html
注意事项:
在windows下使用g++编译需要加上-lws2_32选项,例如编译server.cpp如下
g++ server.cpp -o server.exe -lws2_32
如果用该代码与调试助手通信,需要注意端口和地址格式
例如我拿sockit进行调试,地址格式为AF_INET,端口前需要加上htons
Qt实现
Qt支持跨平台,我在windows上实现的效果,可以运用到linux。
最终实现了client界面和server界面编程,并经测试两端可以互通。
client界面
#include "clientwindow.h"
#include "serverwindow.h"
ClientWindow::ClientWindow(QWidget *parent)
    : QMainWindow(parent)
{
    // 由于linux qt版本较低
    // connect 采用旧版写法
    // 即 connect(Button,&QPushButton::clicked,this,&ButtonClicked);
    // 写成 connect(Button,SIGNAL(clicked()),this,SLOT(ButtonClicked()));
    //模式选择
    ChooseClientButton = new QPushButton(this);
    ChooseClientButton->setText("客户端");
    ChooseClientButton->move(0,0);
    ChooseServerButton = new QPushButton(this);
    ChooseServerButton->setText("服务端");
    ChooseServerButton->move(100,0);
    connect(ChooseServerButton,SIGNAL(clicked()),this,SLOT(ChooseServerButtonClicked())); //本身是客户端 所以只绑定服务端按钮
    //Ip输入框
    IpLabel = new QLabel(this);
    IpLabel->setText("Ip:");
    IpLabel->move(70,50);
    IpInput = new QLineEdit(this);
    IpInput->setFixedSize(200,35);
    IpInput->move(100,50);
    IpInput->setPlaceholderText("Ip地址...");
    IpInput->setClearButtonEnabled(true);
    //端口输入端
    PortLabel = new QLabel(this);
    PortLabel->setText("Port:");
    PortLabel->move(50,100);
    PortInput = new QLineEdit(this);
    PortInput->setFixedSize(200,35);
    PortInput->move(100,100);
    PortInput->setPlaceholderText("端口...");
    PortInput->setClearButtonEnabled(true);
    //连接按钮
    ConnectButton = new QPushButton(this);
    ConnectButton->setText("连接");
    ConnectButton->move(50,150);
    connect(ConnectButton,SIGNAL(clicked()),this,SLOT(ConnectButtonClicked()));
    connected = false;
    //断开按钮
    DisconnectButton = new QPushButton(this);
    DisconnectButton->setText("断开");
    DisconnectButton->move(190,150);
    connect(DisconnectButton,SIGNAL(clicked()),this,SLOT(DisconnectButtonClicked()));
    //文本输入框
    TextInput = new QTextEdit(this);
    TextInput->move(50,230);
    TextInput->setFixedSize(250,300);
    TextInputLabel = new QLabel(this);
    TextInputLabel->move(50,200);
    TextInputLabel->setText("发送内容");
    //发送按钮
    SendButton = new QPushButton(this);
    SendButton->move(50,550);
    SendButton->setText("发送");
    SendButton->setFixedSize(250,35);
    connect(SendButton,SIGNAL(clicked()),this,SLOT(SendButtonClicked()));
    //发送记录
    RecordLabel = new QLabel(this);
    RecordLabel->move(550,50);
    RecordLabel->setText("发送记录");
    RecordText = new QTextEdit(this);
    RecordText->move(350,80);
    RecordText->setFixedSize(500,450);
    RecordText->setReadOnly(true);
    //记录所有发送文本
    AllRecordText = QString();
    //client
    socket = new QTcpSocket();
    //清除按钮
    ClearRecordTextButton = new QPushButton(this);
    ClearRecordTextButton->move(350,550);
    ClearRecordTextButton->setText("清空");
    ClearRecordTextButton->setFixedSize(500,35);
    connect(ClearRecordTextButton,SIGNAL(clicked()),this,SLOT(ClearRecordTextButtonClicked()));
}
ClientWindow::~ClientWindow()
{
}
void ClientWindow::ConnectButtonClicked()
{
    //获取ip和端口
    Ip = IpInput->text();
    Port = PortInput->text().toInt();
    //连接
    socket->connectToHost(Ip, Port);
    if(socket->waitForConnected(1000))
    {
        connected = true;
        AllRecordText += "已连接\n";
        IpInput->setReadOnly(true);
        PortInput->setReadOnly(true);
        connect(socket,SIGNAL(readyRead()),this,SLOT(ReadServerData()));// 兼容
    }
    else
    {
        AllRecordText += "连接失败,请重新连接\n";
    }
    RecordText->setText(AllRecordText);
}
void ClientWindow::DisconnectButtonClicked()
{
    //断开连接
    connected = false;
    socket->disconnectFromHost();
    IpInput->setReadOnly(false);
    PortInput->setReadOnly(false);
}
void ClientWindow::SendButtonClicked()
{
    if(connected)
    {
        //多行文本 要按换行符分割 可以用QString::SkipEmptyParts跳过空行
        QStringList TextInputList = \
                TextInput->toPlainText().split(QRegExp("[\r\n]"),QString::SkipEmptyParts);
        for(int i =0;i<TextInputList.size();i++)
        {
            CurrentTime = QDateTime::currentDateTime();
            QString current_date =CurrentTime.toString("hh:mm:ss");
            TextInputList[i] +='\n';
            AllRecordText += "[" + current_date + "] " + \
                             Ip + ":" + QString::number(Port) + ": " + \
                    "-->: " + TextInputList[i];
            socket->write(TextInputList[i].toUtf8());
        }
        RecordText->setText(AllRecordText);
    }
    else
    {
        AllRecordText += "未连接,请连接\n";
        RecordText->setText(AllRecordText);
    }
}
void ClientWindow::ClearRecordTextButtonClicked()
{
    //清空记录
    AllRecordText = QString();
    RecordText->setText(AllRecordText);
}
void ClientWindow::ChooseServerButtonClicked()
{
    ServerWindow *server_window = new ServerWindow;
    server_window->resize(900,600);
    server_window->setFont(QFont("宋体",8));
    server_window->setWindowTitle("Socket TCP 通信");
    this->deleteLater();
    this->close();
    server_window->move(this->pos().x(),this->pos().y());
    server_window->show();
}
void ClientWindow::ReadServerData()
{
    QString ReadData = socket->readAll();
    //多行文本 要按换行符分割 可以用QString::SkipEmptyParts跳过空行
    QStringList TextInputList = \
            ReadData.split(QRegExp("[\r\n]"),QString::SkipEmptyParts);
    for(int i =0;i<TextInputList.size();i++)
    {
        CurrentTime = QDateTime::currentDateTime();
        QString current_date =CurrentTime.toString("hh:mm:ss");
        TextInputList[i] +='\n';
        AllRecordText += "[" + current_date + "] " + \
                         Ip + ":" + QString::number(Port) + ": " + \
                        "<--: " + TextInputList[i];
    }
    RecordText->setText(AllRecordText);
}
效果如下:
 
server界面
server和client实现类似
#include "serverwindow.h"
#include "clientwindow.h"
ServerWindow::ServerWindow(QWidget *parent)
    : QMainWindow(parent)
{
    //由于linux qt版本较低
    //connect 采用旧版写法
    //即 connect(Button,&QPushButton::clicked,this,&ButtonClicked);
    //写成 connect(Button,SIGNAL(clicked()),this,SLOT(ButtonClicked()));
    //模式选择
    ChooseClientButton = new QPushButton(this);
    ChooseClientButton->setText("客户端");
    ChooseClientButton->move(0,0);
    ChooseServerButton = new QPushButton(this);
    ChooseServerButton->setText("服务端");
    ChooseServerButton->move(100,0);
    connect(ChooseClientButton,SIGNAL(clicked()),this,SLOT(ChooseClientButtonClicked())); //本身是服务端 所以只绑定客户端按钮
    //Ip输入框
    IpLabel = new QLabel(this);
    IpLabel->setText("Ip:");
    IpLabel->move(70,50);
    IpInput = new QLineEdit(this);
    IpInput->setFixedSize(200,35);
    IpInput->move(100,50);
    IpInput->setPlaceholderText("Ip地址...");
    IpInput->setClearButtonEnabled(true);
    //端口输入端
    PortLabel = new QLabel(this);
    PortLabel->setText("Port:");
    PortLabel->move(50,100);
    PortInput = new QLineEdit(this);
    PortInput->setFixedSize(200,35);
    PortInput->move(100,100);
    PortInput->setPlaceholderText("端口...");
    PortInput->setClearButtonEnabled(true);
    //侦听按钮
    ListenButton = new QPushButton(this);
    ListenButton->setText("侦听");
    ListenButton->move(50,150);
    connect(ListenButton,SIGNAL(clicked()),this,SLOT(ListenButtonClicked()));
    listened = false;
    connected = false;
    //取消按钮
    DisListenButton = new QPushButton(this);
    DisListenButton->setText("取消侦听");
    DisListenButton->move(190,150);
    connect(DisListenButton,SIGNAL(clicked()),this,SLOT(DisListenButtonClicked()));
    //文本输入框
    TextInput = new QTextEdit(this);
    TextInput->move(50,230);
    TextInput->setFixedSize(250,300);
    TextInputLabel = new QLabel(this);
    TextInputLabel->move(50,200);
    TextInputLabel->setText("发送内容");
    //发送按钮
    SendButton = new QPushButton(this);
    SendButton->move(50,550);
    SendButton->setText("发送");
    SendButton->setFixedSize(250,35);
    connect(SendButton,SIGNAL(clicked()),this,SLOT(SendButtonClicked()));
    //发送记录
    RecordLabel = new QLabel(this);
    RecordLabel->move(550,50);
    RecordLabel->setText("发送记录");
    RecordText = new QTextEdit(this);
    RecordText->move(350,80);
    RecordText->setFixedSize(500,450);
    RecordText->setReadOnly(true);
    //记录所有发送文本
    AllRecordText = QString();
    //清除按钮
    ClearRecordTextButton = new QPushButton(this);
    ClearRecordTextButton->move(350,550);
    ClearRecordTextButton->setText("清空");
    ClearRecordTextButton->setFixedSize(500,35);
    connect(ClearRecordTextButton,SIGNAL(clicked()),
            this,SLOT(ClearRecordTextButtonClicked()));
}
ServerWindow::~ServerWindow()
{
}
void ServerWindow::ListenButtonClicked()
{
    //获取ip和端口
    Ip = IpInput->text();
    Port = PortInput->text().toInt();
    //进入侦听模式
    server = new QTcpServer(this);
    server->listen(QHostAddress(Ip),Port);
    connect(server,SIGNAL(newConnection()),this,SLOT(NewClientConnect()));
    listened = true;
    //显示侦听
    AllRecordText += "侦听中...\n";
    ListenButton->setEnabled(false);
    //禁止输入
    IpInput->setReadOnly(true);
    PortInput->setReadOnly(true);
    RecordText->setText(AllRecordText);
}
void ServerWindow::ClearRecordTextButtonClicked()
{
    //清空记录
    AllRecordText = QString();
    RecordText->setText(AllRecordText);
}
void ServerWindow::ChooseClientButtonClicked()
{
    //关闭服务端并在同位置显示客户端
    ClientWindow *client_window = new ClientWindow;
    client_window->resize(900,600);
    client_window->setFont(QFont("宋体",8));
    client_window->setWindowTitle("Socket TCP 通信");
    this->deleteLater();
    this->close();
    client_window->move(this->pos().x(),this->pos().y());
    client_window->show();
}
void ServerWindow::NewClientConnect()
{
    //显示
    AllRecordText += "已连接\n";
    RecordText->setText(AllRecordText);
    connected = true;
    //获得套接字并绑定
    socket = new QTcpSocket(this);
    socket = server->nextPendingConnection();
    connect(socket,SIGNAL(readyRead()),this,SLOT(ReadClientData()));
}
void ServerWindow::ReadClientData()
{
    //读取文本 按换行符分割 用QString::SkipEmptyParts跳过空行
    QString ReadData =  socket->readAll();
    QStringList TextInputList = ReadData.split(QRegExp("[\r\n]"),QString::SkipEmptyParts);
    //遍历取出显示内容
    for(int i =0;i<TextInputList.size();i++)
    {
        CurrentTime = QDateTime::currentDateTime();
        QString current_date =CurrentTime.toString("hh:mm:ss");
        TextInputList[i] +='\n';
        AllRecordText += "[" + current_date + "] " + \
                Ip + ":" + QString::number(Port) + ": " + \
                "<--: " + TextInputList[i];
    }
    //显示
    RecordText->setText(AllRecordText);
}
void ServerWindow::DisListenButtonClicked()
{
    if(listened)
    {
        //显示
        AllRecordText += "取消侦听\n";
        RecordText->setText(AllRecordText);
        //取消侦听 close是禁止连接 还是可以通信
        server->deleteLater();
        ListenButton->setEnabled(true);
        IpInput->setReadOnly(false);
        PortInput->setReadOnly(false);
        listened = false;
        connected = false;
    }
}
void ServerWindow::SendButtonClicked()
{
    if(connected)
    {
        //发送文本 要按换行符分割 可以用QString::SkipEmptyParts跳过空行
        QStringList TextInputList = TextInput->toPlainText().split(
                    QRegExp("[\r\n]"),QString::SkipEmptyParts);
        for(int i =0;i<TextInputList.size();i++)
        {
            CurrentTime = QDateTime::currentDateTime();
            QString current_date =CurrentTime.toString("hh:mm:ss");
            TextInputList[i] +='\n';
            AllRecordText += "[" + current_date + "] " + \
                    Ip + ":" + QString::number(Port) + ": " + \
                    "-->: " + TextInputList[i];
            socket->write(TextInputList[i].toUtf8());
        }
        RecordText->setText(AllRecordText);
    }
    else
    {
        AllRecordText += "未连接,请连接\n";
        RecordText->setText(AllRecordText);
    }
}
效果如下:
 
程序打包
为了让程序可以在多个平台上使用
我将qt程序分别在windows和linux编译成可执行文件,并打包依赖环境
windows打包环境采用qt自带的windeployqt,比较简单
linux打包环境采用需要自己编写脚本文件,实现如下:
1.打包依赖项
编写了一个pack.sh用于打包依赖环境,内容如下
#!/bin/sh
dir=$(pwd) # current dir
liblist=$(ldd $1 | awk '{if (match($3,"/")){ printf("%s ",$3)}}') # get lib list
cp $liblist $dir # copy
通过ldd命令显示依赖环境,结合awk,match匹配获得动态链接库的绝对路径
最后通过cp命令将这些文件拷贝到当前文件夹、
2.运行时指定动态库链接路径
如果更换了一台电脑,它并不会默认当前文件夹为动态链接库搜索路径
所以需要一个与可执行文件同名的的脚本,内容如下
#!/bin/sh
appname=$(basename $0 | cut -d . -f1) 
dirname=$(dirname $0)
LD_LIBRARY_PATH=$dirname
export LD_LIBRARY_PATH
chmod +x $dirname/$appname
$dirname/$appname
程序实现的也就是设置动态链接库为该目录
所以在其他机器上只要运行这个socket.sh即可将可执行文件链接该文件夹的动态链接库并成功执行
3.打包成App
为了获得更加方便的可点击执行的可执行文件,我还编写了getDesktop.sh脚本,内容如下
exe="socketTCP"
file_name=$exe".desktop"
echo "#!/usr/bin/env xdg-open" > $file_name
echo "[Desktop Entry]" >> $file_name
echo "Version=1.0" >> $file_name
echo "Type=Application" >> $file_name
echo "Terminal=false" >> $file_name
echo "Exec="$(pwd)"/"$exe".sh" >> $file_name
echo "Name="$exe >> $file_name
chmod +x $file_name
实现原理就是根据所在路径编写可点击执行的.desktop文件
