使用Azure存储客户端库3.0方法repquery <t>编译错误,其中T是tableentity </t>

时间:2014-01-10 19:26:52

标签: c#-4.0 azure-storage azure-table-storage

我必须遗漏一些显而易见的事情,但以下因编译错误而失败:

internal static IEnumerable<T> GetEntitiesWithCommaSeparatedRowKeys<T>(
string tableName, string partitionKey, 
string commaDelimitedStringOfRowKeys) where T: TableEntity
{
 ....
    TableQuery<T> entitiesQuery = new TableQuery<T>().Where(
                 TableQuery.CombineFilters(
                           TableQuery.GenerateFilterCondition("PartitionKey",
                           QueryComparisons.Equal, partitionKey),
                           TableOperators.And,
                           AzureHelper.GetFilterConditionForCommaDelimitedStringOfRowKeys(commaDelimitedStringOfRowKeys)
                           ));
    // compile error on this line
    IEnumerable<T> entities = table.ExecuteQuery<T>(entitiesQuery);
 ...
 }

我得到的错误是:

'T' must be a non-abstract type with a public parameterless constructor 
in order to use it as parameter 'TElement' in the generic type or 
method 'Microsoft.WindowsAzure.Storage.Table.CloudTable.ExecuteQuery<TElement>
(Microsoft.WindowsAzure.Storage.Table.TableQuery<TElement>, 
Microsoft.WindowsAzure.Storage.Table.TableRequestOptions, 
Microsoft.WindowsAzure.Storage.OperationContext)'

TableEntity显然具有公共无参数构造函数,并且是非抽象的。 当我在TableEntity上点击F12时,以下是来自元数据信息的对象(只是为了确保它正确地解析TableEntity类型)。

namespace Microsoft.WindowsAzure.Storage.Table
{

    public class TableEntity : ITableEntity
    {
        // Summary:
        //     Initializes a new instance of the Microsoft.WindowsAzure.Storage.Table.TableEntity
        //     class.
        public TableEntity();
        ...
    }
}

任何想法? 仅供参考我正在使用Azure客户端库3.0.1.0。

更新:添加了解决类似问题的linked issue

1 个答案:

答案 0 :(得分:10)

事实证明,如果一个方法提供了对类型的约束,那么该约束也必须由该类型的任何调用者转发。

我不知道。

在这种情况下,Table.ExecuteQuery的定义如下所示

 public IEnumerable<TElement> ExecuteQuery<TElement>(TableQuery<TElement> query,
         TableRequestOptions requestOptions = null,
         OperationContext operationContext = null) 
                where TElement : ITableEntity, new();

因此在我的T约束中添加new()可以解决问题。

所以最终的方法声明看起来像

internal static IEnumerable<T> GetEntitiesWithCommaSeparatedRowKeys<T>(string tableName,
       string partitionKey,
       string commaDelimitedStringOfRowKeys) 
              where T : TableEntity , new() //This is new (pun intended :))

在出现此问题的related links之一中找到了相关问题。

我猜测编译器总是可以在实际调用时查找类型约束,但由于TableEntity是公共的(并且没有密封),我想它最终可能成为运行时问题。

另外有趣的是我的方法被标记为内部,这应该真正使编译器能够检查库中的调用者。

无论如何,学到了新东西:)。