当前位置: 首页 > 编程笔记 >

golang判断net.Conn 是否已关闭的操作

东郭阳德
2023-03-14
本文向大家介绍golang判断net.Conn 是否已关闭的操作,包括了golang判断net.Conn 是否已关闭的操作的使用技巧和注意事项,需要的朋友参考一下

在多线程的线程里边,用一个线程处理一条连接,如何判断连接已经关闭?

试了一下,如果连接中断,读写会出现这种net.OpError,这个就可以判断是否断开

但是我也不确定有没有更好的方法?如果有的话,希望朋友们不吝赐教

func dealConn(conn net.Conn){
	//defer conn.Close()
	//defer conn.Flush()
 //长连接里边的读写操作必须放到循环里面这样才能进行多次的读写
 // 如果连接已经断开,就把这个线程中断掉,怎么判断这个连接已经断开? 
 thread_c:=0;//如果连续100秒中读取不到内容,就终止循环
 for{
  defer func() {
   if r := recover(); r != nil {
    buf:=make([]byte,666)
    buf=buf[:runtime.Stack(buf,false)]
    log.Printf("运行时错误:%v.Runtime error caught: %s",r, buf)
   }
  }()
  // 注意continue这里也要等待,不然造成内存耗尽,处理器耗尽
  time.Sleep(50*time.Millisecond)
  //#log.Println(len,string(text))
  thread_c++
  if thread_c>20*100{
  	log.Println(conn.RemoteAddr(),"超过100秒未读取到内容,本连接将关闭")
  	conn.Close();
  	c--;
  	break;
  }
  frame,op_err:=readAllShut(conn)
  if op_err!=nil{
  	log.Println(conn.RemoteAddr(),"出现读写错误,连接不可用,将会被关闭")
  	conn.Close();
  	c--;
  	break;//这种已经关闭的连接,要终止循环,退出这条线程
  }
  if(len(frame)==0){
  	//
  	//time.Sleep(50*time.Millisecond)
   continue
  }
  thread_c=0;
  log.Printf("-----------------收到tcp请求:报文的长度是%v,详细内容如下:%s,转换成16进制是:%x", len(frame),frame,frame)
  //TODO
  //这里写自己的业务代码
  }
 }
func readAllShut(conn net.Conn) ([]byte,error){	//这个手动方法可以避免粘包的问题
	//bufio.NewWriter
	re:=bytes.NewBuffer(nil)
	const N=666
	for{
		var text [N]byte
		lens,err:=conn.Read(text[0:])
		re.Write(text[:lens])
		if lens==0 || err!=nil{
			//log.Println(err) //在这个死循环里面,不要有任何的输出
			// if errors.As(err,*net.OpError) 	//
			if _,ok:=err.( *net.OpError) ;ok{		
				return nil,err
			}
			break
		}
		//conn
		//log.Println(lens,text)
		if lens<N{
			break
		}
	}
	rb:= re.Bytes()
	//log.Println(rb,"len",len(rb))
	return rb,nil
	/*data,err:=ioutil.ReadAll(conn)
	if err!=nil{
		log.Printf("读取出现错误%T:%v",err,err)
	}
	return data;*/
} 

补充:Go -- 判断chan channel是否关闭的方法

如果不判断chan是否关闭

Notice: 以下代码会产生死循环

代码如下:

package main
import (
 "fmt"
)
func main() {
 c := make(chan int, 10)
 c <- 1
 c <- 2
 c <- 3
 close(c)
 for {
  fmt.Println(<-c)
 }
}

判断短chan是否关闭

代码如下:

package main
import (
 "fmt"
)
func main() {
 c := make(chan int, 10)
 c <- 1
 c <- 2
 c <- 3
 close(c)
 for {
  i, isClose := <-c
  if !isClose {
   fmt.Println("channel closed!")
   break
  }
  fmt.Println(i)
 }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持小牛知识库。如有错误或未考虑完全的地方,望不吝赐教。

 类似资料:
  • 问题内容: 我遇到了Java套接字API的一些问题。我正在尝试显示当前连接到我的游戏的玩家数量。很容易确定播放器何时连接。但是,似乎似乎很难确定玩家何时使用套接字API断开连接。 呼叫已断开远程似乎总是返回一个插座上。同样,调用已远程关闭的套接字似乎总是会返回。我读过要真正确定套接字是否已关闭,必须将数据写入输出流,并且必须捕获异常。这似乎是处理这种情况的一种非常不干净的方法。我们只需要不断地通过

  • 本文向大家介绍js判断子窗体是否关闭的方法,包括了js判断子窗体是否关闭的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了js判断子窗体是否关闭的方法。分享给大家供大家参考。具体如下: 希望本文所述对大家的javascript程序设计有所帮助。

  • 我想测试一个返回流的方法。 现在,我已经为当somecondition=true时的情况编写了以下测试

  • 本文向大家介绍javascript判断网页是关闭还是刷新,包括了javascript判断网页是关闭还是刷新的使用技巧和注意事项,需要的朋友参考一下 原理就是通过离开页面行为时间onunload触发时间去检测此时的浏览器的窗口大小,根据大小由此判断用户是刷新,跳转或是关闭行为程序  代码如下  用浏览器右上角的关闭按钮时好用,但在选项卡上关闭和在任务栏上关闭,这个方法就不作用了 js标签只有onlo

  • 问题内容: 我下载了封装在zip文件中的proguard,并将其解压缩到硬盘上。我没有以任何方式 安装 它(因为我不知道如何安装)。然后,我将其添加到我的project.properties文件中。然后,我做了一个“导出android应用程序”,完全希望eclipse抱怨它不知道我的proguard安装在哪里,但是没有抱怨。确实,我的密钥库中出现了一个新的apk文件,而我的应用程序的proguar

  • sismember key member 存在返回1,0表示不存在或者key不存在