当前位置: 首页 > 面试题库 >

为什么Go HTTPS Client不重用连接?

卢朝
2023-03-14
问题内容

我有一个http客户端,可以创建与主机的多个连接。我想设置它可以设置到特定主机的最大连接数。go的request.Transport中没有此类选项。我的代码看起来像

package main

import (
  "fmt"
  "net/http"
  "net/url"
)

const (
  endpoint_url_fmt      = "https://blah.com/api1?%s"
)


func main() {

  transport := http.Transport{ DisableKeepAlives : false }

  outParams := url.Values{}
  outParams.Set("method", "write")
  outParams.Set("message", "BLAH")

  for {
    // Encode as part of URI.
    outboundRequest, err := http.NewRequest(
      "GET",
      fmt.Sprintf(endpoint_url_fmt, outParams.Encode()),
      nil
    )
    outboundRequest.Close = false
    _ , err = transport.RoundTrip(outboundRequest)
    if err != nil {
      fmt.Println(err)
    }
  }

}

我希望这会创建1个连接。正如我在for循环中所说的那样。但这一直在创建无限数量的连接。

使用请求库的类似python代码仅创建一个连接。

#!/usr/bin/env python
import requests
endpoint_url_fmt      = "https://something.com/restserver.php"
params = {}
params['method'] = 'write'
params['category'] = category_errors_scuba
params['message'] = "blah"
while True:
  r = requests.get(endpoint_url_fmt, params = params)

由于某种原因,go代码没有重用http连接。

编辑:go代码需要关闭主体才能重用连接。

 resp , err = transport.RoundTrip(outboundRequest)
 resp.Close() //  This allows the connection to be reused

问题答案:

根据OP的进一步说明。默认客户端 不会 重用连接。

确保关闭响应。

读取完后,呼叫者应关闭res.Body。如果未关闭resp.Body,则客户端的基础RoundTripper(通常是Transport)可能无法将与服务器的持久TCP连接重新用于后续的“保持活动”请求。

另外,我发现在调用Close()之前,我还 需要阅读直到响应完成

例如

res, _ := client.Do(req)
io.Copy(ioutil.Discard, res.Body)
res.Body.Close()

为确保重用http.Client连接,请确保执行以下两项操作:

  • 读取直到响应完成(即ioutil.ReadAll(resp.Body)
  • 呼叫 Body.Close()

旧答案,对速率限制很有用,但对OP的作用却不大:

我认为无法通过golang 1.1 http
API设置最大连接数。这意味着如果您不小心的话,可以通过大量的TCP连接(直到用完文件描述符或其他内容)使自己陷入困境。

也就是说, 您可以 通过time.Tick 限制 为特定主机 调用go例程的速率 (并因此限制出站请求和连接)。

例如:

import "time"

requests_per_second := 5
throttle := time.Tick(1000000000 / requests_per_second)

for i := 0; i < 16; i += 1 {
  <-throttle  
  go serveQueue()
}


 类似资料:
  • 我正在使用OkHttp 3.5.0执行超文本传输协议-基准测试。我正在向同一个URL发送数千个请求。 我期望OkHttp-client使用ConnectionPool并一遍又一遍地重用它的连接。但是如果我们研究,我们会看到许多连接处于TIME_WAIT状态: 在数千个请求之后,我得到了一个

  • 在我的应用程序与RabbitMQ服务器失去连接后,我在日志中看到 然后,当我的应用程序重新连接到RabbitMQ服务器时,我在日志中看到 但是,当应用程序重新连接到RabbitMQ服务器时,通道不会重新打开,在RabbitMQ管理控制台中,我可以在“通道”选项卡中看到没有任何通道。 重新连接后,在RabbitMQ服务器日志中: 我使用的是Spring Boot 1.5.3和Spring Boot

  • 如果我的理解是正确的,那么为什么不将的==和!=运算符重载为: 那我们就可以省下两个铸件了?

  • 我找不到任何原因来解释为什么每个自动连接的bean都不是通过代理自动连接的。我知道,因为事务注释不起作用,我在eclipse中调试时检查了自动连线组件。当然,每个组件都实现了一些接口,我使用了与接口相关的注释。我只有一种aop配置: 我将JPA与hibernate、spring mvc、spring webflow、spring security和spring数据一起使用。扩展组织的接口。spri

  • 问题内容: 我正在使用Ubuntu 14.04 64位。这是我的C ++代码,以了解如何使用内存。 这是我的输出: 我希望操作系统会连续分配内存。因此,ptr [0] [0]将位于0x1195020而不是0x1195030!OS在0x1195020-0x119502F,0x1195038-0x0x119504F上做什么用? 问题答案: 因为: 在分配的内存的每个块的开始和结尾处都有一些空间通常用于

  • 问题内容: 对于许多人来说,这可能是一个简单的问题,但令我感到困惑。我从Kathy Sierra挑选了一个示例,该示例显示了抽象类的效用,但我无法理解抽象类的总体重要性。 示例 我们有一个带有抽象方法- &的抽象类。这些方法在 BMW , Volkswagen 和 Audi的 子类中实现。 我的问题是- 为什么我们首先需要有抽象类来为每种汽车类型自定义方法?为什么在宝马的任何一种子类型中都没有这两