Как узнать классы во время выполнения в пути к классам, который реализует определенный интерфейс?
ServiceLoader хорошо подходит (я думаю, я его не использовал), но мне нужно сделать это в Java 1.5.
Как узнать классы во время выполнения в пути к классам, который реализует определенный интерфейс?
ServiceLoader хорошо подходит (я думаю, я его не использовал), но мне нужно сделать это в Java 1.5.
В этом нет ничего встроенного в Java 1.5. Я сам реализовал это; это не слишком сложно. Однако, когда мы переходим на Java 6, мне придется заменить вызовы на мою реализацию вызовами ServiceLoader
. Я мог бы определить небольшой мост между приложением и загрузчиком, но я использую его только в нескольких местах, и сама оболочка будет хорошим кандидатом для ServiceLoader.
Это основная идея:
public <S> Iterable<S> load(Class<S> ifc) throws Exception {
ClassLoader ldr = Thread.currentThread().getContextClassLoader();
Enumeration<URL> e = ldr.getResources("META-INF/services/" + ifc.getName());
Collection<S> services = new ArrayList<S>();
while (e.hasMoreElements()) {
URL url = e.nextElement();
InputStream is = url.openStream();
try {
BufferedReader r = new BufferedReader(new InputStreamReader(is, "UTF-8"));
while (true) {
String line = r.readLine();
if (line == null)
break;
int comment = line.indexOf('#');
if (comment >= 0)
line = line.substring(0, comment);
String name = line.trim();
if (name.length() == 0)
continue;
Class<?> clz = Class.forName(name, true, ldr);
Class<? extends S> impl = clz.asSubclass(ifc);
Constructor<? extends S> ctor = impl.getConstructor();
S svc = ctor.newInstance();
services.add(svc);
}
}
finally {
is.close();
}
}
return services;
}
Лучшая обработка исключений остается в качестве упражнения для читателя. Кроме того, метод может быть параметризован, чтобы принять ClassLoader выбора вызывающего.
javax.imageio.spi.ServiceRegistry
эквивалентен предыдущим версиям Java. Он доступен с Java 1.4.
Он не похож на общий класс утилиты, но он есть. Он даже немного более мощный, чем ServiceLoader
, поскольку он позволяет контролировать порядок возвращенных поставщиков и прямой доступ к реестру.
См. http://docs.oracle.com/javase/7/docs/api/index.html?javax/imageio/spi/ServiceRegistry.html
ServiceLoader довольно простой и используется (неформально) в JDK начиная с версии 1.3. ServiceLoader только что сделал его гражданином первого класса. Он просто ищет файл ресурсов, названный для вашего интерфейса, который в основном связан в каталоге META-INF библиотеки jar.
Этот файл содержит имя загружаемого класса.
Итак, у вас будет файл с именем:
META-INF/услуги/com.example.your.interface
и внутри него находится одна строка: com.you.your.interfaceImpl.
Вместо ServiceLoader мне нравится Netbeans Lookup. Он работает с 1,5 (и, возможно, 1,4).
Из коробки он делает то же самое, что и ServiceLoader, и это тривиально использовать. Но он предлагает гораздо большую гибкость.
Здесь ссылка: http://openide.netbeans.org/lookup/
Вот статья о ServiceLoader, но в ней упоминается Netbeans Lookup внизу: http://weblogs.java.net/blog/timboudreau/archive/2008/08/simple_dependen.html
К сожалению,
В Java 1.5 ничего не встроено для этого...
является лишь частью истины.
Существует нестандартный sun.misc.Service
.
http://www.docjar.com/docs/api/sun/misc/Service.html
Остерегайтесь, это не входит в стандартный J2SE API!
Это нестандартная часть Sun JDK.
Поэтому вы не можете полагаться на это, если используете, скажем, JRockit
.
Это старый вопрос, но другой вариант - использовать Аннотации уровня пакета. См. Мой ответ для: Найти классы Java, реализующие интерфейс
Аннотации уровня пакета - это аннотации, которые содержатся в классах package-info.java.
JAXB использует это вместо сервисных загрузчиков. Я также считаю его более гибким, чем сервисный загрузчик.
Нет надежного способа узнать, какие классы находятся в пути к классам. Согласно его документации, ServiceLoader полагается на внешние файлы, чтобы рассказать, какие классы загружать; вы можете сделать то же самое. Основная идея состоит в том, чтобы загрузить файл с именем класса (ов) для загрузки, а затем использовать отражение для его создания/их.
Считаете ли вы, что используете инфраструктуру OSGI?