Могу ли я использовать params в делегатах Action или Func?

Когда я пытаюсь использовать params в делегате Action...

private Action<string, params object[]> WriteToLogCallBack;

Я получил эту ошибку времени проектирования:

Недопустимые маркеры 'params' в объявлении члена класса, структуры или интерфейса

Любая помощь!

5 ответов

Решение

Как насчет этого обходного пути?

private Action<string, object[]> writeToLogCallBack;
public void WriteToLogCallBack(string s, params object[] args)
{
  if(writeToLogCallBack!=null)
    writeToLogCallBack(s,args);
}

Или вы можете определить свой собственный тип делегата:

delegate void LogAction(string s, params object[] args);

Параметры типа Variadic невозможны в C#.

Вот почему есть много объявлений для Action<...>, Func<...>, а также Tuple<...>, например. Это была бы интересная особенность. C++0x имеет их.

Вы можете попробовать это. Он допускает любое количество аргументов, и вы получите ошибку времени компиляции, если передадите неверное число или тип аргументов.

public delegate T ParamsAction<T>(params object[] oArgs);

public static T LogAction<T>(string s, ParamsAction<T> oCallback)
{
    Log(s);
    T result = oCallback();
    return T;
}

Foo foo = LogAction<Foo>("Hello world.", aoArgs => GetFoo(1,"",'',1.1));

Ты можешь использовать params в фактическом объявлении делегата, но не в типе одного. Общими параметрами для Action являются только типы, а не фактические аргументы, передаваемые при вызове делегата. params это не тип, это ключевое слово.

Я сделал небольшое расширение для вышеупомянутого кода от Брайана, чтобы показать, как обернуть несколько вызовов методов.
Я использую это, чтобы обернуть несколько методов, которые содержат вызовы базы данных, в одну транзакцию.
Спасибо Брайану:-)
(Вы можете запустить следующее в LINQPad для тестирования)

//Wrapper code
public delegate void MyAction(params object[] objArgs);
public static void RunActions(params MyAction[] actnArgs)
{
    Console.WriteLine("WrapperBefore: Begin transaction code\n");
    actnArgs.ToList().ForEach( actn =>  actn() );
    Console.WriteLine("\nWrapperAfter: Commit transaction code");
}

//Methods being called
public void Hash  (string s, int i, int j)  => Console.WriteLine("   Hash-method call: " + s + "###" + i.ToString() + j.ToString());
public void Slash (int i, string s)         => Console.WriteLine("   Slash-method call: " + i.ToString()+ @"////" + s);

//Actual calling code
void Main()
{  
  RunActions( objArgs => Hash("One", 2, 1)
             ,objArgs => Slash(3, "four")   );
}

//Resulting output: 
// 
//  WrapperBefore: Begin transaction code
//  
//  Hash-method call: One###21
//  Slash-method call: 3////four
//  
//  WrapperAfter: Commit transaction code  
Другие вопросы по тегам