有关socket的几个问题

作者&投稿:瞿蚂 (若有异议请与网页底部的电邮联系)
关于socket的问题~

你是不是没有判断数据是否接收完毕,就是数据已经全部被你接收完成了,接收缓冲区已经不存在数据了,就会阻塞在lnr.readLine()这个地方
String line;
while ((line = lnr.readLine()) != null) {
responseInfo.append(line + "
");
// 要在这里统计接收的数据长度,如果接收完成,即跳出循环,或做别的事情

}

include
#include
int jc(int m){
if(m!=1) return m*jc(m-1);
else return 1;
}
int c(int m,int n){
if(m>=n) return jc(m)/(jc(n)*jc(m-n));
}
int main(void){
int m,n;
scanf("%d%d",&m,&n);
printf("%d
",c(m,n));
return 0;
}

Sockets是在Unix系统上提出来的,一开始主要是用于本地通讯,但很快就应用到C/S体系上。MicroSoft公司在此基础上创建了WinSock控件,专门用于Windows接口,与Sockets完全兼容。Winsock控件对用户来说是不可见的,它提供了访问 TCP 和 UDP 网络服务的方便途径。Microsoft Access、Visual Basic、Visual C++ 或 Visual FoxPro 的开发人员都可使用它。为编写客户或服务器应用程序,不必了解 TCP 的细节或调用低级的 Winsock APIs。通过设置控件的属性并调用其方法就可轻易连接到一台远程机器上去,并且还可双向交换数据。下面就利用VB5中的WinSock控件编写一个网上聊天程序。

一)网络通信协议的基础和选择

1.1 TCP(数据传输协议)基础

数据传输协议允许创建和维护与远程计算机的连接。连接两台计算机就可彼此进行数据传输。
如果创建客户应用程序,就必须知道服务器计算机名或者 IP 地址(RemoteHost 属性),还要知道进行“侦听”的端口(RemotePort 属性),然后调用 Connect 方法。
如果创建服务器应用程序,就应设置一个收听端口(LocalPort 属性)并调用 Listen 方法。当客户计算机需要连接时就会发生 ConnectionRequest 事件。为了完成连接,可调用 ConnectionRequest 事件内的 Accept 方法。
建立连接后,任何一方计算机都可以收发数据。为了发送数据,可调用 SendData 方法。当接收数据时会发生 DataArrival 事件。调用 DataArrival 事件内的 GetData 方法就可获取数据。

1.2 UDP(用户数据文报协议)基础

用户数据文报协议 (UDP) 是一个无连接协议。跟 TCP 的操作不同,计算机并不建立连接。另外 UDP 应用程序可以是客户机,也可以是服务器。
为了传输数据,首先要设置客户计算机的 LocalPort 属性。然后,服务器计算机只需将 RemoteHost 设置为客户计算机的 Internet 地址,并将 RemotePort 属性设置为跟客户计算机的 LocalPort 属性相同的端口,并调用 SendData 方法来着手发送信息。于是,客户计算机使用 DataArrival 事件内的 GetData 方法来获取已发送的信息。

1.3 选择通讯协议

在使用 WinSock 控件时,首先需要考虑使用什么协议。可以使用的协议包括 TCP 和 UDP。两种协议之间的重要区别在于它们的连接状态:
TCP 协议是有连接的协议,可以将它同电话系统相比。在开始数据传输之前,用户必须先建立连接。
UDP 协议是一种无连接协议,两台计算机之间的传输类似于传递邮件:消息从一台计算机发送到另一台计算机,但是两者之间没有明确的连接。另外,单次传输的最大数据量取决于具体的网络。

到底选择哪一种协议通常是由需要创建的应用程序决定的。下面的几个问题将有助于选择适宜的协议:

1. 在收发数据的时候,应用程序是否需要得到客户端或者服务器的确认信息?如果需要,使用 TCP 协议,在收发数据之前先建立明确的连接。

2. 数据量是否特别大(例如图象与声音文件)?在连接建立之后,TCP 协议将维护连接并确保数据的完整性。不过,这种连接需要更多的计算资源,因而是比较“昂贵”的。

3. 数据发送是间歇的,还是在一个会话内?例如,如果应用程序在某个任务完成的时候需要通知某个计算机,UDP 协议是更适宜的。UDP 协议适合发送少量的数据。

通讯协议的选择是通过设置WinSock的Protocol属性来实现的。下面选择TCP通讯协议编写网上聊天程序,在此之前必须知道一个极其重要的参数---服务器端的IP地址或计算机名。

二)确定计算机的名字

1. 在计算机的桌面上,右键单击“网上邻居”。

2. 选择“属性”。

3. 单击“标识”选项卡。

4. 在“计算机名称”框中可以找到计算机的名称。

确定计算机的 IP地址

1. 单击“任务条”上的“启动”。

2. 选择“运行”。

3. 若服务器端操作系统为win95则在“打开”中填入“winipcfg”,若服务器端操作系统为winnt则在“打开”中填入“ipconfig”。

4. 按下“确定”键。

上面找到的计算机名称或IP地址可以作为WinSock的RemoteHost 属性的值。

三) winsock控件的State属性。

state 属性的设置值是: 常 数
值 描 述

sckclosed 0 缺省的。关闭
sckopen 1 打开
scklistening 2 侦听
sckconnectionpending 3 连接挂起
sckresolvinghost 4 识别主机
sckhostresolved 5 已识别主机
sckconnecting 6 正在连接
sckconnected 7 已连接
sckclosing 8 同级人员正在关闭连接
sckerror 9 错误

下面主要要用到sckClosed.sckConnected两个State属性的值。

四)网上聊天程序的编制

4.1 程序中服务器端所起的作用。

从图示中可以看到服务器端的两个winsock控件之间并不存在直接的通讯,同时sckServer1和sckClient2及sckServer2和sckClient1之间是不能直接通讯的。这也即是说若sckClient1向sckClient2发出信息,信息首先被sckServer1接受,sckServer1再将信息传给程序的信息处理部分,信息处理部分再将处理好的信息传给sckServer2,再由sckServer2传给sckClient2。反之亦然。那么服务器端的信息处理部分又进行什么工作呢?

1. 对通讯的通道数作一些限制。

2. 对使用后已关闭的通道,必须能够重新使用以节省资源。

3. 必须对所传递的数据包信息作甑别,从而作出不同的处理。

通过解开数据的包头就可区分不同的信息。

网上聊天有两种方式:第一种,以广播方式;第二种,以点对点的方式。广播方式即所有客户都能收到某一客户发出的信息。点对点的方式即想说“悄悄话”的一对客户专门开辟了一间谈话的“小屋”,别的客户不能“听”到他们的谈话。在下面的程序中将看到如何利用数据的不同包头来区分用户是想以广播方式还是以点对点的方式进行谈话的(点对点方式数据的包头为“PT”,广播方式则无包头)。

4.2 客户端的程序

1. 在客户端创建一个新的工程将其命名为“ClientPrj”

2. 将缺省窗体命名为 frmClient。

3. 将窗体的标题改为“Client”。

4. 在窗体中添加一个 WinSock 控件,并将其命名为 tcpClient。

5. 在 frmClient 中添加一个ListBox 控件。将其命名为lstReceive。

6. 在 frmClient 中添加一个 TextBox 控件。将其命名为 txtSend。

7. 在窗体上放两个 CommandButton 控件,并将其命名为 cmdConnect和cmdSent。

8. 将cmdConnect控件的标题改为 Connect, 将cmdSent控件的标题改为 Sent。

9. 在窗体中添加如下的代码。

Private Sub cmdConnect_Click()

On Error GoTo ErrorPro

sckClient.Connect

Exit Sub

ErrorPro:

MsgBox "服务器未开或网络出错!"

End

End Sub

Private Sub cmdSent_Click()

sckClient.SendData txtSent.Text

End Sub

Private Sub Form_Load()

' RemoteComputerName为服务器端的计算机名或IP地址。

sckClient.RemoteHost = "RemoteComputerName"

sckClient.RemotePort = 1000

End Sub

Private Sub sckClient_Close()

MsgBox "服务器通道已关闭!"

End

End Sub

Private Sub sckClient_Connect()

MsgBox "连接成功!"

cmdConnect.Enabled = False

End Sub

Private Sub sckClient_DataArrival(ByVal bytesTotal As Long)

Dim s As String

sckClient.GetData s

lstReceive.AddItem s

End Sub

Private Sub sckClient_Error(ByVal Number As Integer, Description As String, ByVal Scode As Long, _ ByVal Source As String, ByVal HelpFile As String, ByVal HelpContext As Long, CancelDisplay As Boolean)

sckClient.Close

cmdConnect.Enabled = True

End Sub

4.3 服务器端的程序

1. 在服务器端创建一个新的工程将其命名为“ServerPrj”。

2. 将缺省窗体命名为“frmServer”。

3. 在窗体中添加一个ListBox控件,将其命名为“lstReceive”。

4. 在窗体中添加三个WinSock控件,将其分别命名为“sckListen”,sckBusy和“sckServer”并将“sckServer”的“Index”属性设置为0。

5. 在窗体中添加如下代码.。

'最大通道数

Private MaxChan As Integer

Private Sub Form_Load()

Dim i As Integer

MaxChan = 10

For i = 1 To MaxChan - 1

Load sckServer(i)

Next i

sckListen.LocalPort = 1000

sckListen.Listen

End Sub

Private Sub sckBusy_Close()

sckBusy.Close

End Sub

Private Sub sckBusy_DataArrival(ByVal bytesTotal As Long)

sckBusy.SendData "服务器忙,请稍后再连接!"

DoEvents

End Sub

Private Sub sckListen_ConnectionRequest(ByVal requestID As Long)

Dim i As Integer

'决定由哪一Winsock接受请求

For i = 0 To MaxChan - 1

If sckServer(i).State = 0 Then

Exit For

End If

Next i

If sckServer(i).State = 0 Then

sckServer(i).Accept requestID

Exit Sub

End If

'如果所有Winsock都用完则由专门的“忙”Winsock接受请求,以免用户要求得不到响应

sckBusy.Close

sckBusy.Accept requestID

End Sub

Private Sub sckListen_Error(ByVal Number As Integer, Description As String, ByVal Scode As Long, _ ByVal Source As String, ByVal HelpFile As String, ByVal HelpContext As Long, CancelDisplay As Boolean)

sckListen.Close

sckListen.LocalPort = 1000

sckListen.Listen

End Sub

Private Sub sckServer_Close(Index As Integer)

sckServer(Index).Close

End Sub

Private Sub sckServer_DataArrival(Index As Integer, ByVal bytesTotal As Long)

Dim s As String

Dim i As Integer

sckServer(Index).GetData s

If UCase(Left(Trim(s), 2)) = "PT" Then '判断是否为悄悄话,点对点方式

If IsNumeric(Mid(Trim(s), 3, 1)) Then

i = Mid(Trim(s), 3, 1)

sckServer(i).SendData "Channel " & Index & " " & Right(Trim(s), Len(Trim(s)) - 3)

DoEvents

End If

Else '广播方式

For i = 0 To MaxChan - 1

'利用winsock的State属性给所有连接在服务器上的客户发消息

If sckServer(i).State = 7 Then

sckServer(i).SendData "Channel " & Index & " " & Trim(s)

DoEvents

End If

Next i

End If

lstReceive.AddItem "Channel " & Index & " " & Trim(s)

End Sub

Private Sub sckServer_Error(Index As Integer, ByVal Number As Integer, Description As String, _

ByVal Scode As Long, ByVal Source As String, ByVal HelpFile As String, ByVal HelpContext As _

Long, CancelDisplay As Boolean)

sckServer(Index).Close

End Sub

从程序中可以看到:第一,程序中限制了通道数(10路)。第二,通过判断WinSock控件的State属性是否为0(关闭状态),来重新使用已关闭的WinSock控件。第三,通过给WinSock控件传递的信息加上包头,来对信息进行不同的处理(程序中若信息前加上了“PT"(Private Talk)+"通道数”的包头,由此就知道客户想要同拥有此“通道数”的另一客户进行“悄悄话”,否则就以广播方式将信息发给所有客户)。

五) 结束语

WinSock控件不仅仅是用来编制网上聊天程序,而且可以用来编制各种网络游戏或网络通信程序。实际上WinSock控件是编制各种C/S程序的利器。在实际使用中通常是将WinSock控件封装在Activex DLL(进程内)、Activex EXE(进程外)部件的类中(类中引用)来使用的。通过区分所传信息前的不同的包头,用RaiseEvent命令引发不同 的事件,再对事件分别进行处理。这样不仅增加了程序的可调试性和安全性,而且更符合事件驱动编程方法的特点。

楼上的什么东西??一大堆没用的

1.sin是BufferdReader的一个实例,这是I/O中的一个读取类,这里你看,System.in是输入流,将之包装为一个InputStreamReader之后,再给BufferdReader,包装成一个更高级的读取流,用来读取你的输入的东西.

os是一个写入流,就是向某个地方写东西,这里使用的是高级的PrintWriter,而真正写东西的,是Socket产生的写入流,只不过这里包装了一下,高级的更好操作

is同理,是一个读取流,读Socket服务端写入的东西.

2.因为都是高级IO,所以,is.readLine()变为可行,就是说,如果不象我上面说的"包装"一下,那么读取和写入就只能以字节的方式来操作,而包装之后,你可以"一行一行"的读写,所以这里是"readLine()"--读一行

3.不能运行的原因是,这仅仅是一个Socket客户端,(可能)你并没有启动你的服务器端的程序,Socket是通信的,你和谁通信呢?

这些问题出现的原因是,首先你对JAVA的I/O操作不熟悉,建议你先去看看,Socket基本的东西你也不熟悉,了解一下C/S结构.


socket中shutdown和closesocket的区别
一、从函数调用上来分析(msdn):一旦完成了套接字的连接,应当将套接字关闭,并且释放其套接字句柄所占用的所有资源。真正释放一个已经打开的套接字句柄的资源直接调用closesocket即可,但要明白closesocket的调用可能会带来负面影响,具体的影响和如何调用有关,最明显的影响是数据丢失,因此一般都要在...

深入浅出TCP之半关闭与CLOSE_WAIT
造成这种情况的原因之一是服务器在业务处理未结束时未主动发送FIN。当客户端发起关闭时,子进程可能仅减小引用计数,而非真正关闭socket,形成CLOSE_WAIT。正确的处理方式是,子进程应在业务处理结束后,先调用shutdown关闭写端,再调用close关闭整个连接,确保发送FIN并进入LAST_ACK状态,最终返回到CLOSED。sh...

closesocket()和close()区别
一个是window下的函数(closesocket),一个是linux下的函数(close)

socket已经close,为什么用netstat命令查看还是 ESTABLISHED
socket关闭——应该说是TCP关闭,需要进行4次握手,一方close后只表示自己不在发送数据(也不能发送了),这个通道要等到对方返回一个fin-ack才会真正关闭。TCP还是UDP??? ==>UDP没有连接的说法 一方断了,别一方也会断的啊... ==>完全错误的说法,这需要程序自己来控制的 查断点看看Socket对象...

怎样判断socket断开并自动重连
对于主动关闭的SOCKET, recv返回-1,而且errno被置为9(#define EBADF 9 \/* Bad file number *\/)或104 (#define ECONNRESET 104 \/* Connection reset by peer *\/)对于被动关闭的SOCKET,recv返回0,而且errno被置为11(#define EWOULDBLOCK EAGAIN \/* Operation would block *\/)对正常的...

tcp通道关闭时,发生了什么? time_wait close_wait
关闭A,则A向B发送FIN;如果程序显式的关闭了B,那么B会向A发送一个FIN,然后B就处于LAST_ACK状态了;A在接受到B的FIN后,发出最后一个ACK,此时A就处于知名的TIME_WAIT状态了。TIME_WAIT时间一般会比较长。尽量避免TIME_WAIT过多的一端主动关闭socket 使用SocketPool,避免频繁创建\/关闭socket 提到...

socket的输入输出流需要关闭吗
正常关闭Socket,特别是TCP Socket的步骤是:先Shutdown关闭流,然后再Close释放资源。

socket关闭后为什么recv不返回
recv是socket编程中最常用的函数之一,在阻塞状态的recv有时候会返回不同的值,而对于错误值也有相应的错误码,分别对应不同的状态,下面是我针对常见的几种网络状态的简单总结。首先阻塞接收的recv有时候会返回0,这仅在对端已经关闭TCP连接时才会发生。而当拔掉设备网线的时候,recv并不会发生变化,仍然...

socket接口详解
Socket接口是应用程序与网络通信的重要桥梁,它抽象了TCP\/IP层的复杂操作,让开发者能以简单接口实现进程间的通信。接口概述:Socket,源于UNIX,遵循"打开-读写-关闭"模式。在服务器端,通过bind绑定本地地址和端口,listen开启监听,accept等待客户端连接,connect用于客户端主动连接,send和recv负责数据的...

如何优雅地关闭一个Socket
轻轻地插进去不要太用力 慢慢的取出来不要太心急(绝对不污)污者不留名,留名不污者

新兴区15024354632: 什么是Socket? -
乌冉澳扶: Socket接口是TCP/IP网络的API(Application Programming Interface,应用程序编程接口),Socket接口定义了许多函数或例程,程序员可以用它们来开发 TCP/IP网络上的应用程序.请参阅以下资料:socket非常类似于电话插座.以一个国家...

新兴区15024354632: 关于socket编程问题,100分急求 -
乌冉澳扶: 1、首先若大量包裹是一起发的,完全不用操心这个问题,客户端发完包时会close链接,此时会发一个FIN信息表示已经发送完毕,这时服务器会自动收到FIN信息知道客户端完成发包完成了.而这些操作都是在socket api级别完成的,不用额外...

新兴区15024354632: 问个写socket程序的问题.
乌冉澳扶: 在client端关闭时服务端会报异常,捕获异常处理就好,当然这个异常不是一关闭就报,有个延时,还有一种常用的方法就是client发心跳包,无心跳包的连接由服务器关闭,socket的问题比较麻烦,最好能具体说一下你的方式如:是tcp还是udp,是同步还是异步等,这些的处理方式都不太一样

新兴区15024354632: socket通信的问题 -
乌冉澳扶: 当然你的主线程监听在某个端口时,如果此时有某个手机客户端连上来了,服务线程就会new scoket一个服务线程,去专门一对一的处理这个客户端的请求,直到这个客户端的连接断掉,不然这个服务线程一直...

新兴区15024354632: socket问题 -
乌冉澳扶: 果然!!!!发送也有缓冲区啊,缓冲区满了就没办法发了.你用异步socket的话,要等FD_WRITE消息过来以后才能接着发.

新兴区15024354632: 请教关于socket的close和shutdown问题 -
乌冉澳扶: 书中解释如下: socket.setSoLinger(true,3600); 执行Socket的close()方法,该方法不会立即返回,而进入阻塞状态.同时,底层的Socket会尝试发送剩余的数据.只有满足以下两个条件之一,close()方法才返回: 1、底层的Socket已经发送完所有的剩

新兴区15024354632: 常见Socket连接错误及原因有哪些? -
乌冉澳扶: socketerrorcode:11001,socketerrorinfo:不知道这样的主机. 表示您的计算机无法连上服务器,请检查您的Proxy设定以及Proxy相关账号,或暂时取消您防毒软件的「个人防火墙」. 2、...

新兴区15024354632: 关于Socket通信的一个问题,求大虾帮助啊~急 -
乌冉澳扶: socket好象是点-点通信吧,采用遍历逐一发送命令.效率上讲是差不多的,就算程序中用直发比较简单(是否可以用?我也没找到办法!),但网卡发送时仍然是逐一发送;除非使用群发方式,但这不是socket能解决的吧.方案一:可以在程序中将网卡设置为“不延时发送”模式,对频发小数据包(低于网呀缓冲区大小)的效率提高数十倍.方案二:服务器端用建立多线程,每一个线程负责与客户端进行通信,效率也会提高.偶也常用Socket通信,以上建议供参考.

新兴区15024354632: 关于vb.net中Socket的问题 -
乌冉澳扶: 我不给你代码了,我才编了SOCKET的,告诉你原理吧(我的编程经验而已)SOCKET传输文件和接受是有大小限制的,默认是81**来则也就是8K多,你可以改小,但改大我没试过,然后你必须把文件分成...

新兴区15024354632: Socket发送问题 -
乌冉澳扶: 把类型名字放在最前面并存入流中(用BinaryWriter的write方法写到流里)再将此类型变量按序...

本站内容来自于网友发表,不代表本站立场,仅表示其个人看法,不对其真实性、正确性、有效性作任何的担保
相关事宜请发邮件给我们
© 星空见康网