C++和pascal之间的通信
// MyFirst.cpp : 定义控制台应用程序的入口点。
//
#include "stdafx.h"
#include "Winsock2.h"
#pragma comment(lib,"ws2_32.lib")
int _tmain(int argc, _TCHAR* argv[])
{
WORD wVersionRequested;
WSADATA wsaData;
int err;
wVersionRequested = MAKEWORD(1,1);
err = WSAStartup(wVersionRequested,&wsaData);
if (err!=0)
{
return 0;
}
if (LOBYTE(wsaData.wVersion)!=1||
HIBYTE(wsaData.wVersion)!=1)
{
WSACleanup();
return 0;
}
SOCKET sockSrv=socket(AF_INET,SOCK_STREAM,0);
SOCKADDR_IN addrSrv;
addrSrv.sin_addr.S_un.S_addr=htonl(INADDR_ANY);
addrSrv.sin_family=AF_INET;
addrSrv.sin_port=htons(5101);
bind(sockSrv,(SOCKADDR*)&addrSrv,sizeof(SOCKADDR));
listen(sockSrv,5);
SOCKADDR_IN addrClient;
int len=sizeof(SOCKADDR);
while(1)
{
SOCKET sockConn=accept(sockSrv,(SOCKADDR*)&addrClient,&len);
char sendBuf[100];
// sprintf_s(sendBuf,"Welcome %s to here!",inet_ntoa(addrClient.sin_addr));
/*send(sockConn,sendBuf,strlen(sendBuf)+1,0);*send函数是将指定长度的数据复制到TCP的系统缓冲区,然后系统缓冲区负责发送流数据到接收端的缓冲区
recv负责从缓冲区复制数据到指定存储地址字节不一样又没事的,recv每次不一定接收固定的数据的*/
char recvBuf[100];
memset(recvBuf,0,sizeof(recvBuf));
recv(sockConn,recvBuf,sizeof(recvBuf),0);
printf("%s\n",recvBuf);
memset(recvBuf,0,sizeof(recvBuf));
recv(sockConn,recvBuf,sizeof(recvBuf),0);
printf("%s\n",recvBuf);
closesocket(sockConn);}
return 0;
}
网上找的一段代码用C++重写了一遍然后用pascal写的客户端发送;
发现总是无法接受到正常的数据,在经过N边检查之后,才明白。
问题出现在客户端发送哪里有一个数据大小验证。
unit uSendMsg;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls,IdTCPClient,WinSock, IdBaseComponent, IdComponent,
IdTCPConnection;
type
TFSendMsg = class(TForm)
btnsend: TButton;
Edit1: TEdit;
Edit2: TEdit;
Label1: TLabel;
Label2: TLabel;
Button1: TButton;
Memo1: TMemo;
Label3: TLabel;
Edit3: TEdit;
Label4: TLabel;
Button2: TButton;
dlg: TOpenDialog;
procedure btnsendClick(Sender: TObject);
procedure Button1Click(Sender: TObject);
procedure FormShow(Sender: TObject);
procedure Button2Click(Sender: TObject);
procedure Memo1Click(Sender: TObject);
procedure Edit3Click(Sender: TObject);
private
{ Private declarations }
client: TIdTCPClient;
procedure sendSplitText(client: TIdTCPClient; s: String);
Procedure SendFile(IFileName:string);
public
{ Public declarations }
end;
var
FSendMsg: TFSendMsg;
implementation
uses IdGlobal,IniFiles;
{$R *.dfm}
procedure TFSendMsg.btnsendClick(Sender: TObject);
var
iLength: Integer;
s,s1,sSendText: String;
flag: Boolean;
aText: string;
begin
try
if client.Connected() then
client.Disconnect();
if not client.Connected() then
begin
client.Host := trim(edit1.Text);
client.Port := strtoint(trim(edit2.Text));
client.Connect();
// showmessage('连接成功');
end;
except on E: Exception do
begin
showMessage('连接主机失败,错误描述:'+ E.Message);
exit;
end;
end;
if not memo1.ReadOnly then
begin
try
aText := trim(memo1.Text);
if atext = '' then begin
showmessage('No message send');
exit;
end;
iLength := Length(aText);
sSendText := intToStr(100000000 + iLength);
Delete(sSendText, 1, 1);
sSendText := sSendText + aText;
if client.Connected() then
sendSplitText(client,sSendText);
showmessage('发送成功');
except on E: Exception do
ShowMessage('发送失败,错误描述:'+ E.Message);
end;
end else
begin
// edit3.Text := 'E:\华腾项目软件\sendmsg\a.txt';
s1 := trim(edit3.Text);
if s1 = '' then begin
showmessage('请选择文件');
exit;
end;
try
if client.Connected() then
sendfile(s1);
showmessage('发送成功');
except on E: Exception do
ShowMessage('发送失败,错误描述:'+ E.Message);
end;
end;
end;
procedure TFSendMsg.Button1Click(Sender: TObject);
begin
close;
end;
procedure TFSendMsg.Button2Click(Sender: TObject);
begin
btnsend.Enabled := true;
edit3.ReadOnly := false;
memo1.ReadOnly := true;
if dlg.Execute then
edit3.Text := dlg.FileName;
end;
procedure TFSendMsg.Edit3Click(Sender: TObject);
begin
btnsend.Enabled := true;
edit3.ReadOnly := false;
memo1.ReadOnly := true;
end;
procedure TFSendMsg.FormShow(Sender: TObject);
var
s: string;
ini: TIniFile;
begin
client := TIdTCPClient.Create(nil);
s := ExtractFilePath(Application.ExeName)+'info.ini';
if not fileExists(s) then
begin
ShowMessage('未找到配置文件'+ s);
Application.Terminate;
exit;
end;
ini := TIniFile.Create(s);
Edit1.Text := ini.ReadString('info','ip','');
Edit2.Text := IntToStr(ini.ReadInteger('info','port',0));
Edit3.Text := ini.ReadString('info','path','');
ini.Free;
end;
procedure TFSendMsg.Memo1Click(Sender: TObject);
begin
btnsend.Enabled := true;
edit3.ReadOnly := true;
memo1.ReadOnly := false;
end;
procedure TFSendMsg.SendFile(IFileName: string);
var
iFileHandle:integer;
iFileLen,cnt:integer;
buf:array[0..4096] of byte;
mdata: TBytes;
cc: array[0..3000] of char;
bb: array[0..3000] of char;
aa: array[0..3000] of char;
sText,sSendText: string;
i,iLength,isum: integer;
begin
iFileHandle:=FileOpen(IFileName,fmOpenRead);
iFileLen:=FileSeek(iFileHandle,0,2); //返回文件共多少个字符
FileSeek(iFileHandle,0,0); //指针指向文件开头
FillChar(cc, sizeof(cc),0);
cnt := FileRead(iFileHandle,cc,iFileLen);
sSendText := intToStr(100000000 + iFileLen);
Delete(sSendText, 1, 1);
FillChar(aa, sizeof(aa),0);
for i := 1 to 8 do
aa[i-1] := sSendText[i];
mdata := rawtobytes(aa, 8);
client.Socket.Write(mdata);
isum := 0;
while iFileLen > 1000 do
begin
FillChar(bb, sizeof(bb),0);
for i := 0 to 1000 - 1 do
bb[i] := cc[i+isum];
mdata := rawtobytes(bb, 1000);
client.Socket.Write(mdata);
isum := isum + 1000;
iFileLen := iFileLen - 1000;
end;
if iFileLen > 0 then
begin
FillChar(bb, sizeof(bb),0);
for i := 0 to iFileLen - 1 do
bb[i] := cc[i+isum];
mdata := rawtobytes(bb, iFileLen);
client.Socket.Write(mdata);
end;
FileClose(iFileHandle);
end;
procedure TFSendMsg.sendSplitText(client: TIdTCPClient; s: String);
var
strStream:TStringStream;
sText, s1: String;
mdata: TBytes;
cc: array[0..3000] of char;
i: Integer;
begin
sText := s;
FillChar(cc, sizeof(cc),0);
for i := 1 to 18 do
cc[i-1] := sText[i];
mdata := rawtobytes(cc, 18);
client.Socket.Write(mdata);
Delete(sText,1,18);
while Length(sText) > 1000 do
begin
s1 := Copy(sText,1,1000);
FillChar(cc, sizeof(cc),0);
for i := 1 to Length(s1) do
cc[i-1] := s1[i];
mdata := rawtobytes(cc, 1000);
client.Socket.Write(mdata);
Delete(sText,1,1000);
end;
if Length(sText) > 0 then
begin
FillChar(cc, sizeof(cc),0);
for i := 1 to Length(sText) do
cc[i-1] := sText[i];
mdata := rawtobytes(cc, Length(sText));
client.Socket.Write(mdata);
end;
end;
C++和pascal之间的通信的更多相关文章
- android中四大组件之间相互通信
好久没有写有关android有关的博客了,今天主要来谈一谈android中四大组件.首先,接触android的人,都应该知道android中有四大组件,activity,service,broadca ...
- angularJS中directive与directive 之间的通信
上一篇讲了directive与controller之间的通信:但是我们directive与directive之间的通信呢? 当我们两个directive嵌套使用的时候怎么保证子directive不会被 ...
- 使用reflux进行react组件之间的通信
前言 组件之间为什么要通信?因为有依赖. 那么,作为React组件,怎么通信? React官网说, 进行 父-子 通信,可以直接pass props. 进行 子-父 通信,往父组件传给子组件的函数注入 ...
- Fragment之间的通信(四)
自定义两个fragment的布局和java类. 在mainactivity中引用布局文件 在其中的一个fragment中的控件上添加监听,获取到另一个fragment中控件的内容,展示出来完成frag ...
- react native 之子组件和父组件之间的通信
react native开发中,为了封装性经常需要自定义组件,这样就会出现父组件和子组件,那么怎么在父组件和子组件之间相互通信呢,也就是怎么在各自界面push和pop.传值. 父组件传递给子组件: 父 ...
- js组件之间的通信
应用场景: 1.在刷微博的时候,滑到某个头像上,会出现一张usercard(用户名片), 微博列表区有这个usercard, 推荐列表也有这个usercard,评论区也有. 2.在网上购物时,购物车安 ...
- react8 组件之间的通信
<body><!-- React 真实 DOM 将会插入到这里 --><div id="example"></div> <!- ...
- Android使用Fragment来实现ViewPager的功能(解决切换Fragment状态不保存)以及各个Fragment之间的通信
以下内容为原创,转载请注明:http://www.cnblogs.com/tiantianbyconan/p/3364728.html 我前两天写过一篇博客<Android使用Fragment来 ...
- Android使用Fragment来实现TabHost的功能(解决切换Fragment状态不保存)以及各个Fragment之间的通信
以下内容为原创,转载请注明:http://www.cnblogs.com/tiantianbyconan/p/3360938.html 如新浪微博下面的标签切换功能,我以前也写过一篇博文(http:/ ...
随机推荐
- grep恢复误删除文件内容(转)
在 Linux 上如果事先没有用别名(alias)修改默认的 rm 功能,rm 后文件就会丢失,幸运的是,在一般的删除文件操作中,Linux 并不会立即清空存储该文件的 block 内容,而只会释放该 ...
- Excel多条件筛选、公式填充
接到一个任务,由于数据操作人员不会使用编辑公式进而无法进行相关筛选,所以要我帮忙.好久不碰Excel了,那就试试看吧. 需求是这样子的(这里做了最大化的简化):要求判断条件,男50岁以上,女40岁以上 ...
- rhel_7.x 安装mysql
http://database.51cto.com/art/201310/413006.htm MariaDB和MySQL --mysql-5.7.12-1.el7.x86_64.rpm-bundle ...
- 在Java中执行js代码
在某些特定场景下,我们需要用Java来执行Js代码(如模拟登录时,密码被JS加密了的情况),操作如下: ScriptEngineManager mgr = new ScriptEngineManage ...
- Eclipse 下 opennms 开发环境搭建
1.eclipse3.5或更高版本,并且使用纯净的java版.下载地址:Eclipse for Java Developers. 2.安装需要的插件.通过Help/Install New Softwa ...
- Linux磁盘空间爆满,MySQL无法启动
OS: Cent OS 6.3 DB: 5.5.14 看到一个帖子,在服务器上安装了oracle和mysql数据库,mysql数据库忘记开启innodb_file_per_table,导致插入测试数据 ...
- 阿里云mysql数据库恢复总结,mysql binlog日志解析
由于意外..阿里云mysql中有一张表被全部删除了,深吸三口气候,开始解决. 首先用凌晨的自动备份的,进行全量恢复,然后找binlog日志(见下文),查找从全量备份到数据删除之间的记录 这导致了一个问 ...
- C#的winform小合集
C#的winform小合集 博主很懒,又想记录一下自己的所做所为,仅此而已,供自己日后所看.这个是博主自主学习C#所写的一些小程序,有好玩的,也有一些无聊闲得蛋疼所作的. 内容介绍 C#入门窗口输出h ...
- Linux进程操作信息
Linux进程操作简单小结 linux上进程有5种状态: 1. 运行(正在运行或在运行队列中等待) 2. 中断(休眠中, 受阻, 在等待某个条件的形成或接受到信号) 3. 不可中断(收到信号不唤醒和不 ...
- nginx——内存池篇
nginx--内存池篇 一.内存池概述 内存池是在真正使用内存之前,预先申请分配一定数量的.大小相等(一般情况下)的内存块留作备用.当有新的内存需求时,就从内存池中分出一部分内存块,若内存块不够再继续 ...