- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我正在尝试使用 Spring 3.1 和嵌入式 Jetty 8 服务器创建一个没有任何 XML 配置的简单 webapp。
但是,我很难让 Jetty 识别我的 Spring WebApplicationInitializer 接口(interface)的实现。
项目结构:
src
+- main
+- java
| +- JettyServer.java
| +- Initializer.java
|
+- webapp
+- web.xml (objective is to remove this - see below).
上面的Initializer类是WebApplicationInitializer的简单实现:
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import org.springframework.web.WebApplicationInitializer;
public class Initializer implements WebApplicationInitializer {
@Override
public void onStartup(ServletContext servletContext) throws ServletException {
System.out.println("onStartup");
}
}
同样,JettyServer 是嵌入式 Jetty 服务器的简单实现:
import org.eclipse.jetty.annotations.AnnotationConfiguration;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.webapp.Configuration;
import org.eclipse.jetty.webapp.WebAppContext;
public class JettyServer {
public static void main(String[] args) throws Exception {
Server server = new Server(8080);
WebAppContext webAppContext = new WebAppContext();
webAppContext.setResourceBase("src/main/webapp");
webAppContext.setContextPath("/");
webAppContext.setConfigurations(new Configuration[] { new AnnotationConfiguration() });
webAppContext.setParentLoaderPriority(true);
server.setHandler(webAppContext);
server.start();
server.join();
}
}
我的理解是,Jetty 在启动时会使用 AnnotationConfiguration 来扫描 ServletContainerInitializer 的注释实现;它应该找到 Initializer 并将其连接到...
但是,当我(从 Eclipse 中)启动 Jetty 服务器时,我在命令行上看到以下内容:
2012-11-04 16:59:04.552:INFO:oejs.Server:jetty-8.1.7.v20120910
2012-11-04 16:59:05.046:INFO:/:No Spring WebApplicationInitializer types detected on classpath
2012-11-04 16:59:05.046:INFO:oejsh.ContextHandler:started o.e.j.w.WebAppContext{/,file:/Users/duncan/Coding/spring-mvc-embedded-jetty-test/src/main/webapp/}
2012-11-04 16:59:05.117:INFO:oejs.AbstractConnector:Started SelectChannelConnector@0.0.0.0:8080
重要的是:
No Spring WebApplicationInitializer types detected on classpath
请注意,src/main/java 在 Eclipse 中被定义为源文件夹,因此应该位于类路径中。另请注意,Dynamic Web Module Facet 设置为 3.0。
我确信有一个简单的解释,但我很难只见树木不见森林!我怀疑关键在于以下行:
...
webAppContext.setResourceBase("src/main/webapp");
...
这对于使用 web.xml 的 2.5 servlet 是有意义的(见下文),但是使用 AnnotationConfiguration 时应该是什么?
注意:如果我将配置更改为以下内容,一切都会正确启动:
...
webAppContext.setConfigurations(new Configuration[] { new WebXmlConfiguration() });
...
在这种情况下,它会在 src/main/webapp 下找到 web.xml 并使用它通过 DispatcherServlet 和 < em>AnnotationConfigWebApplicationContext 以通常的方式(完全绕过上面的 WebApplicationInitializer 实现)。
这感觉很像一个类路径问题,但我很难理解 Jetty 如何将自己与 WebApplicationInitializer 的实现联系起来 - 任何建议都将不胜感激!
有关信息,我正在使用以下内容:
Spring 3.1.1 jetty 8.1.7 STS 3.1.0
最佳答案
问题是Jetty的AnnotationConfiguration
类不扫描classpath下的非jar资源(WEB-INF/classes下除外)。
如果我注册 AnnotationConfiguration
的子类,它会找到我的 WebApplicationInitializer
,它会覆盖 configure(WebAppContext)
以另外扫描主机类路径到容器和 web-inf 位置。
大多数子类(可悲的是)从父类复制粘贴。它包括:
parseHostClassPath
); parseHostClassPath
方法主要是从AnnotationConfiguration
的 parseWebInfClasses
; getHostClassPathResource
方法来自类加载器(至少对我来说,这是我的文件 urleclipse 中的类路径)。我使用的 Jetty (8.1.7.v20120910) 和 Spring (3.1.2_RELEASE) 版本略有不同,但我想同样的解决方案也可以。
编辑:我在 github 中创建了一个工作示例项目,并进行了一些修改(下面的代码在 Eclipse 中可以正常工作,但在阴影 jar 中运行时却不行)-https://github.com/steveliles/jetty-embedded-spring-mvc-noxml
在 OP 的 JettyServer 类中,必要的更改将第 15 行替换为:
webAppContext.setConfigurations (new Configuration []
{
new AnnotationConfiguration()
{
@Override
public void configure(WebAppContext context) throws Exception
{
boolean metadataComplete = context.getMetaData().isMetaDataComplete();
context.addDecorator(new AnnotationDecorator(context));
AnnotationParser parser = null;
if (!metadataComplete)
{
if (context.getServletContext().getEffectiveMajorVersion() >= 3 || context.isConfigurationDiscovered())
{
parser = createAnnotationParser();
parser.registerAnnotationHandler("javax.servlet.annotation.WebServlet", new WebServletAnnotationHandler(context));
parser.registerAnnotationHandler("javax.servlet.annotation.WebFilter", new WebFilterAnnotationHandler(context));
parser.registerAnnotationHandler("javax.servlet.annotation.WebListener", new WebListenerAnnotationHandler(context));
}
}
List<ServletContainerInitializer> nonExcludedInitializers = getNonExcludedInitializers(context);
parser = registerServletContainerInitializerAnnotationHandlers(context, parser, nonExcludedInitializers);
if (parser != null)
{
parseContainerPath(context, parser);
parseWebInfClasses(context, parser);
parseWebInfLib (context, parser);
parseHostClassPath(context, parser);
}
}
private void parseHostClassPath(final WebAppContext context, AnnotationParser parser) throws Exception
{
clearAnnotationList(parser.getAnnotationHandlers());
Resource resource = getHostClassPathResource(getClass().getClassLoader());
if (resource == null)
return;
parser.parse(resource, new ClassNameResolver()
{
public boolean isExcluded (String name)
{
if (context.isSystemClass(name)) return true;
if (context.isServerClass(name)) return false;
return false;
}
public boolean shouldOverride (String name)
{
//looking at webapp classpath, found already-parsed class of same name - did it come from system or duplicate in webapp?
if (context.isParentLoaderPriority())
return false;
return true;
}
});
//TODO - where to set the annotations discovered from WEB-INF/classes?
List<DiscoveredAnnotation> annotations = new ArrayList<DiscoveredAnnotation>();
gatherAnnotations(annotations, parser.getAnnotationHandlers());
context.getMetaData().addDiscoveredAnnotations (annotations);
}
private Resource getHostClassPathResource(ClassLoader loader) throws IOException
{
if (loader instanceof URLClassLoader)
{
URL[] urls = ((URLClassLoader)loader).getURLs();
for (URL url : urls)
if (url.getProtocol().startsWith("file"))
return Resource.newResource(url);
}
return null;
}
},
});
更新:Jetty 8.1.8 引入了与上述代码不兼容的内部更改。对于 8.1.8,以下似乎有效:
webAppContext.setConfigurations (new Configuration []
{
// This is necessary because Jetty out-of-the-box does not scan
// the classpath of your project in Eclipse, so it doesn't find
// your WebAppInitializer.
new AnnotationConfiguration()
{
@Override
public void configure(WebAppContext context) throws Exception {
boolean metadataComplete = context.getMetaData().isMetaDataComplete();
context.addDecorator(new AnnotationDecorator(context));
//Even if metadata is complete, we still need to scan for ServletContainerInitializers - if there are any
AnnotationParser parser = null;
if (!metadataComplete)
{
//If metadata isn't complete, if this is a servlet 3 webapp or isConfigDiscovered is true, we need to search for annotations
if (context.getServletContext().getEffectiveMajorVersion() >= 3 || context.isConfigurationDiscovered())
{
_discoverableAnnotationHandlers.add(new WebServletAnnotationHandler(context));
_discoverableAnnotationHandlers.add(new WebFilterAnnotationHandler(context));
_discoverableAnnotationHandlers.add(new WebListenerAnnotationHandler(context));
}
}
//Regardless of metadata, if there are any ServletContainerInitializers with @HandlesTypes, then we need to scan all the
//classes so we can call their onStartup() methods correctly
createServletContainerInitializerAnnotationHandlers(context, getNonExcludedInitializers(context));
if (!_discoverableAnnotationHandlers.isEmpty() || _classInheritanceHandler != null || !_containerInitializerAnnotationHandlers.isEmpty())
{
parser = createAnnotationParser();
parse(context, parser);
for (DiscoverableAnnotationHandler h:_discoverableAnnotationHandlers)
context.getMetaData().addDiscoveredAnnotations(((AbstractDiscoverableAnnotationHandler)h).getAnnotationList());
}
}
private void parse(final WebAppContext context, AnnotationParser parser) throws Exception
{
List<Resource> _resources = getResources(getClass().getClassLoader());
for (Resource _resource : _resources)
{
if (_resource == null)
return;
parser.clearHandlers();
for (DiscoverableAnnotationHandler h:_discoverableAnnotationHandlers)
{
if (h instanceof AbstractDiscoverableAnnotationHandler)
((AbstractDiscoverableAnnotationHandler)h).setResource(null); //
}
parser.registerHandlers(_discoverableAnnotationHandlers);
parser.registerHandler(_classInheritanceHandler);
parser.registerHandlers(_containerInitializerAnnotationHandlers);
parser.parse(_resource,
new ClassNameResolver()
{
public boolean isExcluded (String name)
{
if (context.isSystemClass(name)) return true;
if (context.isServerClass(name)) return false;
return false;
}
public boolean shouldOverride (String name)
{
//looking at webapp classpath, found already-parsed class of same name - did it come from system or duplicate in webapp?
if (context.isParentLoaderPriority())
return false;
return true;
}
});
}
}
private List<Resource> getResources(ClassLoader aLoader) throws IOException
{
if (aLoader instanceof URLClassLoader)
{
List<Resource> _result = new ArrayList<Resource>();
URL[] _urls = ((URLClassLoader)aLoader).getURLs();
for (URL _url : _urls)
_result.add(Resource.newResource(_url));
return _result;
}
return Collections.emptyList();
}
}
});
关于java - Spring 3.1 WebApplicationInitializer & Embedded Jetty 8 AnnotationConfiguration,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13222071/
我需要一个如何使用 WebApplicationInitializer 的简单示例。我已经看到了大量示例实现,但是我在哪里声明应该使用我的个人 MyVeryIndividualWebApplica
我正在使用 maven 创建一个 spring mvc 应用程序。它使用 WebApplicationInitializer 进行初始化。现在我正在尝试添加 hibernate 依赖项。当我添加它时,
我正在使用 Spring 为学校项目构建 BlogPost。我可以访问我的 index.jsp,但是当我映射到其他 Controller 时,我得到一个 404。当我在我的 WebApplicatio
我正在使用 Spring MVC 和 Maven 创建一个 webapp 项目。我也是使用 Spring 注释驱动配置的新手。 我想为我的容器创建一个包含所有 SpringMVC 配置和 Servle
我正在使用 Spring MVC 和 ZK 框架创建一个网络应用程序。我有一个包含 spring 和 zk 的所有基本配置的根项目 (jar),以及一个调用初始化程序的 Web 应用程序项目 (war
我有一个基于 Spring 3.1 的应用程序托管在 Tomcat 7.x(最新版本)下。该应用程序仅使用 Java 配置(无 web.xml,无 Spring XML 配置)。所有单元测试都通过了,
我有一个使用 Maven 构建的 Web 应用程序(war)。我正在使用旧的 struts 应用程序(别无选择)。我正在使用 AnnotationConfigApplicationContext 和一
我有一个包含各种 Spring 依赖项的项目。 'org.springframework:spring-core:4.3.2.RELEASE', 'org.springframewor
Spring Security 的文档指出,为了使用 Java Config,我们可以扩展 AbstractSecurityWebApplicationInitializer 类,该类将为 Sprin
我需要配置 2 个 servlet:一个用于常规 http 请求,另一个是用于 Java web-socket 的 Atmosphere servlet。 这是我的 WebApplicationIni
WebApplicationInitializers 类的 web.xml 的等价物是什么? package com.concretepage.config; import org.sp
我编写了下面的类以编程方式注册多个 servlet,但它不起作用,任何人都可以帮助我。 public class appIntializer implements WebApplicationInit
我在 TomCat 9 上启动我的 Spring Boot 应用程序时遇到问题。在 Intellij IDE 中,应用程序启动并按预期工作。 当我尝试通过 TomCat Web 应用程序管理器启动应用
我正在尝试使用构建配置文件(开发、测试或生产)构建我的项目。我有一个 Spring Web 应用程序,我正在使用 WebApplicationInitializer 的实现来初始化它。 public
这个问题在这里已经有了答案: How does Tomcat exactly bootstrap the app without web.xml? (2 个答案) 关闭 7 年前。 Spring W
目前我有一个 Web 应用程序,我们在其中使用 web.xml 来配置应用程序。 web.xml 有welcome-file-list。 ... home.html
我的 Eclipse 项目突然不再正确部署。我无法追踪到我对环境所做的任何特定更改。 我已经对多个源代码控制项目进行了测试,它们的行为方式都相同: May 01, 2013 12:00:45 PM o
我正在使用 Spring 4.0.7 我做了一个关于通过 JavaConfig 配置 Spring MVC 的研究。 实际上直到昨天我才看到使用这两个选项的两种配置 扩展 AbstractAnnota
我有一个 CRUD spring 应用程序,由于依赖关系而无法运行。我需要用“war”包装的那部分,但是拿不到。 我的 pom.xml 是: 4.0.0 com.sprhib
我想使用 读取 application.properties @EnableConfigurationProperties and @ConfigurationProperties. 我可以使用以下代
我是一名优秀的程序员,十分优秀!