如何使用contains在Linq to SQL中搜索字符串

3
我正在使用以下代码在多列中搜索特定字符串。
IEnumerable<UserProductDetailResult> _query
    = from eml in dc.UserProductDetails
      join zk in dc.ZeekUsers on eml.aspnet_User.UserId equals zk.UserId
      where eml.aspnet_User.LoweredUserName.Equals(strUserName.ToLower())
         && (eml.Username.Contains(strSearch)
             || eml.ProductURL.Contains(strSearch)
             || eml.Nickname.Contains(strSearch))
         && !eml.IsDeleted
         && eml.IsActive
      select new UserProductDetailResult
          {
            _userProductDetail = eml,
            _zeekUser = zk
          };

dc是DataContext对象。

但是这将返回0个结果。

以上代码生成的查询语句为:

SELECT [t0].[UPID], [t0].[UserId], [t0].[PID], [t0].[Nickname], [t0].[Username], [t0].[ProductURL], [t0].[StartDt], [t0].[EndDt], [t0].[IsActive], [t0].[InfoData], [t0].[SocialNetworkingData], [t0].[AnalyticKey], [t0].[ProfileID], [t0].[UseDefaultAd], [t0].[UseDefaultEmail], [t0].[IsDeleted], [t0].[CreateDt], [t0].[LastUpdateDt], [t1].[ID], [t1].[UserId] AS [UserId2], [t1].[AccountType], [t1].[FirstName], [t1].[LastName], [t1].[Phone], [t1].[Address1], [t1].[Address2], [t1].[City], [t1].[State], [t1].[ZIP], [t1].[CountryID], [t1].[NickName1], [t1].[Nickname2], [t1].[AlternameEmail], [t1].[ProfileImage], [t1].[ZeekIdStatus], [t1].[RefZeekUserId], [t1].[IsActive] AS [IsActive2], [t1].[FailureCount], [t1].[IsBlocked], [t1].[IsFirstVisit], [t1].[IsWizardPassed], [t1].[IPAddress], [t1].[TimeZoneID], [t1].[CreateDt] AS [CreateDt2], [t1].[LastUpdateDt] AS [LastUpdateDt2]
FROM [dbo].[UserProductDetails] AS [t0]
INNER JOIN [dbo].[ZeekUsers] AS [t1] ON ((
    SELECT [t2].[UserId]
    FROM [dbo].[aspnet_Users] AS [t2]
    WHERE [t2].[UserId] = [t0].[UserId]
    )) = [t1].[UserId]
INNER JOIN [dbo].[aspnet_Users] AS [t3] ON [t3].[UserId] = [t0].[UserId]
WHERE ([t3].[LoweredUserName] = 'username') AND (([t0].[Username] LIKE 'a') OR ([t0].[ProductURL] LIKE 'a') OR ([t0].[Nickname] like 'a')) AND (NOT ([t0].[IsDeleted] = 1)) AND ([t0].[IsActive] = 1)

只要删除下面的搜索行,它就可以工作并返回结果。
&& (eml.Username.Contains(strSearch)
    || eml.ProductURL.Contains(strSearch)
    || eml.Nickname.Contains(strSearch))

但是这不能让我进行搜索

有人能告诉我该怎么做吗?


1
请给出一个搜索字符串的示例和您期望找到的可能值。 - Crono
如果我输入搜索 =“a”,则应返回所有行,其中Username、ProductURL、Nickname中有“a”,而不考虑大小写和SQL中的'%a%'等效词。 - Pratik Gaikwad
2个回答

1
通过您生成的查询,我认为您正在使用 linq-to-sql。您可以使用 SqlMethods.Like 生成正确的 like 查询运算符,来自MSDN:

确定特定字符字符串是否与指定模式匹配。目前仅在LINQ to SQL查询中支持此方法。

例如:
// first sample, any part of string
strSearch = string.Format("%[^a-zA-Z]{0}[^a-zA-Z]%", strSearch);

// on the end of the string
strSearch = string.Format("%[^a-zA-Z]{0}", strSearch);

//on the begining of the string
strSearch = string.Format("{0}[^a-zA-Z]%", strSearch);

在你的查询语句中...
(SqlMethods.Like(eml.Username, strSearch) 
|| SqlMethods.Like(eml.ProductURL, strSearch) 
|| SqlMethods.Like(eml.Nickname, strSearch))

否则,您可以在查询之前将%字符添加到strSearch字符串中,以便查询结果包含字符串的任何部分,例如:
strSearch = string.Contat("%", strSearch, "%");

@Felipe... 这个不起作用... 如果我在搜索和列中都使用 ToUpper(),那么它就可以工作... 你能建议一些不考虑大小写的东西吗? - Pratik Gaikwad

1
我创建了一个NuGet包,可以帮助你。它将使你能够使用以下语法:
var result = dc.UserProductDetails
               .Where(eml => eml.IsActive && !eml.IsDeleted)
               .Search(eml => eml.aspnet_User.LoweredUserName) // Search LoweredUsername
               .IsEqual(strUserName)                           // when equals strUsername
               .Search(eml => eml.UserName,                    // Search UserName
                       eml => eml.ProductURL,                  // OR ProductUrl
                       eml => eml.Nickname)                    // OR Nickname
               .Containing(strSearch)                          // When contains strSearch
               .Select(eml => new UserProductDetailResult      // Build result...
                              {
                                   _userProductDetail = eml
                              });

您可以从这里下载软件包。

http://www.nuget.org/packages/NinjaNye.SearchExtensions

希望这能帮到你,同时你也可以查看GitHub页面以获取更详细的信息。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接