001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package org.eclipse.golo.runtime; 012 013import java.util.function.Function; 014 015/** 016 * A {@code Loader} just encapsulate a {@code ClassLoader}. The class implements 017 * {@code java.util.function.Function<String, Class<?>>} so that a loader can be directly mapped to a stream of the 018 * names of the classes to load. Moreover, the {@code ClassNotFoundException} is catched to return 019 * {@code null}, allowing a more "flowing" use. For instance: 020 * <pre class="listing"><code class="lang-java" data-lang="java"> 021 * Loader loader = Loader.forObject(this); 022 * Stream.of("java.lang.String", "gololang.Tuple", "foo.bar.Baz", "java.util.LinkedList") 023 * .map(loader) 024 * .filter(java.util.Objects::nonNull) 025 * .map(klass -> dealWithTheClass(klass)) 026 * </code></pre> 027 */ 028public final class Loader implements Function<String, Class<?>> { 029 private final ClassLoader loader; 030 031 Loader(ClassLoader loader) { 032 this.loader = loader; 033 } 034 035 public static Loader forClassLoader(ClassLoader loader) { 036 return new Loader(loader); 037 } 038 039 /** 040 * Create a loader using the {@code ClassLoader} of the current thread. 041 * 042 * @return a {@code Loader} 043 */ 044 public static Loader forCurrentThread() { 045 return new Loader(Thread.currentThread().getContextClassLoader()); 046 } 047 048 /** 049 * Create a loader using the {@code ClassLoader} of the given class 050 * 051 * @param klass the class whose {@code ClassLoader} to use 052 * @return a {@code Loader} 053 */ 054 public static Loader forClass(Class<?> klass) { 055 return new Loader(klass.getClassLoader()); 056 } 057 058 /** 059 * Create a loader using the {@code ClassLoader} of the class of the given object 060 * 061 * @param obj the object whose class {@code ClassLoader} to use 062 * @return a {@code Loader} 063 */ 064 static Loader forObject(Object obj) { 065 return forClass(obj.getClass()); 066 } 067 068 /** 069 * Load the given class. 070 * <p> 071 * This only delegates to the underlying class loader, but returns {@code null} instead 072 * of throwing a {@code ClassNotFoundException} exception. 073 * 074 * @return the class if load succeed, {@code null} otherwise. 075 */ 076 public Class<?> load(String name) { 077 try { 078 return loader.loadClass(name); 079 } catch (ClassNotFoundException e) { 080 return null; 081 } 082 } 083 084 /** 085 * Just delegate to {@link #load(java.lang.String)} to implement {@code Function}. 086 * 087 * @see #load(java.lang.String) 088 */ 089 @Override 090 public Class<?> apply(String name) { 091 return load(name); 092 } 093} 094 095