我需要从数据库中找到满足特定格式约定的最高值。具体来说,我想找到看起来像
的最高值EU999999('9'为任何数字)
select max(col)将返回类似'EUZ ...'的内容,例如我要排除的内容。 以下查询可以解决问题,但我无法通过Linq-to-SQL生成此问题。似乎没有SQL Server中的isnumeric()函数的转换。
select max(col) from table where col like 'EU%'
and 1=isnumeric(replace(col, 'EU', ''))
编写数据库函数,存储过程或其他任何性质的东西都是我首选解决方案的列表,因为这个表是我的应用程序的核心,我不能轻易地用其他东西替换表对象。
下一个最佳解决方案是什么?
答案 0 :(得分:12)
虽然缺少ISNUMERIC
,但您总是可以尝试几乎等效的NOT LIKE '%[^0-9]%
,即字符串中没有非数字,或者字符串为空或仅包含数字:
from x in table
where SqlMethods.Like(x.col, 'EU[0-9]%') // starts with EU and at least one digit
&& !SqlMethods.Like(x.col, '__%[^0-9]%') // and no non-digits
select x;
当然,如果您知道位数是固定的,可以简化为
from x in table
where SqlMethods.Like(x.col, 'EU[0-9][0-9][0-9][0-9][0-9][0-9]')
select x;
答案 1 :(得分:11)
您可以通过向DataContext的分部类添加方法来使用ISNUMERIC
函数。它类似于使用UDF。
在DataContext的部分类中添加:
partial class MyDataContext
{
[Function(Name = "ISNUMERIC", IsComposable = true)]
public int IsNumeric(string input)
{
throw new NotImplementedException(); // this won't get called
}
}
然后您的代码将以这种方式使用它:
var query = dc.TableName
.Select(p => new { p.Col, ReplacedText = p.Col.Replace("EU", "") })
.Where(p => SqlMethods.Like(p.Col, "EU%")
&& dc.IsNumeric(p.ReplacedText) == 1)
.OrderByDescending(p => p.ReplacedText)
.First()
.Col;
Console.WriteLine(query);
或者你可以使用MAX:
var query = dc.TableName
.Select(p => new { p.Col, ReplacedText = p.Col.Replace("EU", "") })
.Where(p => SqlMethods.Like(p.Col, "EU%")
&& dc.IsNumeric(p.ReplacedText) == 1);
var result = query.Where(p => p.ReplacedText == query.Max(p => p.ReplacedText))
.First()
.Col;
Console.WriteLine("Max: {0}, Result: {1}", max, result);
根据您的最终目标,可以停在max
变量并在其前面添加“EU”文本,以避免获取列名称的第二个查询。
编辑:正如评论中所提到的,这种方法的缺点是对文本而不是数值进行排序,而且SQL上的Int32.Parse
目前没有翻译。
答案 2 :(得分:1)
正如您所说,从LINQ到SQL没有IsNumeric的翻译。有几个选项,你已经编写了数据库函数和存储过程。我想再添加两个。
选项1:您可以通过将LINQ to SQL与LINQ to Objects混合来实现这一点,但是当您拥有一个大型数据库时,不要期望获得出色的性能:
var cols = (from c in db.Table where c.StartsWith("EU") select c).ToList();
var stripped = from c in cols select int.Parse(c.Replace("EU", ""));
var max = stripped.Max();
选项2:更改数据库架构: - )
答案 3 :(得分:0)
我的建议是回退到内联SQL并使用DataContext.ExecuteQuery()方法。您将使用您在开头发布的SQL查询。
这是我在类似情况下所做的。由于缺乏类型检查和可能的语法错误,并不是理想的,只是确保它包含在任何单元测试中。并非Linq语法涵盖了所有可能的查询,因此首先存在ExecuteQuery。