2

我有以下 LINQ2SQL 查询:

var map =
                dbContext.TCPDriverMappings.FirstOrDefault(
                      c => c.DriverFacilityId == tcpDms.FacilityId &&
                                 c.DriverControlledParameterId == controlledParamId &&
                                 c.DriverValue == value);

所有类型都是字符串。

在我的数据库中,我有一行,必须通过查询返回。

value="0", controlledParamId =null and FacilityId ="abc"此查询返回 null 时,但当我写以下内容时:

var test = dbContext.TCPDriverMappings.FirstOrDefault(
                          c => c.DriverFacilityId == "abc" &&
                                     c.DriverControlledParameterId == null &&
                                     c.DriverValue == "0");

测试没有null

我究竟做错了什么?

PS我也试过c.DriverControlledParameterId.Equals(controlledParamId),但它也不起作用。

4

1 回答 1

5

问题是,LINQ2SQL 对表达式有特殊处理c.DriverControlledParameterId == null。它被翻译成 SQL DriverControlledParameterId IS NULL
不过c.DriverControlledParameterId = controlledParamId是翻译成SQL DriverControlledParameterId = :p1,就算controlledParamIdnull。而在 SQLDriverControlledParameterId = NULL中是未定义的,因此 never TRUE

如何解决:具体处理null此案:

TCPDriverMapping test;
if(controlledParamId == null)
    test = dbContext.TCPDriverMappings.FirstOrDefault(
               c => c.DriverFacilityId == "abc" &&
                    c.DriverControlledParameterId == null &&
                    c.DriverValue == "0");
else 
    test = dbContext.TCPDriverMappings.FirstOrDefault(
               c => c.DriverFacilityId == "abc" &&
                    c.DriverControlledParameterId == controlledParamId &&
                    c.DriverValue == "0");

或者像这样:

var test = dbContext.TCPDriverMappings.FirstOrDefault(
               c => c.DriverFacilityId == "abc" &&
                    ((controlledParamId == null &&
                      c.DriverControlledParameterId == null) || 
                     c.DriverControlledParameterId == controlledParamId) &&
                    c.DriverValue == "0");

或者像这样:

IQueryable<TCPDriverMapping> query =
    dbContext.TCPDriverMappings.Where(c => c.DriverFacilityId == "abc" &&
                                           c.DriverValue == "0");
if(controlledParamId == null)
    query = query.Where(c => c.DriverControlledParameterId == null);
else
    query = query.Where(c => c.DriverControlledParameterId == controlledParamId);

var test = query.FirstOrDefault();

第三个选项是我会使用的。在我看来,这比选项 2 更具可读性,并且没有像第一个那样的重复代码。

于 2013-09-25T13:04:59.803 回答