15

The specified cast from a materialized 'System.Guid' type to the 'System.Int32' type is not valid.

我们有几个 WCF 服务具有 Multiple 和InstanceContextModeSingle 的并发模式。我们的架构专注于使用基于构造函数的依赖注入的松散耦合模型。这又是使用 Unity 2.0 实现的(每个服务的 web.config 具有在统一容器部分中定义的接口和类型之间的映射)。我们的依赖项之一是使用 Entity Framework 4 与 MSSql Server 通信的 DAL 程序集(数据访问层)。与数据库对话的类也包含在统一映射中。

当我们运行集成测试时,一切都很好。但是当我们转移到我们的性能环境来运行负载测试(2、3、4 个并发用户)时,我们开始看到以下错误:

System.InvalidOperationException: The 'auth_token' property on 'Session' could not be set to a 'Int32' value. You must set this property to a non-null value of type 'Guid'.

使用以下堆栈:

at System.Data.Common.Internal.Materialization.Shaper.ErrorHandlingValueReader`1.GetValue(DbDataReader reader, Int32 ordinal)
at System.Data.Common.Internal.Materialization.Shaper.GetPropertyValueWithErrorHandling[TProperty](Int32 ordinal, String propertyName, String typeName)
at lambda_method(Closure , Shaper )
at System.Data.Common.Internal.Materialization.Shaper.HandleEntityAppendOnly[TEntity](Func`2 constructEntityDelegate, EntityKey entityKey, EntitySet entitySet)
at lambda_method(Closure , Shaper )
at System.Data.Common.Internal.Materialization.Coordinator`1.ReadNextElement(Shaper shaper)
at System.Data.Common.Internal.Materialization.Shaper`1.SimpleEnumerator.MoveNext()
at System.Linq.Enumerable.First[TSource](IEnumerable`1 source)
at System.Linq.Queryable.First[TSource](IQueryable`1 source)
at MISoa.DataAccessLayer.Authentication.AuthenticationDB.RetrieveSession(Guid authToken)
at MISoa.DataAccessLayer.Authentication.AuthenticationAccess.RetrieveSession(String sessionToken)

这是罪魁祸首方法:

public Session RetrieveSession(Guid authToken)
    {
        CheckDBContext();
        var sessions = (from r in _dbContext.Sessions
                where r.auth_token == authToken
                select r);
        return sessions.Count() > 0 ? sessions.First() : null;
    }

CheckDBContext方法只是检查数据库上下文是否为空,如果是,则抛出自定义异常。

emdx 会话实体对象具有以下公共属性:

Guid auth_token
DateTime time_stamp
String user_id
String app_id

所以,看起来有时上面的 linq 从数据库中返回一些其他对象,其中第一列是 anint而不是 a guid?如果是这样 - 为什么?我是否有多个线程覆盖彼此的数据库上下文的问题?顺便说一句 - 我们将实例化数据库上下文的代码抽象为一个单独的类(BaseDB),该类也由统一处理。所以,因为服务是单例的,所以我为每个人都有一个 BaseDB 实例,对吧?这是这里的问题吗?

哦,还有一件事。我们被告知我们将拥有 MSSql 2005,因此在 edmx 文件中我们拥有ProviderManifestToken="2005". 但是我刚查了一下,我们性能数据库的服务器是2008版的。这是个问题吗?

谢谢您的帮助。

4

2 回答 2

2

我是否有多个线程覆盖彼此的数据库上下文的问题?

是的。见这里:http: //msdn.microsoft.com/en-us/library/system.data.objects.objectcontext.aspx

引用上面链接中的大黄色框:

该类ObjectContext不是线程安全的。ObjectContext多线程场景下无法保证数据对象的完整性。

您可能需要考虑投入[ThreadStaticAttribute]您的_dbContext领域。

于 2012-05-17T17:19:21.510 回答
0

我怀疑当没有结果时会出现您的问题,并且在生成的 SQL 中的某处将 null value==0 用于 Guid 字段比较。尝试将比较作为字符串(两者都为 .ToString() ),看看是否能发现导致问题的情况。

于 2012-06-01T12:22:52.323 回答