>백엔드 개발 >Golang >golang tcp 닫기 오류 보고서

golang tcp 닫기 오류 보고서

PHPz
PHPz원래의
2023-05-13 11:24:37704검색

최근 Golang을 이용하여 TCP 통신 프로그램을 작성하던 중 이상한 Close 오류가 발생하여 녹화하여 공유하였습니다.

Background

제가 작성한 TCP 통신 프로그램에서는 Go 표준 라이브러리의 net 패키지를 사용하여 DialTCP 기능을 통해 서버에 연결했습니다. 통신이 완료된 후 Close 메서드를 사용하여 연결을 닫습니다. 대부분의 경우 이 프로세스는 성공적입니다. 그러나 때때로 Close 메서드를 호출할 때 이상한 오류가 발생하고 프로그램이 충돌하는 경우가 있습니다.

오류 설명

-Close tcp 호출 오류-
설명: 피어에 의한 재설정`
오류 로그에 따르면 Close 메서드를 사용하여 연결을 닫을 때 "피어에 의한 재설정" 오류가 보고됩니다. 서버가 적극적으로 연결을 닫습니다.

Troubleshooting

첫 번째 단계에서는 Close 메서드를 호출하기 전에 소스 코드를 확인하여 연결이 닫히지 않았는지 확인했습니다. 두 번째 단계에서는 코드 관점에서 문제를 분석한 후, 코드의 결함을 제거하려고 노력합니다.

3단계에서 이 문제를 검색해본 결과 비슷한 사례를 발견했습니다. 오픈 소스 커뮤니티 Github에서 많은 사람들이 비슷한 문제에 직면했습니다. 어떤 사람들은 이것이 운영 체제의 TCP 사양 구현에 문제가 있다고 생각하고, 어떤 사람들은 네트워크 카드 드라이버의 오류라고 생각합니다. 이 정보를 보고 TCP 프로토콜의 사양을 확인한 결과 실제로 연결 종료 작업을 기다리는 데 TIME_WAIT 시간이 있다는 것을 확인했습니다.

구체적으로, TCP 연결이 종료된 후 운영체제는 관련 리소스를 해제하기 전에 연결이 완전히 종료되었음을 확인할 때까지 일정 시간 동안 대기합니다. 이 시간은 일반적으로 2MSL(Maximum Segment Lifetime)이며, Linux에서는 기본값이 60초입니다. 이 대기 시간 동안 동일한 주소와 포트를 가진 새로운 연결 요청이 있으면 RST 세그먼트가 트리거됩니다. 즉, 상대방이 연결을 일찍 종료하고 "reset by Peer" 오류가 발생합니다.

그렇다면 이 문제를 어떻게 해결할 수 있을까요?

해결 방법

이 문제에는 두 가지 해결 방법이 있습니다.

  1. TIME_WAIT 시간을 연장하세요.

Linux 커널의 매개변수를 수정하여 TIME_WAIT 시간을 연장하면 연결이 완전히 닫힐 수 있습니다. 물론, 이 연장된 기간 동안 시스템에서 오랫동안 TIME_WAIT 상태에 있었던 연결 수도 증가합니다.

  1. 연결 주소를 재사용 가능하게 하려면 SO_REUSEADDR 옵션을 설정하세요.

연결을 닫을 때 SO_REUSEADDR 옵션을 활성화하면 연결 주소를 재사용할 수 있습니다. 이러한 방식으로 연결이 닫힌 후 다음 새 연결은 "피어에 의한 재설정" 오류를 방지하면서 원래 주소를 직접 재사용할 수 있습니다. 구체적인 구현 방법은 다음과 같습니다.

conn, err := net.DialTCP("tcp", nil, tcpAddr)
err = conn.SetReuseAddr(true)
err = conn.Close()

Summary

위는 Golang에서 TCP 통신 프로그램을 작성할 때 겪었던 이상한 Close 오류에 대한 사례입니다. 그 이유는 TCP 사양에서 요구하는 TIME_WAIT 시간으로 인해 연결이 즉시 해제되지 않기 때문입니다. 대기 시간을 연장하거나 SO_REUSEADDR 옵션을 활성화하면 이러한 유형의 오류를 피할 수 있습니다. 동시에 이는 네트워크 프로그래밍을 할 때 불필요한 오류를 피하기 위해 TCP 사양의 일부 세부 사항에 주의를 기울여야 한다는 점을 상기시켜 줍니다.

위 내용은 golang tcp 닫기 오류 보고서의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.