由于异步,在 WebApi 中使用 HttpContext.Current 是危险的

2023-12-14

我的问题与此有点相关:WebApi 相当于具有依赖注入的 HttpContext.Items.

我们想使用 Ninject 在 WebApi 区域中使用 HttpContext.Current 注入一个类。

我担心的是,这可能是非常危险,如 WebApi (一切?) 是异步的。

如果我在这些方面有错误,请纠正我,这是我到目前为止所调查的:

  1. HttpContext.Current通过Thread获取当前上下文(我直接查看了实现)。

  2. 在异步任务中使用 HttpContext.Current 是不可能的,因为它可以在另一个线程上运行。

  3. WebApi 使用 IHttpController 和方法Task<HttpResponseMessage> ExecuteAsync=>每个请求都是异步的=>你不能在action方法中使用HttpContext.Current。甚至可能会发生更多请求在同一个线程上巧合执行的情况。

  4. 用于创建控制器并将内容注入到构造函数中IHttpControllerActivator与使用sync method IHttpController Create。这是 ninject 创建 Controller 及其所有依赖项的地方。


  • 如果我在所有这 4 点上都是正确的,那么使用HttpContext.Current操作方法内部或下面的任何层都是非常危险的,并且可能会产生意想不到的结果。我在 StackOverflow 上看到很多已接受的答案都表明了这一点。在我看来,这可以工作一段时间,但在负载下会失败。

  • 但是当使用 DI 创建一个 Controller 及其依赖项时,就没有问题了,因为它运行在一个单独的线程上。我可以从构造函数中的 HttpContext 获取一个值,这会安全吗?。我想知道是否每个控制器都是在每个请求的单线程上创建的,因为这可能会在重负载下导致问题,其中来自 IIS 的所有线程都可能被消耗。

只是为了解释为什么我想注入 HttpContext 的东西:

  • 一种解决方案是在控制器操作方法中获取请求,并将所需的值作为参数传递给所有层,直到在代码深处的某个地方使用它。
  • 我们想要的解决方案:之间的所有层都不受此影响,我们可以在代码深处的某个地方使用注入的请求(例如,在某些ConfigurationProvider这取决于 URL)

如果我完全错误或者我的建议是正确的,请给我你的意见,因为这个主题似乎非常复杂。


HttpContext.Current通过Thread获取当前上下文(我直接查看了实现)。

更正确的说法是HttpContext应用于线程;或者一个线程“进入”HttpContext.

在异步任务中使用 HttpContext.Current 是不可能的,因为它可以在另一个线程上运行。

一点也不;的默认行为async/await将在任意线程上恢复,但该线程将在恢复之前进入请求上下文async method.


这其中的关键是SynchronizationContext。我有一个关于该主题的 MSDN 文章如果你不熟悉它。 ASynchronizationContext为平台定义“上下文”,常见的是 UI 上下文(WPF、WinPhone、WinForms 等)、线程池上下文和 ASP.NET 请求上下文。

ASP.NET 请求上下文管理HttpContext.Current以及其他一些事情,例如文化和安全。 UI 上下文都与单个线程紧密关联(theUI 线程),但 ASP.NET 请求上下文不绑定到特定线程。然而,它只允许请求上下文中有一个线程一次.

解决方案的另一部分是如何async and await工作。我有一个async我的博客上有介绍,描述了他们的行为。总之,await默认情况下将捕获当前上下文(即SynchronizationContext.Current除非是null),并使用该上下文来恢复async方法。所以,await自动捕获 ASP.NETSynchronizationContext并将恢复async该请求上下文中的方法(从而保留文化、安全性和HttpContext.Current).

If you await ConfigureAwait(false),那么你明确地告诉await to not捕捉上下文。

请注意,ASP.NET 确实必须更改其SynchronizationContext干净利落地工作async/await。您必须确保应用程序是针对 .NET 4.5 编译的,并且还在其 web.config 中明确针对 4.5;这是新 ASP.NET 4.5 项目的默认设置,但如果您从 ASP.NET 4.0 或更早版本升级现有项目,则必须显式设置。

您可以通过针对 .NET 4.5 执行应用程序并观察来确保这些设置正确SynchronizationContext.Current。如果是AspNetSynchronizationContext,那你就很好了;如果它是LegacyAspNetSynchronizationContext,那么设置是错误的。

只要设置正确(并且您使用的是 ASP.NET 4.5AspNetSynchronizationContext),然后你就可以安全地使用HttpContext.Currentawait不用担心。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

由于异步,在 WebApi 中使用 HttpContext.Current 是危险的 的相关文章

随机推荐

  • ColdFusion 相当于 PHP hash_hmac

    key 12345678876543211234567887654321 iv 1234567887654321 plaindata This is a test string enc base64 encode mcrypt encryp
  • 如何使用 System.Drawing 绘制表格

    我想使用 System Drawings 绘制一个表格 然后用一些文本填充单元格 该文本每隔几秒钟就会在不同的时刻发生变化 这是一个游戏 其中有一个网格 每隔几秒钟 随机单元格就会显示一个数字 然后用户必须在其下面的文本框中输入答案 此外
  • R中字符类的计数函数

    我的代码正常工作 然后 RStudio 崩溃了 当我重新打开它时 我的一行代码现在不起作用 CodeTable lt count unique Data Code 以前 这创建了一个包含 3 列的简单数据库 1 数字顺序 2 唯一代码 3
  • 从android中点击通知获取数据

    嘿伙计们 我需要帮助如何从使用广播接收器设置的待处理意图中获取数据 我想要发生的是在单击通知时获取我的活动所需的 id 数据 这就是我制作额外内容的方式 public class AlertReceiver extends Broadcas
  • 如何在 Haskell 中计算直方图?

    I found Statistics Sample Histogram 但我似乎无法使用它 如果我希望能够将列表分为四个类别 我希望能够执行以下操作 import Statistics Sample Histogram histogram
  • Opencv:您的设备似乎不支持相机(或者已锁定)

    我在 Android 4 0 4 的三星平板电脑上使用 opencv 2 4 5 的 Face Detection 和 Android 4 2 2 该应用程序可以使用前置摄像头来校准我的脸部 但是 当我使用此应用程序同时播放视频和面部检测时
  • 应用程序启动时导航抽屉始终处于膨胀状态

    我正在尝试将导航抽屉添加到我的主要活动中 在 Activity main xml 的设计视图中 它应该作为阴影动画在活动布局的左侧可见 我不知道我应该在这里使用什么术语 向右拖动 滑动时应该可见 但就我而言 它默认覆盖整个活动 并且不显示活
  • 获取 Dynamics 365 的身份验证令牌时出现错误 AADSTS90002

    我在尝试从 Net 客户端使用 Dynamics 365 进行身份验证时遇到以下错误 AADSTS90002 Tenant authorize not found This may happen if there are no active
  • 无法在带有 Xcode 4.3.1 和 IOS 的 iPhone 4.2.1 上运行应用程序

    我希望有人可以提供帮助 这个问题已经困扰我一段时间了 我正在运行 Xcode 4 3 1 和基本 SDK iOS 5 1 iPhone版本是4 2 1 我相信是旧的3S型号 当点击 运行 时 Xcode 编译正常 并表示它正在手机上运行我的
  • 将 unicode 转换为 char

    如何将 Unicode 字符串转换为char or char const in 内河码头 c String text Hello world char txt AnsiString text c str Older text t str i
  • 如何在 C# 中监控剪贴板内容变化? [复制]

    这个问题在这里已经有答案了 I want to have this feature in my C program When the user do Ctrl C or Copy anywhere i e when the clipboar
  • 在matlab中找到峰值

    假设我们确定向量中的峰值如下 我们有长度为 m 的实数值一维向量 或者 x 1 x 2 x 米 如果 x 1 gt x 2 则显然对于第一个点 Peak 1 x 1 否则我们将 x 3 与 x 2 进行比较 如果 x 3 indexes p
  • 如何将工具提示添加到 JavaFX Canvas 的矩形区域

    在我的 JavaFX 应用程序中 我有一个TableView具有多个列 其中一列以图形形式显示数据 为此 我创建了一个CanvasCell创建并管理自己的对象Canvas来处理绘图 绘图部分工作得很好 我现在想把Tooltips范围内的一些
  • 我怎样才能得出这个观点?

    List gt l user我有一个List具有用户引用字段的节点 l user Story gt s user然后我有Story也有用户引用字段的节点 s user 之间没有直接联系List and Story 我想添加一个视图List列
  • GWT 在 CloseHandler 中检测浏览器刷新

    我有一个 GWT 应用程序 我想在用户离开应用程序时运行一些代码以强制注销并删除任何数据等 为此 我使用 CloseHandler 并使用 Window addCloseHandler 注册它 我注意到 当单击刷新按钮时 会运行 onClo
  • dalvikvm:找不到类“android.*”

    在此输入图像描述开发时 应用程序在Android 5 0系统上可以正常使用 但是在5 0以下的系统中会出现应用程序无响应的情况 但不知道是什么原因 错误日志如下 E dalvikvm Could not find class android
  • 如何限制使用 argparse 解析的值(例如,将整数限制为正值)?

    到目前为止我有这个代码 import argparse parser argparse ArgumentParser parser add argument g games type int default 162 help The num
  • 如何在 C# 中重置计时器?

    有三种Timer我所知道的课程 System Threading Timer System Timers Timer and System Windows Forms Timer 但这些都没有 Reset 函数会将当前经过的时间重置为 0
  • C# 将 PDF 文件上传到 Firebase 项目存储?

    你们知道如何将 PDF 文件直接上传到 Firebase 项目存储吗 我在互联网上搜索了很多 但一无所获 有一些 C 的库吗 或者有 C 和 Firebase 的文档吗 请大家帮忙 谢谢 EDIT 好的 我找到了一个库 FirebaseSh
  • 由于异步,在 WebApi 中使用 HttpContext.Current 是危险的

    我的问题与此有点相关 WebApi 相当于具有依赖注入的 HttpContext Items 我们想使用 Ninject 在 WebApi 区域中使用 HttpContext Current 注入一个类 我担心的是 这可能是非常危险 如 W