我希望在对象列表中实现一个功能,因为我会在c#中使用扩展方法。

就像这样:

List<DataObject> list;
// ... List initialization.
list.getData(id);

在Java中怎么做呢?


当前回答

在Java中没有扩展方法,但你可以通过以下方式来获得它:

你通过下面的例子定义字符串的“echo”方法,

@Extension
public class MyStringExtension {
    public static void echo(@This String thiz) {
        System.out.println(thiz);
    }
}

在那之后,你可以在任何地方对字符串使用这个方法(echo),

"Hello World".echo();   //prints "Hello World"
"Welcome".echo();       //prints "Welcome"
String name = "Jonn";
name.echo();            //prints "John"


当然,你也可以有这样的参数,

@Extension
public class MyStringExtension {
    public static void echo(@This String thiz, String suffix) {
        System.out.println(thiz + " " + suffix);
    }
}

像这样使用,

"Hello World".echo("programming");   //prints "Hello World programming"
"Welcome".echo("2021");              //prints "Welcome 2021"
String name = "John";
name.echo("Conor");                  //prints "John Conor"

你也可以看看这个样品,歧管样品

其他回答

Project Lombok提供了一个注释@ExtensionMethod,可用于实现您所要求的功能。

另一个选择是使用来自google-guava库的ForwardingXXX类。

Java没有这样的特性。 相反,你可以创建列表实现的常规子类或创建匿名内部类:

List<String> list = new ArrayList<String>() {
   public String getData() {
       return ""; // add your implementation here. 
   }
};

问题是调用这个方法。你可以“就地”做:

new ArrayList<String>() {
   public String getData() {
       return ""; // add your implementation here. 
   }
}.getData();

扩展方法不仅仅是静态方法,也不仅仅是方便的语法糖,实际上它们是非常强大的工具。主要的事情是基于不同泛型的参数实例化重写不同方法的能力。这类似于Haskell的类型类,事实上,它们看起来像是在c#中支持c#的单子(即LINQ)。即使放弃LINQ语法,我仍然不知道如何在Java中实现类似的接口。

而且我认为在Java中实现它们是不可能的,因为Java的泛型参数的类型清除语义。

可以使用面向对象的装饰器设计模式。在Java的标准库中使用这种模式的一个例子是DataOutputStream。

下面是一些增强List功能的代码:

public class ListDecorator<E> implements List<E>
{
    public final List<E> wrapee;

    public ListDecorator(List<E> wrapee)
    {
        this.wrapee = wrapee;
    }

    // implementation of all the list's methods here...

    public <R> ListDecorator<R> map(Transform<E,R> transformer)
    {
        ArrayList<R> result = new ArrayList<R>(size());
        for (E element : this)
        {
            R transformed = transformer.transform(element);
            result.add(transformed);
        }
        return new ListDecorator<R>(result);
    }
}

附注:我是Kotlin的忠实粉丝。它有扩展方法,也运行在JVM上。