【问题标题】:realm.xamarin throws error: The method 'Contains' is not supported. same for 'Any'realm.xamarin 抛出错误:不支持“包含”方法。 '任何'都一样
【发布时间】:2016-06-07 09:25:03
【问题描述】:

我正在编写一个查询,它需要在 realm.xamarin linq 查询中使用类似“WHERE IN”的子句。以下是我尝试过的查询:

var IdsToMatch = observations.Select(x => x.Id)
var results = from d in realm.All<Observations>() where IdsToMatch.Any(p => p == d.Id) select d;

也试过Contains:

var results = from d in realm.All<Observations>() where IdsToMatch.Contains(d.Id) select d;

但这会引发错误:

The method 'Contains' is not supported
The method 'Any' is not supported

解决办法是什么?

编辑:

这是错误的堆栈跟踪:

未处理的异常: 06-07 15:04:55.097 I/MonoDroid(24526): System.NotSupportedException: 方法 'Any' 不受支持 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.RealmResultsVisitor.VisitMethodCall (System.Linq.Expressions.MethodCallExpression m) [0x00596] in :0 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.ExpressionVisitor.Visit (System.Linq.Expressions.Expression exp) [0x000ec] in :0 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.RealmResultsVisitor.VisitMethodCall (System.Linq.Expressions.MethodCallExpression m) [0x0006a] in :0 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.ExpressionVisitor.Visit (System.Linq.Expressions.Expression exp) [0x000ec] in :0 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.RealmResults1[T].CreateResultsHandle () [0x00037] in <filename unknown>:0 06-07 15:04:55.097 I/MonoDroid(24526): at Realms.RealmResults1[T].get_ResultsHandle () [0x0000d] in :0 06-07 15:04:55.097 I/MonoDroid(24526): 在 Realms.RealmResults1[T].GetEnumerator () [0x00000] in <filename unknown>:0 06-07 15:04:55.097 I/MonoDroid(24526): at System.Collections.Generic.List1[T]..ctor (IEnumerable1 collection) [0x00073] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/collections/generic/list.cs:104 06-07 15:04:55.098 I/MonoDroid(24526): at System.Linq.Enumerable.ToList[TSource] (IEnumerable1 源) [0x00011] 在 /Users/builder/data/lanes/ 3053/a94a03b5/source/mono/external/referencesource/System.Core/System/Linq/Enumerable.cs:835 06-07 15:04:55.098 I/MonoDroid(24526): 在 VHS.MobileApp.Mediq.DataAccess.ObservationDatabase+c__DisplayClass13.b__8 () [0x00147] 在 d:\rwagh\mediq\Code\VHS.MobileApp.Mediq .DataAccess\ObservationDatabase.cs:65 06-07 15:04:55.098 I/MonoDroid(24526): 在 System.Threading.Tasks.Task1[TResult].InnerInvoke () [0x00012] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/threading/Tasks/Future.cs:686 06-07 15:04:55.098 I/MonoDroid(24526): at System.Threading.Tasks.Task.Execute () [0x00016] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/threading/Tasks/Task.cs:2523 06-07 15:04:55.098 I/MonoDroid(24526): --- End of stack trace from previous location where exception was thrown --- 06-07 15:04:55.098 I/MonoDroid(24526): at System.Runtime.ExceptionServices.ExceptionDispatchInfo.Throw () [0x0000c] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/runtime/exceptionservices/exceptionservicescommon.cs:143 06-07 15:04:55.098 I/MonoDroid(24526): at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess (System.Threading.Tasks.Task task) [0x00047] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/runtime/compilerservices/TaskAwaiter.cs:201 06-07 15:04:55.098 I/MonoDroid(24526): at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification (System.Threading.Tasks.Task task) [0x0002e] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/runtime/compilerservices/TaskAwaiter.cs:170 06-07 15:04:55.098 I/MonoDroid(24526): at System.Runtime.CompilerServices.TaskAwaiter.ValidateEnd (System.Threading.Tasks.Task task) [0x0000b] in /Users/builder/data/lanes/3053/a94a03b5/source/mono/external/referencesource/mscorlib/system/runtime/compilerservices/TaskAwaiter.cs:142 06-07 15:04:55.098 I/MonoDroid(24526): at System.Runtime.CompilerServices.TaskAwaiter1[TResult].GetResult () [0x00000] 在 /Users/builder/data/lanes/3053/a94a03b5 /source/mono/external/referencesource/mscorlib/system/runtime/compilerservices/TaskAwaiter.cs:372

编辑 2:

var IdsToMatch = observations.Select(x => x.Id)
var results = from d in realm.All<Observations>() where IdsToMatch.Any(p => p == d.Id) select d;

试过了:var IdsToMatch = observations.Select(x =&gt; x.Id).AsQueryable() 按照https://github.com/realm/realm-dotnet/blob/7187390529201ba843cd105fc6e3e11acb0c6217/Realm.Shared/linq/RealmResultsVisitor.cs#L121 所说的

internal override Expression VisitMethodCall(MethodCallExpression m)
        {
            if (m.Method.DeclaringType == typeof(Queryable)) { ...

但这也没有用。

【问题讨论】:

  • 我的猜测是表达式访问者不支持您正在使用的外部枚举的那些方法。您要么必须: - 获取不带 Where 子句的所有记录(使用 .ToLisT()),然后在这些内存结果中使用 Contains(可能不是一个好主意),要么 - 循环遍历可枚举和查询中的每个 Id对于那些ID一个接一个。虽然。你试过p =&gt; p.Id.In(IdsToMatch)
  • @AndyJames "Id" 是原始类型(Int/String),因此 "In" 不是可用的函数。
  • 很多框架都提供In作为通用扩展方法,所以它对object有效。但是,如果您没有在任何地方定义它,尤其是在 Realms 中,表达式访问者将不知道那是什么并且无论如何都会死去。在这种情况下可能不是一个选择。
  • 我已经为此github.com/realm/realm-dotnet/issues/604 添加了一个领域问题,但请提供更多信息,因为从您的示例中并不能完全清楚您在做什么。 IdsToMatch 只是一个 ID 列表吗?
  • 我怀疑您正在尝试用 SQL 术语执行类似于 IN 查询的操作,请参阅 stackoverflow.com/questions/2334327/…

标签: c# linq xamarin realm realm-net


【解决方案1】:

好的,这很有趣。答案(现在)是构建一个 LINQ Expression Tree

Realm 支持 LINQ - 我们解析表达式树(好的,缺少一些位,但基础知识已经存在,我们会定期改进 - 我们添加包含)。

因此,任何时候您需要在运行时构建完全任意的查询表达式,您都可以借助 LINQ 的强大功能做到这一点。

下面的示例可能看起来有点复杂,但重点是它使用了标准表达式树,因此您可以使用任何其他构建此类树的代码。

class HasIntKey : RealmObject {
    public int UserKey { get; set; }
}


private void FindingManyMatches()
{
    using (var theRealm = Realm.GetInstance ("TestingManyKeys.realm")) {
        // create some sample data
        theRealm.Write (() => {
            for (int i = 1; i < 10000; i++) {
                var obj = theRealm.CreateObject<HasIntKey> ();
                obj.UserKey = i;
            }
        });
        var allInts = theRealm.All<HasIntKey>();
        Console.WriteLine ($"Created {allInts.Count()} objects");

        var idsToMatch = new[] {42,1003,400,57, 6009};

        // technique for how to search for many matches
        // use Expression Trees to dynamically build the LINQ statement
        // see https://msdn.microsoft.com/en-us/library/mt654267.aspx
        ParameterExpression pe = Expression.Parameter (typeof(HasIntKey), "p");

        // building an expression like:
        // allInts.Where (p => p.UserKey == idsToMatch[0] || p.UserKey == idsToMatch[1]...);
        Expression chainedByOr = null;
        // left side of the == will be the same for each clause we add in the loop
        Expression left = Expression.Property(pe, typeof(HasIntKey).GetProperty("UserKey"));  
        foreach (int anId in idsToMatch) {
            // Create an expression tree that represents the expression 'p.UserKey == idsToMatch[n]'.
            Expression right = Expression.Constant(anId);
            Expression anotherEqual = Expression.Equal(left, right);
            if (chainedByOr == null)
                chainedByOr = anotherEqual;
            else 
                chainedByOr = Expression.OrElse (chainedByOr, anotherEqual);
        }
        MethodCallExpression whereCallExpression = Expression.Call(
            typeof(Queryable),
            "Where",
            new Type[] { allInts.ElementType },
            allInts.Expression,
            Expression.Lambda<Func<HasIntKey, bool>>(chainedByOr, new ParameterExpression[] { pe }));

        // Create an executable query from the expression tree.
        IQueryable<HasIntKey> results = allInts.Provider.CreateQuery<HasIntKey>(whereCallExpression);

        // now we have our results, display some details!
        Console.WriteLine ($"Found {results.Count()} objects");

        // Enumerate the results.
        foreach (HasIntKey anObj in results)
            Console.WriteLine($"Found key {anObj.UserKey}");
    }
}

【讨论】:

猜你喜欢
  • 2021-10-20
  • 1970-01-01
  • 1970-01-01
  • 2015-04-30
  • 2019-11-19
  • 1970-01-01
  • 1970-01-01
  • 2021-03-12
  • 1970-01-01
相关资源
最近更新 更多