C # 4.0 «динамический» не устанавливает аргументы ref / out - PullRequest
11 голосов
/ 19 марта 2010

Я экспериментирую с DynamicObject. Я пытаюсь установить значения аргументов ref / out, как показано в приведенном ниже коде. Однако я не могу правильно установить значения i и j в Main() (даже если они правильно установлены в TryInvokeMember()). Кто-нибудь знает, как вызвать объект DynamicObject с аргументами ref / out и получить значения, установленные внутри метода?

class Program
{
    static void Main(string[] args)
    {
        dynamic proxy = new Proxy(new Target());
        int i = 10;
        int j = 20;
        proxy.Wrap(ref i, ref j);
        Console.WriteLine(i + ":" + j); // Print "10:20" while expect "20:10"
    }
}

class Proxy : DynamicObject
{
    private readonly Target target;

    public Proxy(Target target)
    {
        this.target = target;
    }

    public override bool TryInvokeMember(InvokeMemberBinder binder, object[] args, out object result)
    {
        int i = (int) args[0];
        int j = (int) args[1];
        target.Swap(ref i, ref j);
        args[0] = i;
        args[1] = j;
        result = null;
        return true;
    }
}

class Target
{
    public void Swap(ref int i, ref int j)
    {
        int tmp = i;
        i = j;
        j = tmp;
    }
}

Обновление 7/15: Microsoft утверждает, что исправила проблему в следующем выпуске .NET http://connect.microsoft.com/VisualStudio/feedback/details/543101/net-4-0s-dynamicobject-doesn-t-set-ref-out-arguments

Обновление от 9.08.2012: Протестировано на VS.NET 2012 с обеими версиями .NET 4.0 и 4.5, подтвердите: оно уже исправлено.

Ответы [ 3 ]

5 голосов
/ 19 марта 2010

Похоже, что может быть ошибкой - вероятно, в DynamicObject. Если вы добавите метод Wrap к Proxy, например:

public void Wrap(ref int x, ref int y)
{
    target.Swap(ref x, ref y);
}

Тогда, хотя это все еще вызывается динамически (т. Е. Код в Main остается неизменным), код работает ... так что, по крайней мере, общий уровень "как работает динамический объект" поддерживает передачу по ссылке.

Я подозреваю, что если действительно действительно ошибка в DLR, то может быть слишком поздно исправлять для .NET 4 - но в любом случае стоит сообщить о Connect , поэтому исправлено в пакете обновления. В качестве альтернативы, если это преднамеренное ограничение / ограничение, оно должно быть четко задокументировано в MSDN (насколько я могу судить, на данный момент это не так).

5 голосов
/ 23 марта 2010

Это не ошибка. Как уже было сказано, DynamicObject не поддерживает параметры ref и out в TryInvokeMember. Все, что передается этому методу, обрабатывается «по значению». Вкратце, метод TryInvokeMember просто игнорирует эти ключевые слова, и именно поэтому ваш метод не работает.

Если вы последуете совету Джона Скита и создадите свой собственный метод Wrap в классе, унаследованном от DynamicObject, это будет немного другой сценарий. Рабочий процесс выглядит следующим образом: при вызове метода для DynamicObject связыватель времени выполнения C # сначала ищет метод в самом классе. Если он может найти его, он вызывает этот метод. На данный момент информация о параметрах «ref» и «out» все еще сохраняется. Если он не может найти такой метод, он вызывает метод TryInvokeMember и просто выбрасывает информацию о ключевых словах «ref» и «out» и начинает воспринимать все как «по значению». Помните, что DynamicObject должен поддерживать взаимодействие с другим языком, который может не иметь всех функций C #.

Правда, информация о "ref" и "out" теперь отсутствует в документации. Я добавлю его в следующее обновление документации.

2 голосов
/ 19 марта 2010

Короче говоря, DynamicObject не поддерживает передачу по ссылке, поэтому то, что вы хотите сделать, напрямую невозможно.

...