Методы расширения должны быть определены в неуниверсальном статическом классе


214

Я получаю ошибку:

Методы расширения должны быть определены в неуниверсальном статическом классе

На линии:

public class LinqHelper

Вот вспомогательный класс, основанный на коде Марка Гевелса. Я действительно не понимаю, что означает эта ошибка, так как я уверен, что она работала нормально, когда я оставил ее в пятницу!

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Linq.Expressions;
using System.Reflection;

/// <summary>
/// Helper methods for link
/// </summary>
public class LinqHelper
{
    public static IOrderedQueryable<T> OrderBy<T>(this IQueryable<T> source, string property)
    {
        return ApplyOrder<T>(source, property, "OrderBy");
    }
    public static IOrderedQueryable<T> OrderByDescending<T>(this IQueryable<T> source, string property)
    {
        return ApplyOrder<T>(source, property, "OrderByDescending");
    }
    public static IOrderedQueryable<T> ThenBy<T>(this IOrderedQueryable<T> source, string property)
    {
        return ApplyOrder<T>(source, property, "ThenBy");
    }
    public static IOrderedQueryable<T> ThenByDescending<T>(this IOrderedQueryable<T> source, string property)
    {
        return ApplyOrder<T>(source, property, "ThenByDescending");
    }
    static IOrderedQueryable<T> ApplyOrder<T>(IQueryable<T> source, string property, string methodName)
    {
        string[] props = property.Split('.');
        Type type = typeof(T);
        ParameterExpression arg = Expression.Parameter(type, "x");
        Expression expr = arg;
        foreach (string prop in props)
        {
            // use reflection (not ComponentModel) to mirror LINQ
            PropertyInfo pi = type.GetProperty(prop);
            expr = Expression.Property(expr, pi);
            type = pi.PropertyType;
        }
        Type delegateType = typeof(Func<,>).MakeGenericType(typeof(T), type);
        LambdaExpression lambda = Expression.Lambda(delegateType, expr, arg);

        object result = typeof(Queryable).GetMethods().Single(
                method => method.Name == methodName
                        && method.IsGenericMethodDefinition
                        && method.GetGenericArguments().Length == 2
                        && method.GetParameters().Length == 2)
                .MakeGenericMethod(typeof(T), type)
                .Invoke(null, new object[] { source, lambda });
        return (IOrderedQueryable<T>)result;
    }
}

Ответы:


309

изменение

public class LinqHelper

в

public static class LinqHelper

При создании метода расширения необходимо учитывать следующие моменты:

  1. Класс , который определяет метод расширения должен быть non-generic, staticиnon-nested
  2. Каждый метод расширения должен быть staticметодом
  3. Первый параметр метода расширения должен использовать thisключевое слово.

Если вы поместили класс в App_Code, то он обязательно должен иметь ключевое слово static в определении класса, но если вы поместите его в любую другую папку, тогда его можно использовать как обычный класс.
DT

1
В одном случае я использовал, public static class IQueryable<T> where T : MyBaseClassкоторый также генерирует эту ошибку. where T : MyBaseClassФраза принадлежит на отдельных методов без <T>на статическом классе.
Брон Дэвис

1
Но что, если класс частичный? Это решение не сработало для меня.
Fandango68

1
Спасибо, приятель, я попал на параметр "это"!
Роберто Гата,

1
Помните, что вы можете получить этот компилятор, если вы случайно конвертировали свой класс в метод расширения (согласно компилятору). Посмотрите этот ответ относительно статических методов и этот ответ относительно thisаргументов метода.
Мартен Бодевес

27

если вы не собираетесь использовать статические функции, просто избавьтесь от ключевого слова this в аргументах.


7
Некоторое время я чесал голову, пытаясь понять, почему Visual Studio подумала, что я пытаюсь сделать один из моих классов методом расширения. Оказывается, у меня thisзакопано ключевое слово в сигнатуре моего метода. Удаление это очистило ошибку.
Fütemire

20

Добавьте ключевое слово staticв объявление класса:

// this is a non-generic static class
public static class LinqHelper
{
}



15

Обходной путь для людей, которые испытывают ошибку, такую ​​как Натан:

Компилятор на лету, похоже, имеет проблему с этой ошибкой метода расширения ... добавление staticтакже не помогло мне.

Я хотел бы знать, что вызывает ошибку?

Но обходной путь - написать новый класс Extension (не вложенный) даже в том же файле и перестроить.

Понял, что этот поток получает достаточно просмотров, и его стоит использовать (ограниченное) решение, которое я нашел. Большинство людей, вероятно, пытались добавить «static» перед поиском решения для Google! и я не видел ни одного другого решения этой проблемы.


Я была такая же проблема. Я только что понял, что добавил статическую функцию в класс и забыл ее закомментировать. Так что это сделало мой класс статичным, и, следовательно, дал эту ошибку. Проверьте, есть ли в вашем классе статические объекты.
Махеш

1

Метод расширения должен быть внутри статического класса. Поэтому, пожалуйста, добавьте ваш метод расширения в статический класс.

так, например, это должно быть так

public static class myclass
    {
        public static Byte[] ToByteArray(this Stream stream)
        {
            Int32 length = stream.Length > Int32.MaxValue ? Int32.MaxValue : Convert.ToInt32(stream.Length);
            Byte[] buffer = new Byte[length];
            stream.Read(buffer, 0, length);
            return buffer;
        }

    }

Чем это отличается от ответов, данных годами ранее?
Maarten Bodewes

1

Попробуйте изменить его на статический класс и обратно. Это может разрешить визуальные жалобы студии, когда это ложное срабатывание.


0

Я столкнулся с подобной проблемой, я создал папку «foo» и создал «класс» внутри foo, а затем я получил вышеупомянутую ошибку. Одним из исправлений является добавление «static», как уже упоминалось ранее, к классу, который будет «public static class LinqHelper».

Я предполагаю, что когда вы создаете класс в папке foo, он рассматривает его как класс расширения, поэтому к нему применяется следующее правило:

1) Каждый метод расширения должен быть статическим

Временное решение Если вы не хотите статического. Мой обходной путь - создать класс прямо под пространством имен, а затем перетащить его в папку «foo».

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.