网上演练
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

146 lines
4.4 KiB

// QueryableEnumerable<TElement>.cs
//
// Authors:
// Roei Erez (roeie@mainsoft.com)
// Jb Evain (jbevain@novell.com)
//
// Copyright (C) 2008 Novell, Inc (http://www.novell.com)
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
// Needed for NET35 (LINQ)
#if !NET_4_6
using System.Collections;
using System.Collections.Generic;
//using System.Linq.Expressions;
using System.Linq.Expressions.Reimplement;
namespace System.Linq.Reimplement
{
internal interface IQueryableEnumerable : IQueryable
{
IEnumerable GetEnumerable();
}
internal interface IQueryableEnumerable<TElement> : IQueryableEnumerable, IOrderedQueryable<TElement>
{
// Empty
}
internal class QueryableEnumerable<TElement> : IQueryableEnumerable<TElement>, IQueryProvider
{
private readonly IEnumerable<TElement> _enumerable;
private readonly Expression _expression;
public QueryableEnumerable(IEnumerable<TElement> enumerable)
{
_expression = Expression.Constant(this);
_enumerable = enumerable;
}
public QueryableEnumerable(Expression expression)
{
_expression = expression;
}
public Type ElementType
{
get { return typeof(TElement); }
}
public Expression Expression
{
get { return _expression; }
}
public IQueryProvider Provider
{
get { return this; }
}
public IQueryable CreateQuery(Expression expression)
{
return (IQueryable)Activator.CreateInstance
(
typeof(QueryableEnumerable<>).MakeGenericType
(
expression.Type.GetFirstGenericArgument()
),
expression
);
}
public IQueryable<TElem> CreateQuery<TElem>(Expression expression)
{
return new QueryableEnumerable<TElem>(expression);
}
public object Execute(Expression expression)
{
var lambda = Expression.Lambda(TransformQueryable(expression));
return lambda.Compile().DynamicInvoke();
}
public TResult Execute<TResult>(Expression expression)
{
var lambda = Expression.Lambda<Func<TResult>>(TransformQueryable(expression));
return lambda.Compile().Invoke();
}
public IEnumerable GetEnumerable()
{
return _enumerable;
}
public IEnumerator<TElement> GetEnumerator()
{
return Execute<IEnumerable<TElement>>(_expression).GetEnumerator();
}
IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
public override string ToString()
{
if (_enumerable != null)
{
return _enumerable.ToString();
}
else if (_expression == null)
{
return base.ToString();
}
else
{
var constant = _expression as ConstantExpression;
return constant != null && constant.Value == this ? base.ToString() : _expression.ToString();
}
}
private static Expression TransformQueryable(Expression expression)
{
return new QueryableTransformer().Transform(expression);
}
}
}
#endif