java中的扩展方法

扩展方法

如果想要对现有的类添加新的方法,但是又不太可能去修改原有类的时候,我们就可以通过扩展方法进行扩展(当然,也有其他的方式可以达到目的),java原生是不支持扩展方法的,熟悉其他语言的同学可能知道,例如:C#、go、kotlin等都是原生支持扩展方法的,那如果想在java中也要实现扩展方法,比如:“hello world”.print();实现 这样的功能,该如何做呢?

Manifold介绍

Manifold是一个java的编译插件,其中包含了很多的功能,其中扩展方法就是其中之一,更多的功能介绍可以参考官网以及源代码

接下来我们简单地通过代码的方式看看manifold是如何实现扩展方法的。

  • 首先,添加依赖,参考pom.xml文件如下

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>groupId</groupId>
    <artifactId>untitled</artifactId>
    <version>1.0-SNAPSHOT</version>

    <properties>
        <maven.compiler.source>17</maven.compiler.source>
        <maven.compiler.target>17</maven.compiler.target>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <manifold.version>2022.1.33</manifold.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>systems.manifold</groupId>
            <artifactId>manifold-ext-rt</artifactId>
            <version>${manifold.version}</version>
        </dependency>
    </dependencies>




</project>
  • 定义一个类,例如:StringExtension

package extensions.java.lang.String;


import manifold.ext.rt.api.Extension;
import manifold.ext.rt.api.This;

@Extension
public class StringExtension {
    public static void print(@This String thiz) {
        System.out.println(thiz);
    }
    @Extension
    public static String lineSeparator() {
        return System.lineSeparator();
    }

    @Extension
    public static String sayHello(String name) {
        return "hello " + name;
    }
}

特别说明下:扩展的包的定义是有一定的限制,所有的扩展类必须是在extensions包下面定义,后面加上要扩展的类的包,例如上代码:extensions.java.lang.String,在java9以及后续的版本中,我们还可以加上自己的模块名称,比如:com.test,那么扩展的包名就可以是:com.test.extensions.java.lang.String,在java8中extensions必须是根。

扩展方法所在的类必须添加@Extension注解,扩展方法必须是静态的公共方法,而且第一个参数为:@This+扩展的类型+参数名称,如果是静态的扩展方法那么需要在静态方法上面加@Extension注解,参数不再需要@This 进行限定,定义完成之后我们可以通过如下的方式进行调用了

"hello world".print();System.out.println(String.sayHello("sherman"));

当然,为了更棒的体验,还是要在idea中装一个插件Manifold

image.png

其他类型扩展

比如,我想针对Iterable<T>这个类型进行扩展,可以参考如下的代码:

packageextensions.java.lang.Iterable;importmanifold.ext.rt.api.Extension;importmanifold.ext.rt.api.Self;importmanifold.ext.rt.api.This;importjava.util.NoSuchElementException;importjava.util.function.Predicate;@ExtensionpublicclassCollectionExtension{publicstatic<T>Tfirst(@ThisIterable<T> thiz,Predicate<T> predicate){for(T element : thiz){if(predicate.test(element)){return element;}}thrownewNoSuchElementException();}}
List<String> list =newArrayList<>();
list.add("one");
list.add("two");
list.add("three");Predicate<String> getFirst = s ->Objects.equals(s,"one");String first = list.first(getFirst);System.out.println(first);

Manifold已经为我们提供了一些扩展

  • Collections

定义在manifold-io模块中,扩展了以下的类

  • java.io.BufferedReader

  • java.io.File

  • java.io.InputStream

  • java.io.OutputStream

  • java.io.Reader

  • java.io.Writer

  • Text

定义在manifold-text模块中,扩展了以下的类

  • java.lang.CharSequence

  • java.lang.String

  • I/O

定义在manifold-io模块中,扩展了以下的类

  • java.io.BufferedReader

  • java.io.File

  • java.io.InputStream

  • java.io.OutputStream

  • java.io.Reader

  • java.io.Writer

  • Web/JSON

定义在manifold-json模块中,扩展了以下的类

  • manifold.rt.api.Bindings

我们也可以将自己的扩展以包的形式提供出去,具体注意事项可以参考官方的文档。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值