什么是 Socket 通信?
Socket 通信是一种网络通信方式,允许不同设备之间通过网络进行数据交换。它是网络编程的基础,用于实现客户端和服务器之间的通信。Socket 通信支持多种协议,其中最常用的是 TCP 和 UDP。
什么是 TCP 和 UDP 协议?
- TCP(Transmission Control Protocol):传输控制协议,是一种面向连接的、可靠的、基于字节流的通信协议。在通信之前,客户端和服务器需要建立连接,确保数据的可靠传输。
- UDP(User Datagram Protocol):用户数据报协议,是一种无连接的通信协议。它不保证数据的可靠性,但具有传输速度快、开销小的特点,适用于对实时性要求高的场景。
效果截图
打包下载地址:
UDP远程桌面显示_主机端.ahk
/* 使用说明: 先在一台电脑上打开主机端脚本,然后修改【连接端】的第27行参数2的对应IP地址来连接通讯做摄像头视频采集显示。 如果是局域网连接则没有什么限制,填写正确的主机端IP地址即可连接。 如果想互联网连接,至少需要主机端是公网IP+路由端口映射才能互相连接。内网穿透连接没试过 参考资料:https://www.autohotkey.com/boards/viewtopic.php?p=74463#p74463 */ SetBatchLines -1 SetWorkingDir %A_ScriptDir% #Include %A_ScriptDir%\Socket.ahk #include %A_ScriptDir%\Gdip_all.ahk pToken := Gdip_StartUp() ; GDI库加载 回复人IP := A_Args[1], 回复人端口 := A_Args[2] ; 首次运行脚本时,回复人IP会为空。此时会建立任意人连接的等待。被连接后会自动更新连接者的IP和端口建立双向通信 if (回复人IP="") { ; 建立任意连接的接收对象 myUdpIn := new SocketUDP() ; 创建一个新的 udp 对象 "myUdpIn" myUdpIn.bind(["addr_any", 65511]) ; "addr_any"为接收所有IP,也可接受指定IP推送 myUdpIn.onRecv := Func("myRecvCallback") ; 对传入消息执行回调 "myRecvCallback"。 } else { ; 建立指定IP的收发对象 myUdp := New BidirectionalUDP() ; 参1建立自身接收IP端口,参2为要通信发送的IP端口 myUdp.Bidirectional(["0.0.0.0", 65511], [回复人IP, 回复人端口]) ; 指定后只能与参2的IP端口进行通信 myUdp.onRecv := Func("myRecvCallback") ; 对传入消息执行回调 "myRecvCallback"。 SetTimer 远程桌面, 20 } Gui, Add, Picture, vPic, % "HBITMAP:*" hBitmap Gui, Show, w1000 h620, AHK窗口显示 Return 远程桌面: pBitmap := Gdip_BitmapFromScreen("0|0|1000|600") ; 屏幕截图 , outbytes := GdiPlus_SaveImageToBuffer(pBitmap, outBuf, "JPG", 30) , myUdp.Send(&outBuf, outbytes) , Gdip_DisposeImage(pBitmap) Return myRecvCallback(this) { Global inbytes := this.Recv(inbuf) ; 用于接收发送端的二进制,不能大于65536字节=64KB if (this.GetRecvIPfrom()!=回复人IP) or (this.GetRecvPort()!=回复人端口) RunWait % A_AhkPath " /r """ A_ScriptFullPath """ " this.GetRecvIPfrom() " " this.GetRecvPort() DeleteObject(hBitmap) , hBitmap := GDIPlus_hBitmapFromBuffer(inbuf, inbytes) GuiControl, , Pic, % "HBITMAP:*" hBitmap ; ToolTip % "发送者IP:" this.GetRecvIPfrom() ":" this.GetRecvPort() } ; 在SocketUDP类的基础上扩展延伸 Class BidirectionalUDP extends SocketUDP { Bidirectional(inHostPort, outHostPort) { if ((this.socket != -1) || (!(faddr := inNext := this.GetAddrInfo(inHostPort)))) return 0 if ((this.socket != -1) || (!(outNext := this.GetAddrInfo(outHostPort)))) return 0 While inNext { inSockAddrlen := NumGet(inNext + 0, 16, "Uint") , inSockAddr := NumGet(inNext + 0, 16 + (2 * A_PtrSize), "Ptr") if ((this.socket := DllCall("ws2_32\socket", "int", NumGet(inNext+0, 4, "int"), "int", this.SocketType, "int", this.ProtocolId, "Ptr")) != -1) { if (DllCall("ws2_32\bind", "Ptr", this.Socket, "Ptr", inSockAddr, "Uint", inSockAddrlen, "int") = 0) { ; 建立接收 outSockAddrlen := NumGet(outNext + 0, 16, "Uint") , outSockAddr := NumGet(outNext + 0, 16 + (2 * A_PtrSize), "Ptr") if ((r := DllCall("ws2_32\WSAConnect", "Ptr", this.Socket, "Ptr", outSockAddr, "Uint", outSockAddrlen, "Ptr", 0, "Ptr", 0, "Ptr", 0, "Ptr", 0, "int")) = 0) { ; 建立发送 DllCall("ws2_32\freeaddrinfo", "Ptr", faddr) return this.EventProcRegister(this.FD_READ | this.FD_CLOSE) } } this.Disconnect() } inNext := NumGet(inNext + 0, 16 + (3 * A_PtrSize), "Ptr") } throw Exception("Error connecting") } } GDIPlus_hBitmapFromBuffer(ByRef Buffer, nSize) { ; by SKAN Static hBitmap hData := DllCall("GlobalAlloc", "Uint", 2, "Uint", nSize) , pData := DllCall("GlobalLock", "Uint", hData) , DllCall("RtlMoveMemory", "Uint", pData, "Uint", &Buffer, "Uint", nSize) , DllCall("GlobalUnlock", "Uint", hData) , DllCall("ole32\CreateStreamOnHGlobal", "Uint", hData, "int", True, "UIntP", pStream) , DllCall("gdiplus\GdipCreateBitmapFromStream", "Uint", pStream, "UIntP", pBitmap) , DllCall("DeleteObject", "UPtr", hBitmap) , DllCall("gdiplus\GdipCreateHBITMAPFromBitmap", "Uint", pBitmap, "UIntP", hBitmap, "Uint", DllCall("ntdll\RtlUlongByteSwap", "Uint", DllCall("GetSysColor", "int", 15 ) <<8 ) | 0xFF000000) , DllCall("GlobalFree", "Ptr", hData) , DllCall("gdiplus\GdipDisposeImage", "Uint", pBitmap) , DllCall(NumGet( NumGet(1*pStream)+8 ), "Uint", pStream) ; IStream::Release Return hBitmap } GdiPlus_SaveImageToBuffer(pBitmap, ByRef Buffer, Type:="JPG", Quality:=75) { nCount := nSize := pStream := hData := _p := 0 If !RegExMatch(Type, "^(?i:BMP|DIB|RLE|JPG|JPEG|JPE|JFIF|GIF|TIF|TIFF|PNG)$") Return -1 Type := "." Type , DllCall("gdiplus\GdipGetImageEncodersSize", "uint*", nCount, "uint*", nSize) , VarSetCapacity(ci, nSize) , DllCall("gdiplus\GdipGetImageEncoders", "uint", nCount, "uint", nSize, "UPtr", &ci) If !(nCount && nSize) Return -2 If A_IsUnicode { StrGet_Name := "StrGet" , N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { sString := %StrGet_Name%(NumGet(ci, (idx := (48+7*A_PtrSize)*(A_Index-1))+32+3*A_PtrSize), "UTF-16") If !InStr(sString, "*" Type) Continue pCodec := &ci+idx Break } } Else { N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { Location := NumGet(ci, 76*(A_Index-1)+44) , nSize := DllCall("WideCharToMultiByte", "uint", 0, "uint", 0, "uint", Location, "int", -1, "uint", 0, "int", 0, "uint", 0, "uint", 0) , VarSetCapacity(sString, nSize) , DllCall("WideCharToMultiByte", "uint", 0, "uint", 0, "uint", Location, "int", -1, "str", sString, "int", nSize, "uint", 0, "uint", 0) If !InStr(sString, "*" Type) Continue pCodec := &ci+76*(A_Index-1) Break } } If !pCodec Return -3 If (Quality!=75) { Quality := (Quality < 0) ? 0 : (Quality > 100) ? 100 : Quality If (quality>95 && toBase64=1) Quality := 95 If RegExMatch(Type, "^\.(?i:JPG|JPEG|JPE|JFIF)$") { DllCall("gdiplus\GdipGetEncoderParameterListSize", "UPtr", pBitmap, "UPtr", pCodec, "uint*", nSize) , VarSetCapacity(EncoderParameters, nSize, 0) , DllCall("gdiplus\GdipGetEncoderParameterList", "UPtr", pBitmap, "UPtr", pCodec, "uint", nSize, "UPtr", &EncoderParameters) , nCount := NumGet(EncoderParameters, "UInt") , N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { elem := (24+A_PtrSize)*(A_Index-1) + 4 + (pad := A_PtrSize = 8 ? 4 : 0) If (NumGet(EncoderParameters, elem+16, "UInt") = 1) && (NumGet(EncoderParameters, elem+20, "UInt") = 6) { _p := elem+&EncoderParameters-pad-4 NumPut(Quality, NumGet(NumPut(4, NumPut(1, _p+0)+20, "UInt")), "UInt") Break } } } } ; Save Image to Stream and copy it to Buffer DllCall("ole32\CreateStreamOnHGlobal", "Uint", 0, "int", 1, "UintP", pStream) , DllCall("gdiplus\GdipSaveImageToStream", "Uint", pBitmap, "Uint", pStream, "Uint", pCodec, "Uint", _p) , DllCall("gdiplus\GdipDisposeImage", "Uint", pBitmap) , DllCall("ole32\GetHGlobalFromStream", "Uint", pStream, "UintP", hData) , pData := DllCall("GlobalLock", "Uint", hData) , nSize := DllCall("GlobalSize", "Uint", pData) , VarSetCapacity(Buffer, nSize, 0) , DllCall("RtlMoveMemory", "Uint", &Buffer, "Uint", pData, "Uint", nSize) , DllCall("GlobalUnlock", "Uint", hData) , DllCall(NumGet(NumGet( 1*pStream ) + 8), "Uint", pStream) , DllCall("GlobalFree", "Uint", hData) Return nSize }
UDP远程桌面显示_连接端.ahk
/* 使用说明: 先在一台电脑上打开主机端脚本,然后修改第27行的参数2的对应IP地址来连接通讯做摄像头视频采集显示。 如果是局域网连接则没有什么限制,填写正确的主机端IP地址即可连接。 如果想互联网连接,至少需要主机端是公网IP+路由端口映射才能互相连接。内网穿透连接没试过 参考资料:https://www.autohotkey.com/boards/viewtopic.php?p=74463#p74463 */ SetBatchLines -1 SetWorkingDir %A_ScriptDir% #Include %A_ScriptDir%\Socket.ahk #include %A_ScriptDir%\Gdip_all.ahk pToken := Gdip_StartUp() ; GDI库加载 ; 建立指定IP的收发对象 myUdp := New BidirectionalUDP() ; 参1建立自身接收IP端口,参2为要通信发送的IP端口【参2一定要设置正确的主机IP和端口】 myUdp.Bidirectional(["0.0.0.0", 65500], ["10.0.0.10", 65511]) ; 指定后只能与参2的IP端口进行通信 myUdp.onRecv := Func("myRecvCallback") ; 对传入消息执行回调 "myRecvCallback"。 myUdp.SendText("首次连接确认") SetTimer 远程桌面, 20 Gui, Add, Picture, vPic, % "HBITMAP:*" hBitmap Gui, Show, w1000 h620, AHK窗口显示 Return 远程桌面: pBitmap := Gdip_BitmapFromScreen("0|0|1000|600") ; 屏幕截图 , outbytes := GdiPlus_SaveImageToBuffer(pBitmap, outBuf, "JPG", 30) , myUdp.Send(&outBuf, outbytes) , Gdip_DisposeImage(pBitmap) Return myRecvCallback(this) { Global inbytes := this.Recv(inbuf) ; 用于接收发送端的二进制,不能大于65536字节=64KB , DeleteObject(hBitmap) , hBitmap := GDIPlus_hBitmapFromBuffer(inbuf, inbytes) GuiControl, , Pic, % "HBITMAP:*" hBitmap ; ToolTip % "发送者IP:" this.GetRecvIPfrom() ":" this.GetRecvPort() } ; 在SocketUDP类的基础上扩展延伸 Class BidirectionalUDP extends SocketUDP { Bidirectional(inHostPort, outHostPort) { if ((this.socket != -1) || (!(faddr := inNext := this.GetAddrInfo(inHostPort)))) return 0 if ((this.socket != -1) || (!(outNext := this.GetAddrInfo(outHostPort)))) return 0 While inNext { inSockAddrlen := NumGet(inNext + 0, 16, "Uint") , inSockAddr := NumGet(inNext + 0, 16 + (2 * A_PtrSize), "Ptr") if ((this.socket := DllCall("ws2_32\socket", "int", NumGet(inNext+0, 4, "int"), "int", this.SocketType, "int", this.ProtocolId, "Ptr")) != -1) { if (DllCall("ws2_32\bind", "Ptr", this.Socket, "Ptr", inSockAddr, "Uint", inSockAddrlen, "int") = 0) { ; 建立接收 outSockAddrlen := NumGet(outNext + 0, 16, "Uint") , outSockAddr := NumGet(outNext + 0, 16 + (2 * A_PtrSize), "Ptr") if ((r := DllCall("ws2_32\WSAConnect", "Ptr", this.Socket, "Ptr", outSockAddr, "Uint", outSockAddrlen, "Ptr", 0, "Ptr", 0, "Ptr", 0, "Ptr", 0, "int")) = 0) { ; 建立发送 DllCall("ws2_32\freeaddrinfo", "Ptr", faddr) return this.EventProcRegister(this.FD_READ | this.FD_CLOSE) } } this.Disconnect() } inNext := NumGet(inNext + 0, 16 + (3 * A_PtrSize), "Ptr") } throw Exception("Error connecting") } } GDIPlus_hBitmapFromBuffer(ByRef Buffer, nSize) { ; by SKAN Static hBitmap hData := DllCall("GlobalAlloc", "Uint", 2, "Uint", nSize) , pData := DllCall("GlobalLock", "Uint", hData) , DllCall("RtlMoveMemory", "Uint", pData, "Uint", &Buffer, "Uint", nSize) , DllCall("GlobalUnlock", "Uint", hData) , DllCall("ole32\CreateStreamOnHGlobal", "Uint", hData, "int", True, "UIntP", pStream) , DllCall("gdiplus\GdipCreateBitmapFromStream", "Uint", pStream, "UIntP", pBitmap) , DllCall("DeleteObject", "UPtr", hBitmap) , DllCall("gdiplus\GdipCreateHBITMAPFromBitmap", "Uint", pBitmap, "UIntP", hBitmap, "Uint", DllCall("ntdll\RtlUlongByteSwap", "Uint", DllCall("GetSysColor", "int", 15 ) <<8 ) | 0xFF000000) , DllCall("GlobalFree", "Ptr", hData) , DllCall("gdiplus\GdipDisposeImage", "Uint", pBitmap) , DllCall(NumGet( NumGet(1*pStream)+8 ), "Uint", pStream) ; IStream::Release Return hBitmap } GdiPlus_SaveImageToBuffer(pBitmap, ByRef Buffer, Type:="JPG", Quality:=75) { nCount := nSize := pStream := hData := _p := 0 If !RegExMatch(Type, "^(?i:BMP|DIB|RLE|JPG|JPEG|JPE|JFIF|GIF|TIF|TIFF|PNG)$") Return -1 Type := "." Type , DllCall("gdiplus\GdipGetImageEncodersSize", "uint*", nCount, "uint*", nSize) , VarSetCapacity(ci, nSize) , DllCall("gdiplus\GdipGetImageEncoders", "uint", nCount, "uint", nSize, "UPtr", &ci) If !(nCount && nSize) Return -2 If A_IsUnicode { StrGet_Name := "StrGet" , N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { sString := %StrGet_Name%(NumGet(ci, (idx := (48+7*A_PtrSize)*(A_Index-1))+32+3*A_PtrSize), "UTF-16") If !InStr(sString, "*" Type) Continue pCodec := &ci+idx Break } } Else { N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { Location := NumGet(ci, 76*(A_Index-1)+44) , nSize := DllCall("WideCharToMultiByte", "uint", 0, "uint", 0, "uint", Location, "int", -1, "uint", 0, "int", 0, "uint", 0, "uint", 0) , VarSetCapacity(sString, nSize) , DllCall("WideCharToMultiByte", "uint", 0, "uint", 0, "uint", Location, "int", -1, "str", sString, "int", nSize, "uint", 0, "uint", 0) If !InStr(sString, "*" Type) Continue pCodec := &ci+76*(A_Index-1) Break } } If !pCodec Return -3 If (Quality!=75) { Quality := (Quality < 0) ? 0 : (Quality > 100) ? 100 : Quality If (quality>95 && toBase64=1) Quality := 95 If RegExMatch(Type, "^\.(?i:JPG|JPEG|JPE|JFIF)$") { DllCall("gdiplus\GdipGetEncoderParameterListSize", "UPtr", pBitmap, "UPtr", pCodec, "uint*", nSize) , VarSetCapacity(EncoderParameters, nSize, 0) , DllCall("gdiplus\GdipGetEncoderParameterList", "UPtr", pBitmap, "UPtr", pCodec, "uint", nSize, "UPtr", &EncoderParameters) , nCount := NumGet(EncoderParameters, "UInt") , N := (A_AhkVersion < 2) ? nCount : "nCount" Loop %N% { elem := (24+A_PtrSize)*(A_Index-1) + 4 + (pad := A_PtrSize = 8 ? 4 : 0) If (NumGet(EncoderParameters, elem+16, "UInt") = 1) && (NumGet(EncoderParameters, elem+20, "UInt") = 6) { _p := elem+&EncoderParameters-pad-4 NumPut(Quality, NumGet(NumPut(4, NumPut(1, _p+0)+20, "UInt")), "UInt") Break } } } } ; Save Image to Stream and copy it to Buffer DllCall("ole32\CreateStreamOnHGlobal", "Uint", 0, "int", 1, "UintP", pStream) , DllCall("gdiplus\GdipSaveImageToStream", "Uint", pBitmap, "Uint", pStream, "Uint", pCodec, "Uint", _p) , DllCall("gdiplus\GdipDisposeImage", "Uint", pBitmap) , DllCall("ole32\GetHGlobalFromStream", "Uint", pStream, "UintP", hData) , pData := DllCall("GlobalLock", "Uint", hData) , nSize := DllCall("GlobalSize", "Uint", pData) , VarSetCapacity(Buffer, nSize, 0) , DllCall("RtlMoveMemory", "Uint", &Buffer, "Uint", pData, "Uint", nSize) , DllCall("GlobalUnlock", "Uint", hData) , DllCall(NumGet(NumGet( 1*pStream ) + 8), "Uint", pStream) , DllCall("GlobalFree", "Uint", hData) Return nSize }
声明:站内资源为整理优化好的代码上传分享与学习研究,如果是开源代码基本都会标明出处,方便大家扩展学习路径。请不要恶意搬运,破坏站长辛苦整理维护的劳动成果。本站为爱好者分享站点,所有内容不作为商业行为。如若本站上传内容侵犯了原著者的合法权益,请联系我们进行删除下架。
评论(0)