Делегирование вызовов методов с использованием переменного числа аргументов - PullRequest
0 голосов
/ 21 января 2010

Этот вопрос возник в ходе моего программирования работы; это не имеет отношения к текущей задаче, но мне все еще любопытно, есть ли у кого-нибудь ответ.

В Java 1.5 и выше вы можете иметь сигнатуру метода, используя переменное число аргументов, с синтаксисом многоточия:

public void run(Foo... foos) {
  if (foos != null) {
   for (Foo foo: foos) { //converted from array notation using autoboxing
    foo.bar();
   }
  }
}

Предположим, я хочу выполнить некоторую операцию с каждым foo в списке foos, а затем делегировать этот вызов некоторому полю в моем объекте, сохраняя тот же API. Как мне это сделать? То, что я хочу, это:

public void run(Foo... foos) {
  MyFoo[] myFoos = null;
  if (foos != null) {
   myFoos = new MyFoo[foos.length];
   for (int i = 0; i < foos.length; i++) {
    myFoos[i] = wrap(foos[i]);
   }
  }
  run(myFoos);
}

public void run(MyFoo... myFoos) {
  if (myFoos!= null) {
   for (MyFoo myFoo: myFoos) { //converted from array notation using autoboxing
    myFoo.bar();
   }
  }
}

Это не компилируется. Как я могу это сделать (передав переменное число MyFoo в метод run (MyFoo ...))?

Ответы [ 4 ]

2 голосов
/ 21 января 2010

Это то, что вы хотите?

public class VarArgsTest {

    public static class Foo {}

    public static class MyFoo extends Foo {
        public MyFoo(Foo foo) {}
    }

    public static void func(Foo... foos) {
        MyFoo [] myfoos = new MyFoo[foos.length];
        int i=0;
        for (Foo foo : foos) {
            myfoos[i++] = new MyFoo(foo);
        }
        func(myfoos);
    }

    public static void func(MyFoo... myfoos) {
        for (MyFoo m : myfoos) {
            System.out.println(m);
        }
    }

    public static void main(String [] args) throws Exception {
        func(new Foo(), new Foo(), new Foo());
    }

}
1 голос
/ 21 января 2010

Я попробовал это и не получил ошибку компиляции. Какую фактическую ошибку вы видите? Вот код, который я использовал. Возможно, я сделал что-то другое:

public class MultipleArgs {
public static void main(String [] args){
    run(new Foo("foo1"), new Foo("foo2"), new Foo("foo3"));
}
public static void run(Foo... foos){
    MyFoo[] myFoos = null;
    if (foos != null) {
    myFoos = new MyFoo[foos.length];
       for (int i = 0; i < foos.length; i++) {
        myFoos[i] = wrap(foos[i]);
       }
      }
      run(myFoos);
}
public static void run(MyFoo... myFoos){
     if (myFoos!= null) {
           for (MyFoo myFoo: myFoos) {
            myFoo.bar();
           }
          }

}
private static class Foo {
    public final String s;
    public Foo(String s){
        this.s = s;
    }
    @Override
    public String toString(){
        return s;
    }
}
private static class MyFoo{
    private final String s;
    public MyFoo(String s){
        this.s = s;
    }
    public void bar(){
        System.out.println(s);
    }
    @Override
    public String toString(){
        return s;
    }
}
private static MyFoo wrap(Foo foo){
    return new MyFoo(foo.s); 
}

}

0 голосов
/ 21 января 2010

Это не отвечает на ваш вопрос; это случайно, но вам не нужен нулевой тест. Вот доказательство:

public class VarargsTest extends TestCase {

    public void testVarargs() throws Exception {
        assertEquals(0, fn());
    }

    private int fn(String...strings) {
        return strings.length;
    }
}

Если метод вызывается без аргументов, список varargs представляет собой пустой массив, а не ноль.

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

0 голосов
/ 21 января 2010

использовать отражения java.

...