基本上,我试图编写以下扩展方法,以避免必须不断编写.Suffix("keyword")
。我不喜欢到处都是字符串文字,并且我的ElasticSearch索引的多个属性碰巧需要添加关键字后缀才能正确查询。
public static class NestHelperExtensions
{
public static object UseKeywordSuffix(this object @object)
{
return @object.Suffix("keyword");
}
}
如果我使用上面介绍的方法,则后缀将不适用,但我不确定为什么它不起作用。我可能会以错误的方式进行操作,也许有一种方法可以添加模型属性或某些东西来构建nestCilent.Search<T>
,但是当我尝试使用KeywordAttribute
时,似乎并没有也可以工作。
任何人都可以提供关于为何并非如此简单的解释,如果可能的话,您可以提供示例或解决方法吗?使用属性?
答案 0 :(得分:1)
这不起作用,因为Suffix
method is specifically handled when visiting the field expression
if (methodCall.Method.Name == nameof(SuffixExtensions.Suffix) && methodCall.Arguments.Any())
{
VisitConstantOrVariable(methodCall, _stack);
var callingMember = new ReadOnlyCollection<Expression>(
new List<Expression> { { methodCall.Arguments.First() } }
);
Visit(callingMember);
return methodCall;
}
因此,像上面这样的扩展方法必须被称为Suffix
,并且它必须至少具有一个参数。您可能会想到提供一个默认值为“ keyword”的可选参数会起作用,但是表达式树不支持此功能,因此不起作用。
另一种方法是利用AppendSuffix
上的Expression<Func<T, object>>
扩展方法来构建一些东西。最好的方法是将lambda表达式从流利的调用中拉出并放入变量
public static class NestHelperExtensions
{
public static Expression<Func<T, object>> KeywordSuffix<T>(this Expression<Func<T, object>> expression)
{
return expression.AppendSuffix("keyword");
}
}
var client = new ElasticClient();
Expression<Func<Person, object>> firstName = f => f.FirstName;
var searchResponse = client.Search<Person>(s => s
.Query(q => q
.Match(m => m
.Field(firstName.KeywordSuffix())
.Query("bar")
)
)
);
不太好的方式是将lambda表达式强制转换为Expression<Func<T, object>>
内联
var searchResponse = client.Search<Person>(s => s
.Query(q => q
.Match(m => m
.Field(((Expression<Func<Person, object>>)(f => f.FirstName)).KeywordSuffix())
.Query("bar")
)
)
);
另一种也许更简单的方法是为字符串"keyword"
引入一个常量,并在Suffix
扩展方法中使用该常量;避免在各处使用字符串文字。