判断SOCKET通信中客户端在s内未发送数据或者未收到服务器发送的数据自动断开.docx
- 文档编号:6513109
- 上传时间:2023-01-07
- 格式:DOCX
- 页数:13
- 大小:18.53KB
判断SOCKET通信中客户端在s内未发送数据或者未收到服务器发送的数据自动断开.docx
《判断SOCKET通信中客户端在s内未发送数据或者未收到服务器发送的数据自动断开.docx》由会员分享,可在线阅读,更多相关《判断SOCKET通信中客户端在s内未发送数据或者未收到服务器发送的数据自动断开.docx(13页珍藏版)》请在冰豆网上搜索。
判断SOCKET通信中客户端在s内未发送数据或者未收到服务器发送的数据自动断开
如何判断SOCKET通信中,客户端在10s内未发送数据或者未收到服务器发送的数据,自动断开
————————————————————————————————作者:
————————————————————————————————日期:
请教如何判断SOCKET长连接通信中,一段时间内是否收到对方的信息.及相关问题.
对方的接口文档中要求:
1.2业务接口
服务端地址:
211.151.234.131端口号:
9007
Tcp协议适合发送量大,发送和接收比较及时,对于Tcp客户端由于某些原因一段时间内不能连接到分发中心服务器,所有的发向该客户端的MO短信和报告都会被缓存起来,等该客户端一旦连接,便会很快补发。
建议合作方建立一个发送一个接收2个连接,如果业务量很大,可以申请多于2个的连接,但须向业务申请后方可加连接,系统对于连接数是有限制的,每个合作方没有特殊声明时连接数是3,一般可以建2个连接,一个连接用于缓冲在某些情况下断开不能很好识别的情况。
还有,对于每条连接,我们目前的速度限制是10条/秒,如果你的业务超过每连接每秒10条的限制,可以向业务申请调高每条连接的最高限速。
服务器端要求每连接每分钟都要能从客户端接收到至少一条指令数据,如果超过一分钟没有收到,服务器会向客户端发送一个测试指令,只要合作方回应该指令,那么就认为连接是处于激活状态,如果3分钟内服务端都无法接收到客户端的测试回应,将主动断开连接。
客户端程序也应该设置3分钟内不能收到服务器端任何指令将主动断开连接并重新连接。
为了防止分发中心连接负载太重,我们要求任何一个连接因任何原因在断开后20秒内不得连接,20秒后再尝试连接,否则如果在1分钟内超过我们设定的连接次数,我们将限制其在后续10分钟之内不能连接,并向系统管理员告警,由系统管理员根据情况停止该用户的合作帐户。
1.2.1连接登陆指令
连接登陆指令是在客户端成功连接后首先应当而且只能在此时发送的指令:
格式:
LoginName=【注册名】&Pwd=【注册密码】&Type=【注册类型,0:
接收和发送;1:
接收;2:
发送;默认为0】(回车换行)
如果所有服务注册成功,服务器返回给客户端字符串:
Pass(回车换行)
否则将断开连接。
对于一次未连接成功,应至少在20秒以后再重试连接,禁止连续的重试连接。
请问红色的话,我应该如何设置三分钟内不收到服务器任何指令将主动断开连接并重新连接.
我写的主要的客户端的代码:
父类线程:
publicclassParentThreadextendsThread{
privateSocketsocket;
protectedBufferedReaderreader;
protectedBufferedWriterwriter;
protectedbooleanisConnection=false;
protectedbooleanlogin(inttype){
LoginRequestlogin=newLoginRequest(type);
//login.setRegType(type);
login.create();
try{
writer.write(login.getOrder());
writer.flush();
Stringstr_order=reader.readLine();
System.out.println(str_order);
//PassResponsepass=newPassResponse(str_order);
//if(pass.isPass()){
if("Pass".equals(str_order)){
returntrue;
}else{
returnfalse;
}
}catch(IOExceptione){
e.printStackTrace();
}
returnfalse;
}
protectedvoidinit(){
try{
socket=newSocket(Const.distributeCenterIP,Const.distributeCenterPort);
//socket.setSoTimeout(50000);
reader=newBufferedReader(newInputStreamReader(socket.getInputStream()));
writer=newBufferedWriter(newOutputStreamWriter(socket.getOutputStream()));
isConnection=true;
}catch(UnknownHostExceptione){
//TODOAuto-generatedcatchblock
e.printStackTrace();
}catch(IOExceptione){
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
}
protectedvoidcloseSocket(){
isConnection=false;
try{
if(reader!
=null)reader.close();
}catch(IOExceptione){
e.printStackTrace();
}
try{
if(writer!
=null)writer.close();
}catch(IOExceptione){
e.printStackTrace();
}
try{
if(socket!
=null)socket.close();
}catch(IOExceptione){
e.printStackTrace();
}
}
}
接收短信连接,继承父类线程
publicclassSmsReceiverextendsParentThread{
privatebooleantoReconnection=false;
@Override
publicvoidrun(){
//TODOAuto-generatedmethodstub
try{
init();
//登陆
booleanb=login(Const.reg_type_receive);
if(!
b){//登陆失败,退出
SmsMain.log.error("登陆失败,程序退出!
请检查注册名和口令是否正确";
SmsMain.log.info("注册名名:
"+Const.reg_name);
SmsMain.log.info("用户口令:
"+Const.reg_pwd);
SmsMain.log.info("注册类型:
"+Const.reg_type_receive);
}
}catch(Exceptione){
toReconnection=true;
}
while(true){
if(toReconnection){
/**20秒连接一次*/
try{
sleep(20000);
init();
//登陆
booleanb=login(Const.reg_type_receive);
toReconnection=false;
if(!
b){//登陆失败,退出
SmsMain.log.error("登陆失败,程序退出!
请检查注册名和口令是否正确";
SmsMain.log.info("注册名名:
"+Const.reg_name);
SmsMain.log.info("用户口令:
"+Const.reg_pwd);
SmsMain.log.info("注册类型:
"+Const.reg_type_receive);
break;
}
}catch(InterruptedExceptione){
e.printStackTrace();
}
}
if(!
isConnection){//连接失败
toReconnection=true;
continue;
}
try{
Stringstr_order=reader.readLine();
System.out.println(str_order);
Orderorder=newOrder(str_order);
//如果为测试指令,发送回应
//if(order.isTestOrder()){
if("test".equals(str_order)){
System.out.println("---测试----";
writer.write(order.testOrder);
writer.flush();
toReconnection=false;
continue;
}
//处理
SMSBusinessCentersbc=newSMSBusinessCenter(order);
Propertiesprop=sbc.performTask();
//发送回应
System.out.println(prop.getProperty("CommandId");
received(prop.getProperty("CommandId");
}catch(IOExceptione){
e.printStackTrace();
}
}
}
privatevoidreceived(Stringcid){
//Receivedreceived=newReceived(Utils.getCurrentCommandId());
//Receivedreceived=newReceived(cid);
//received.create();
try{
writer.write("ReceivedCommandId="+cid+"\r\n";
writer.flush();
}catch(IOExceptione){
e.printStackTrace();
}
}
}
发送短信连接,继承父类线程:
publicclassSmsSendextendsParentThread{
privatebooleantoReconnection=false;
@Override
publicvoidrun(){
try{
init();
//登陆
booleanb=login(Const.reg_type_send);
if(!
b){//登陆失败,退出
SmsMain.log.error("登陆失败,程序退出!
请检查注册名和口令是否正确";
SmsMain.log.info("注册名名:
"+Const.reg_name);
SmsMain.log.info("用户口令:
"+Const.reg_pwd);
SmsMain.log.info("注册类型:
"+Const.reg_type_send);
}
}catch(Exceptione){
toReconnection=true;
}
while(true){
if(toReconnection){
/**20秒连接一次*/
try{
sleep(20000);
init();
//登陆
booleanb=login(Const.reg_type_send);
toReconnection=false;
if(!
b){//登陆失败,退出
SmsMain.log.error("登陆失败,程序退出!
请检查注册名和口令是否正确";
SmsMain.log.info("注册名名:
"+Const.reg_name);
SmsMain.log.info("用户口令:
"+Const.reg_pwd);
SmsMain.log.info("注册类型:
"+Const.reg_type_send);
break;
}
}catch(InterruptedExceptione){
e.printStackTrace();
}
}
if(!
isConnection){//连接失败
toReconnection=true;
continue;
}
try{
Stringstr_order=reader.readLine();
Orderorder=newOrder(str_order);
//如果为测试指令,发送回应
if(order.isTestOrder()){
writer.write(order.testOrder);
toReconnection=false;
//continue;
}
////处理
//SMSBusinessCentersbc=newSMSBusinessCenter(order);
//Vectormsgs=sbc.getSendMsg();
////发送短信
////for(){
//writer.write("";
//
//
//发送短信
if(SmsMain.list!
=null&&SmsMain.list.size()>0){
synchronized(SmsMain.list){
for(Iteratoriterator=SmsMain.list.iterator();iterator
.hasNext(){
SubmitRequestsr=(SubmitRequest)iterator.next();
StringBuffersb=newStringBuffer();
sb.append(Const.order_submit+"")
.append("CommandId="+sr.getCommandId())
.append("&GateWay="+sr.getGateWay())
.append("&GateName="+sr.getGateName())
.append("&ItemId="+sr.getItemId())
.append("&SpNumber="+sr.getSpNumber())
.append("&UserNumber:
="+bin2hex(sr.getUserNumber()))
.append("&UserNumberType="+sr.getUserNumberType())
.append("&FeeNumber="+sr.getFeeNumber())
.append("&FeeNumberType="+sr.getFeeNumberType())
.append("&FeeType="+sr.getFeeType())
.append("&ScheduleTime="+sr.getScheduleTime())
.append("&ExpireTime="+sr.getExpireTime())
.append("&MtFlag="+sr.getMtFlag())
.append("&ReportFlag="+sr.getReportFlag())
.append("&MsgCode="+sr.getMsgCode())
.append("&MsgId="+sr.getMsgId())
.append("&ExtData:
="+sr.getExtData())
.append("&TP_pId="+sr.getTp_pId())
.append("&TP_udhi="+sr.getTp_udhi())
.append("&Msg:
="+bin2hex(sr.getMsg()))
.append("&LinkID="+sr.getLinkID())
.append("&ItemType="+sr.getItemType()+"\r\n");
writer.write(sb.toString());
writer.flush();
SmsMain.list.remove(sr);
}
}
//接受回应
received();
}
}catch(IOExceptione){
e.printStackTrace();
}
}
}
privatevoidreceived(){
try{
Stringline=reader.readLine();
//Receivedreceived=newReceived(line);
//received.parse();
System.out.println(line);
SmsMain.log.info(line);
}catch(IOExceptione){
e.printStackTrace();
}
}
/**
*字符串转换成十六进制值
*@parambinString转换成十六进制的字符串
*@return
*/
publicstaticStringbin2hex(Stringbin){
char[]digital="0123456789ABCDEF".toCharArray();
StringBuffersb=newStringBuffer("");
byte[]bs=bin.getBytes();
intbit;
for(inti=0;i bit=(bs&0x0f0)>>4; sb.append(digital[bit]); bit=bs&0x0f; sb.append(digital[bit]); } returnsb.toString(); } } ---------------------------------------------------------- 语言: C# 问题: 在socket通信时,怎么样判断socket双方是否断开连接 我在Server端new了一个socket,然后bind,开了一个线程来accept前来连接的client,每接到一个client前来连接就新开一个线程和它进行通信。 我把Server端得到的socket放到一个集合里,我想知道集合里的socket是否断开连接,如果断开连接我就把它从这个集合里移除。 判断socket是否断开连接,网上有N种说法: 1.Socket.Connected 这个属性只能说明上一次通信时还是正常的。 2.Socket.Pool 这个方法是可以,但是它要从socket里读或写部分数据,如果其他线程从这个socket读写数据,那么数据将会出错。 我在写一个远程通信软件时候就出现这个问题了。 而且 此方法不能检测某些类型的连接问题,例如,网络电缆中断或远程主机意外关闭。 您必须尝试发送或接收数据以检测这些类型的错误。 3.使用心跳包 每隔一段时间,socket一方向另一方发送一段特殊的数据,socket的另一方接到这个数据后回发回来。 这样就能判断socket两方都没有掉线了。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 判断 SOCKET 通信 客户端 发送 数据 或者 收到 服务器 自动 断开