ASP.NET и ThreadStatic как часть реализации TransactionScope
Мне было интересно, как класс TransactionScope работает для сохранения транзакции между вызовами различных методов (без необходимости передавать ее в качестве параметра), и я пришел к этому сомнению. У меня есть два соображения по этому вопросу:
1
Рассматривая реализацию TransactionScope с помощью Telerik JustDecompile, я обнаружил, что текущая транзакция хранится в элементе ThreadStatic класса System.Transactions.ContextData (код ниже).
internal class ContextData
{
internal TransactionScope CurrentScope;
internal Transaction CurrentTransaction;
internal DefaultComContextState DefaultComContextState;
[ThreadStatic]
private static ContextData staticData;
internal WeakReference WeakDefaultComContext;
internal static ContextData CurrentData
{
get
{
ContextData contextDatum = ContextData.staticData;
if (contextDatum == null)
{
contextDatum = new ContextData();
ContextData.staticData = contextDatum;
}
return contextDatum;
}
}
public ContextData()
{
}
}
Свойство CurrentData вызывается методом PushScope() TransactionScope, а последнее используется большинством конструкторов TransactionScope.
private void PushScope()
{
if (!this.interopModeSpecified)
{
this.interopOption = Transaction.InteropMode(this.savedCurrentScope);
}
this.SetCurrent(this.expectedCurrent);
this.threadContextData.CurrentScope = this;
}
public TransactionScope(TransactionScopeOption scopeOption)
{
// ...
this.PushScope();
// ...
}
Хорошо, я думаю, я нашел, как они это делают.
2
Я читал о том, как плохо использовать члены ThreadStatic для хранения объектов в ASP.NET (http://www.hanselman.com/blog/ATaleOfTwoTechniquesTheThreadStaticAttributeAndSystemWebHttpContextCurrentItems.aspx) из-за переключения потоков ASP.NET, которое может произойти, поэтому эти данные могут быть потеряны среди рабочих потоков.
Итак, похоже, что TransactionScope не должен работать с ASP.NET, верно? Но поскольку я использовал его в своих веб-приложениях, я не помню ни одной проблемы, с которой я столкнулся в связи с потерей данных транзакций.
Мой вопрос здесь заключается в том, "какова хитрость TransactionScope в переключении потоков ASP.NET?".
Сделал ли я поверхностный анализ того, как TransactionScope хранит свои объекты транзакций? Или класс TransactionScope не был создан для работы с ASP.NET, и меня можно считать счастливчиком, у которого никогда не было никаких проблем с этим?
Может ли кто-нибудь, кто знает "очень глубокие тайны".NET, объяснить это мне?
Спасибо
1 ответ
Я полагаю, что переключение потоков ASP.NET происходит только в определенных ситуациях (включая асинхронные операции ввода-вывода) и в начале жизненного цикла запроса. Как правило, когда управление передается фактическому обработчику http (например, Page), поток не переключается. Я считаю, что в большинстве случаев объем транзакции будет инициализирован только после этого (после page_init/load) и не должен быть проблемой.
Вот несколько ссылок, которые могут вас заинтересовать:
http://piers7.blogspot.com/2005/11/threadstatic-callcontext-and_02.html
http://piers7.blogspot.com/2005/12/log4net-context-problems-with-aspnet.html