拡張メソッドは、非ジェネリック静的クラスで定義する必要があります


213

エラーが発生します:

拡張メソッドは、非ジェネリック静的クラスで定義する必要があります

行:

public class LinqHelper

Mark Gavellsコードに基づくヘルパークラスを次に示します。このエラーが何を意味するのか、私は本当に混乱しています。

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-genericstaticそしてnon-nested
  2. すべての拡張メソッドでなければならないstatic方法
  3. 拡張メソッドの最初のパラメーターはthisキーワードを使用する必要があります。

App_Codeにクラスを配置した場合は、クラス定義にstaticキーワードが含まれるようにバインドされますが、他のフォルダーに配置した場合は、通常のクラスとして使用しても問題ありません。
DT

1
あるケースでは、public static class IQueryable<T> where T : MyBaseClassこれもこのエラーを生成するものを使用していました。このwhere T : MyBaseClass句は<T>、静的クラスではなく、個々のメソッドに属しています。
Bron Davies

1
しかし、クラスが部分的である場合はどうなりますか?この解決策は私にはうまくいきませんでした。
Fandango68

1
バディのおかげで、私は "this"パラメータに引っかかりました!
Roberto Gata

1
(コンパイラによると)クラスを誤って拡張メソッドに変換した場合、このコンパイラが表示される可能性があることに注意してください。静的メソッドに関してはこの回答を、メソッド引数に関してはこの回答を参照してください。this
Maarten Bodewes

26

静的関数を使用しない場合は、引数の「this」キーワードを削除してください。


6
なぜVisual Studioが自分のクラスの1つを拡張メソッドにしようとしているのかを理解しようとする間、頭を悩ませていました。私判明は持っていたthis私のメソッドのシグネチャの上に埋もれキーワードを。削除するとエラーは解消されました。
フューテミア

20

キーワードstaticをクラス宣言に追加します。

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

16

変更してみてください

public class LinqHelper

 public static class LinqHelper


15

Nathanのようなバグを経験している人のための回避策:

オンザフライコンパイラーは、この拡張メソッドエラーに問題があるようです...追加staticしても役に立ちませんでした。

バグの原因を知りたいのですが?

しかし、回避策は、同じファイル内でも(ネストされていない)新しい拡張クラスを記述して、再構築することです。

このスレッドは、私が見つけた(制限された)ソリューションを渡す価値がある十分なビューを取得していることを理解しました。ほとんどの人はおそらく、解決策をググる前に「静的」を追加しようとしました!そして、私はこの回避策の修正を他のどこにも見ませんでした。


同じ問題がありました。クラス内に静的関数を追加したことに気づき、コメントアウトするのを忘れました。そのため、クラスが静的になり、このエラーが発生していました。クラスに静的オブジェクトがあるかどうかを確認します。
Mahesh、

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内に「クラス」を作成すると、前述のエラーが発生します。1つの修正は、「パブリック静的クラスLinqHelper」になるクラスに前述の「静的」を追加することです。

私の想定では、fooフォルダー内にクラスを作成すると、それは拡張クラスと見なされるため、次のような規則が適用されます。

1)すべての拡張メソッドは静的メソッドでなければなりません

静的にしたくない場合。私の回避策は、名前空間の直下にクラスを作成し、それを「foo」フォルダーにドラッグすることでした。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.