百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Tcp Socket 编程之Delphi与其他语言的字节码通信

yuyutoo 2024-12-31 15:36 6 浏览 0 评论

关键字:Tcp Scoket、Delphi、Indy、Python、Twisted

对于 Tcp Socket 编程,异种语言之间的通信在日常开发中经常会用到。今天,我们通过 Delphi 和 Python 语言来做一个 Socket 通信的示例。通信数据采用字节码(也就是字节数组)形式来直接传输。

客户端:Delphi、Indy

服务端:Python、Twisted

由于是不同语言之间的数据通信,显然,在 Delphi 中使用 Record 进行通信的可能性就不大了。所以,在 Delphi 中采用 Indy 进行数据通信的情况下,主要会使用到如下一些数据类型:

  • TIdBytes
  • TIdBuffer

8.1 TIdBytes

该类型被定义在 IdGlobal 单元中,定义如下:

TIdBytes = TBytes;

TBytes 定义如下:

TBytes = array of Byte;

可见是一个字节数组。

8.2 TIdBuffer

该类定义在 IdBuffer 单元中,定义如下:

TIdBuffer = class(TIdBaseObject);

TIdBuffer 是一个 TObject 子类,它实现了一个用于 Indy 库中输入和输出操作的缓冲区。TIdBuffer用作通信层的读和/或写缓冲器。TIdBuffer 针对在缓冲区末尾添加数据和从缓冲区开始提取数据进行了优化。

TIdBuffer 隔离了在 Indy 库支持的平台上实现的内存分配和指针操作之间的差异。

TIdBuffer实现了用于管理缓冲区类实例的大小、增长和编码的属性,包括:

Property

Usage

Capacity

为缓冲区分配的最大尺寸。

Encoding

用于添加到缓冲区的字符串值。

GrowthFactor

分配额外缓冲区存储的阈值。

Size

缓冲区存储中的字节数。

AsString

作为字符串数据类型的缓冲区内容。

在开发中,我们主要会使用其 Write 方法和 Extract 系列方法。

Write 方法:

  • TIdBuffer.Write (Byte, Integer)
  • TIdBuffer.Write (Cardinal, Integer)
  • TIdBuffer.Write (Int64, Integer)
  • TIdBuffer.Write (TIdBytes, Integer)
  • TIdBuffer.Write (TIdIPv6Address, Integer)
  • TIdBuffer.Write (TIdStream, Integer)
  • TIdBuffer.Write (Word, Integer)
  • TIdBuffer.Write (string, TIdEncoding, Integer)

Extract 系列方法:

  • Extract
  • ExtractToByte
  • ExtractToBytes
  • ExtractToCardinal
  • ExtractToIdBuffer
  • ExtractToInt64
  • ExtractToIPv6
  • ExtractToStream
  • ExtractToWord

具体可以参考 Indy 的帮助文档。

8.3 示例阐述

示例:客户端定时实时检测所在机器的屏幕分辨率上行到服务端,服务端接收到数据后,根据其屏幕分辨率随机生成一个坐标并下发给客户端,客户端将应用程序的窗体位置放置到相应的坐标上。

  • 当发送屏幕宽度时,回复 x 坐标;
  • 当发送屏幕高度时,回复 y 坐标;
  • 当发送结束时,回复结束;客户端收到结束时更改窗体位置;

通信协议:

8.4 服务端

服务端采用 Python 及其 Twisted 框架开发,业务逻辑也比较简单,所以,不做赘述,代码如下:

# coding: utf-8
from twisted.internet.protocol import Protocol
from twisted.internet.protocol import Factory
from twisted.internet.endpoints import TCP4ServerEndpoint
from twisted.internet import reactor
import struct
import random

class Device(Protocol):
    def connectionMade(self):
        print('Connected from %s:%s.' % (self.transport.getPeer().host, self.transport.getPeer().port))
        self.transport.write('$Welcome!#39;.encode('utf-8'))

    def connectionLost(self, reason):
        print('Disconnected from %s:%s.' % (self.transport.getPeer().host, self.transport.getPeer().port))
        print(reason)

    def dataReceived(self, data):
        # 接收数据
        print('Recieved from %s:%s.' % (
            self.transport.getPeer().host, self.transport.getPeer().port))
        # 输出十六进制表示
        print(data.hex())
        total_length = len(data)
        print('命令长度: ', total_length)
        # 初始化宽度和高度
        width = height = 0
        # 解包,取长度
        start, length, tmp, end = struct.unpack('>1s H {0}s 1s'.format(total_length - 4), data)
        # 解包,取各个字段
        start, length, part, desc, value, end = struct.unpack('>1s H 1s {0}s H 1s'.format(length - 3), data)
        print(start, length, part, desc, value, end)
        
        if part == b'W':
            # 宽度处理
            width = value
            print(desc.decode('utf-8'), width)
            # 计算 x 坐标
            x = random.randint(0, width)
            command = struct.pack('>1s H 1s {0}s H 1s'.format(4), b'#', 4 + 3, b'X', b'left', x, b'#')
            print(command.hex())
            # 下发指令
            self.transport.write(command)

        if part == b'H':
            # 宽度处理
            height = value
            print(desc.decode('utf-8'), height)
            # 计算 y 坐标
            y = random.randint(0, height)
            command = struct.pack('>1s H 1s {0}s H 1s'.format(3), b'#', 3 + 3, b'Y', b'top', y, b'#')
            print(command.hex())
            # 下发指令
            self.transport.write(command)
        if part == b'E':
            # 结束处理
            print(desc.decode('utf-8'))
            command = struct.pack('>1s H 1s {0}s H 1s'.format(3), b'#', 3 + 3, b'E', b'end', 0, b'#')
            print(command.hex())
            # 下发结束指令
            self.transport.write(command)


class DeviceFactory(Factory):
    def buildProtocol(self, addr):
        print(addr)  # addr为创建连接时客户端的地址
        return Device()


endpoint = TCP4ServerEndpoint(reactor, 9123)
endpoint.listen(DeviceFactory())
reactor.run()

注:代码仅用于测试,没有异常处理,多线程处理等

8.5 客户端

客户端采用 Typhon 、Indy 组件进行开发,界面如下:

界面主要组件元素及其属性:

组件

属性

说明

TEdit

name

HostEdit

主机

TSpinEdit

name

PortSpinEdit

端口

TButton

name

ConnectButton

连接按钮


caption

连接


TButton

name

DisconnectButton

断开按钮


caption

断开


TMemo

name

DataMemo

传输的数据内容显示


readonly

True


TIdTcpClient

name

IdTCPClient


TTimer

name

Timer1



interval

1000


客户端主要代码包括:TTimer 的定时事件发送数据和读取线程接收数据。

  • TTimer 的定时事件发送数据
procedure TForm1.Timer1Timer(Sender: TObject);
var
  w, h: Integer;
  bytes: TIdBytes;
begin
  // 定时器
  Count:=Count+1;
  if Count > 99 then Count:=1;

  w:=Screen.Width;
  h:=Screen.Height;

  if IdTCPClient.Connected then
  try
    if Count mod 3 = 1 then
    begin
      bytes:=BuildOrder('W', 'width', w);
      IdTCPClient.IOHandler.Write(bytes);
      DataMemo.Lines.Add('发送 width: ' + inttostr(w));
    end;

    if Count mod 3 = 2 then
    begin
      bytes:=BuildOrder('H', 'height', h);
      IdTCPClient.IOHandler.Write(bytes);
      DataMemo.Lines.Add('发送 height: ' + inttostr(h));
    end;

    if Count mod 3 = 0 then
    begin
      bytes:=BuildOrder('E', 'end', 0);
      IdTCPClient.IOHandler.Write(bytes);
      DataMemo.Lines.Add('发送 end: ' + inttostr(0));
    end;
  except
  end;
end;

在上面的代码中主要完成每次定时器触发事件,根据计数器 Count 的值来决定发送什么,发送数据时,调用了一个 BuildOrder 函数,该函数完成指令构建的过程,代码如下:

function BuildOrder(Part, Desc: String; Value: UInt16): TIdBytes;
var
  buffer: TIdBuffer;
  bytes: TIdBytes;
  l: UInt16;
begin
  buffer:=TIdBuffer.Create;

  l := length(Desc) + 3;

  buffer.Write(bytesof('#'));
  buffer.Write(l);
  buffer.Write(bytesof(Part));
  buffer.Write(bytesof(Desc));
  buffer.Write(Value);
  buffer.Write(bytesof('#'));
  buffer.ExtractToBytes(bytes, -1, False);
  buffer.Free;

  Result:=bytes;
end;

构建指令时,主要采用了 TIdBuffer 来组装数据,然后导出 TIdBytes,在组装过程中,对于字符串可以使用 bytesof 函数获取其字节数据,返回值为字节数组。

  • 读取线程接收数据
unit unitreadthread;

{$mode ObjFPC}{$H+}

interface

uses
  Classes, SysUtils, IdGlobal, IdBuffer;

type
  TReadThread = class(TTHread)
    private
      procedure AddTextToMainDataMemo;
      procedure ChangeMainPos;
    protected
      procedure Execute; Override;
    end;

var
  info: String;
  X, Y: Integer;

implementation

uses unitmain;

procedure TReadThread.Execute;
var
  bytes: TIdBytes;
  buffer: TIdBuffer;
  startflag, endflag, part, desc: String;
  length, value: UInt16;
begin
  X:=0;
  Y:=0;

  while not Self.Terminated do
  begin
    if not Form1.IdTCPClient.Connected then
      Self.Terminate
    else
    try
      Form1.IdTCPClient.IOHandler.ReadBytes(bytes, -1, False);

      buffer:=TIdBuffer.Create(bytes);
      startflag := buffer.ExtractToString(1);
      length := buffer.ExtractToUInt16(-1);
      if length > 10 then continue;
      part := buffer.ExtractToString(1);
      desc := buffer.ExtractToString(length - 3);
      value := buffer.ExtractToUInt16(-1);
      endflag := buffer.ExtractToString(1);

      info:= startflag + ' ' + inttostr(length) + ' ' + part + ' ' + desc + ' ' + inttostr(value) + ' ' + endflag;
      Synchronize(@AddTextToMainDataMemo);
      buffer.Free;

      if part = 'X' then X:=value;
      if part = 'Y' then Y:=value;
      if part = 'E' then
      begin
        info:='改变窗体位置:(' + inttostr(X) + ',' + inttostr(Y) + ')';
        Synchronize(@AddTextToMainDataMemo);
        Synchronize(@ChangeMainPos);
      end;
    except
    end;
  end;
end;

procedure TReadThread.AddTextToMainDataMemo;
begin
  Form1.DataMemo.Lines.Add(info);
end;

procedure TReadThread.ChangeMainPos;
begin
  Form1.Left:=X;
  Form1.Top:=Y;
end;

end. 

读取线程在收到数据时,首先通过 IdTCPClient.IOHandler.ReadBytes(bytes, -1, False) 来获取接收到的数据,然后采用 TIdBuffer.Create 创建 TIdBuffer 的实例,其参数为 bytes,最后通过 TIdBuffer 的 Extract 系列函数分解其中的数据。

注:以上代码仅用于测试

8.6 运行效果

相关推荐

建筑福利-pdf转dwg格式转换器,再也不用描图-极客青年

作为一名经常熬夜画图的建筑狗或者cad用户,你体验过pdf图纸描图到cad吗?前几天一个老同学找我,说他的毕业设计需要我帮忙,发给我一份pdf图纸文件,问我怎么把pdf图纸转换成dwg格式。机智的我灵...

想学 HTML,不知从何入手?看完这篇文章你就知道了

很多人都说HTML是一门很简单的语言,看看书,看看视频就能读懂。但是,如果你完全没有接触过,就想通过看一遍教程,背背标签,想要完全了解HTML,真的有点太天真了。HTML中文...

「前端」HTML之结构

今天继续为大家分享前端的知识,如果对前端比较感兴趣的小伙伴,可以关注我,我会更大家继续分享更多与前端相关的内容,当然如果内容中又不当的或者文字错误的,欢迎大家在评论区留言,我会及时修改纠正。1.初识H...

手把手教你使用Python网络爬虫下载一本小说(附源码)

大家好,我是Python进阶者。前言前几天【磐奚鸟】大佬在群里分享了一个抓取小说的代码,感觉还是蛮不错的,这里分享给大家学习。...

用于处理pdf文件格式的转换器

在上传过程中如果单个文件太大则容易中断,而且文件太大的话对与存储也有些弊端。那么我们应该想到将文件进行压缩(注意这里压缩指的是不改变文件格式的压缩,而不是用变成压缩文件。这里就将以下用专门的软件压缩P...

乐书:在线 Kindle 电子书制作和转换工具

之前Kindle伴侣曾推荐过可以在Windows和Mac系统平台上运行的kindle电子书制作软件Sigil(教程),用它可以制作出高质量的的ePub格式电子书,当然最后还需要通...

付费文档怎么下载?教你5种方法,任意下载全网资源

网上查资料的时候,经常遇到需要注册登录或者付费的才能复制或者是下载,遇到这种情况大多数人都会选择重新查。...

捡来的知识!3种方法随便复制网页内容,白嫖真香呀

网上的资源真的多,所以许多人常常会从网上找资料。我们看到感兴趣的内容,第一时间可能会想要收入囊中。比如说截个图啊,或者挑选有意思的句子复制粘贴,记录下来。可是,有些时候,却会遇到这样的情况:1、内容不...

AI的使用,生成HTML网页。

利用deepseek,豆包,kimi以及通义千问,写入相同的需求。【写一个网页,实现抽奖功能,点击“开始”,按键显示“停止”,姓名开始显示在屏幕上,人员包括:“张三”,“里斯”,“Bool”,“流水废...

pdf转换成jpg转换器 4.1 官方正式版

pdf转换成jpg工具软件简介pdf转换成jpg转换器是一款界面简洁,操作方便的pdf转换成jpg转换器。pdf转换成jpg转换器可以将PDF文档转换为JPG,BMP,GIF,PNG,TIF图片文件。...

办公必备的office转换成pdf转换器怎么用?

2016-02-2415:53:37南方报道网评论(我要点评)字体刚从校园走出社会,对于快节奏的办公环境,难免会觉得有些吃力。在起步阶段力求将手头上的事情按时完工不出错,但是渐渐的你会发现,别人只...

为什么PDF转Word大多要收费?

PDF转Word大多都要收费?并非主要是因为技术上的难度,而是基于多方面的商业和版权考虑的,下面给大家浅分析下原因:...

如何用python生成简单的html report报告

前提:用python写了一个简单的log分析,主要也就是查询一些key,value出来,后面也可以根据需求增加。查询出来后,为了好看,搞个html表格来显示。需要的组件:jinja2flask...

学用系列|如何搞定word批量替换修改和格式转换?这里一站搞定

想必不少朋友都会碰到批量修改word文档内容、压缩文档图片、文件格式转换等重复性文档处理工作的需要,今天胖胖老师就推荐给大家一个免费工具XCLWinKits,一站搞定你所有的需要。什么是XCLWinK...

这款PDF文档转换神器,能帮你解决PDF使用中的许多难点

不管是平时的学习还是工作,相信许多朋友都经常接触PDF文件。可以说,PDF文件在我们的日常办公学习过程中的重要性和Word文档一样重要。在之前的更新中,小编介绍了几款非常不错的PDF文档格式转换软件,...

取消回复欢迎 发表评论: