我已经无计可施了。我花了更多的时间让我的构建工作,而不是实际开发软件。我目前正在开发基于Tomcat 6的大型Java web应用程序。
目前的代码库大约是25k LOC(虽然这不是很有代表性,因为其中一些是为web服务自动生成的——但是它很大),我一直专门使用eclipse来做从调试,构建路径管理到构建的所有事情。在过去,eclipse总是绰绰有余,但我以前从未参与过这么大的项目。
我遇到的第一个问题是当我开始添加GWT内容时。它工作正常,但是构建过程有点粗糙:我必须手动编译,然后将js输出复制到适当的目录中,调试是一场噩梦(我意识到其中一些问题只是GWT和它的工作方式…)。现在,我遇到的问题是试图在Windows机器上进行项目(我一直在Mac上工作,并将不时地在Mac上工作)。Eclipse将Mac OS JVM库添加到构建路径中,当然,Windows无法找到这些库。
我问了一个关于在两个不同的环境中工作的问题,大多数答案都与使用Ant+Ivy或Maven这样的构建工具有关。我已经开始研究Maven,并试图让我的项目使用它,但它只是一个接一个令人头痛的问题,我甚至还没有开始尝试通过eclipse在Tomcat中实际执行/调试我的应用程序。对我来说,最令人沮丧的部分是,这些构建工具似乎异常复杂(公平地说,非常灵活),但是,至少在最初,当您试图弄清楚如何简单地编译Java文件时,生活变得更加困难。
说了这么多,有人有什么建议可以简化这个场景吗?依赖关系管理很好,但我不介意自己查找jar。我需要的最大的东西是,将只是离开我的方式,让我工作在我真正想要的工作,而不是花几个小时在构建系统使用的xml文件中调试我的错别字。
Thanks in advance
我同意。你不需要Maven;你不需要艾薇。从Ant开始。
这是一个相对通用的Ant build.xml。自定义
<?xml version="1.0" encoding="UTF-8"?>
<project name="xslt-converter" basedir="." default="package">
<property name="version" value="1.6"/>
<property name="haltonfailure" value="no"/>
<property name="out" value="out"/>
<property name="production.src" value="src"/>
<property name="production.lib" value="lib"/>
<property name="production.resources" value="config"/>
<property name="production.classes" value="${out}/production/${ant.project.name}"/>
<property name="test.src" value="test"/>
<property name="test.lib" value="lib"/>
<property name="test.resources" value="config"/>
<property name="test.classes" value="${out}/test/${ant.project.name}"/>
<property name="exploded" value="out/exploded/${ant.project.name}"/>
<property name="exploded.classes" value="${exploded}/WEB-INF/classes"/>
<property name="exploded.lib" value="${exploded}/WEB-INF/lib"/>
<property name="reports.out" value="${out}/reports"/>
<property name="junit.out" value="${reports.out}/junit"/>
<property name="testng.out" value="${reports.out}/testng"/>
<path id="production.class.path">
<pathelement location="${production.classes}"/>
<pathelement location="${production.resources}"/>
<fileset dir="${production.lib}">
<include name="**/*.jar"/>
<exclude name="**/junit*.jar"/>
<exclude name="**/*test*.jar"/>
</fileset>
</path>
<path id="test.class.path">
<path refid="production.class.path"/>
<pathelement location="${test.classes}"/>
<pathelement location="${test.resources}"/>
<fileset dir="${test.lib}">
<include name="**/junit*.jar"/>
<include name="**/*test*.jar"/>
</fileset>
</path>
<path id="testng.class.path">
<fileset dir="${test.lib}">
<include name="**/testng*.jar"/>
</fileset>
</path>
<available file="${out}" property="outputExists"/>
<target name="clean" description="remove all generated artifacts" if="outputExists">
<delete dir="${out}" includeEmptyDirs="true"/>
<delete dir="${reports.out}" includeEmptyDirs="true"/>
</target>
<target name="create" description="create the output directories" unless="outputExists">
<mkdir dir="${production.classes}"/>
<mkdir dir="${test.classes}"/>
<mkdir dir="${reports.out}"/>
<mkdir dir="${junit.out}"/>
<mkdir dir="${testng.out}"/>
<mkdir dir="${exploded.classes}"/>
<mkdir dir="${exploded.lib}"/>
</target>
<target name="compile" description="compile all .java source files" depends="create">
<!-- Debug output
<property name="production.class.path" refid="production.class.path"/>
<echo message="${production.class.path}"/>
-->
<javac srcdir="src" destdir="${out}/production/${ant.project.name}" debug="on" source="${version}">
<classpath refid="production.class.path"/>
<include name="**/*.java"/>
<exclude name="**/*Test.java"/>
</javac>
<javac srcdir="${test.src}" destdir="${out}/test/${ant.project.name}" debug="on" source="${version}">
<classpath refid="test.class.path"/>
<include name="**/*Test.java"/>
</javac>
</target>
<target name="junit-test" description="run all junit tests" depends="compile">
<!-- Debug output
<property name="test.class.path" refid="test.class.path"/>
<echo message="${test.class.path}"/>
-->
<junit printsummary="yes" haltonfailure="${haltonfailure}">
<classpath refid="test.class.path"/>
<formatter type="xml"/>
<batchtest fork="yes" todir="${junit.out}">
<fileset dir="${test.src}">
<include name="**/*Test.java"/>
</fileset>
</batchtest>
</junit>
<junitreport todir="${junit.out}">
<fileset dir="${junit.out}">
<include name="TEST-*.xml"/>
</fileset>
<report todir="${junit.out}" format="frames"/>
</junitreport>
</target>
<taskdef resource="testngtasks" classpathref="testng.class.path"/>
<target name="testng-test" description="run all testng tests" depends="compile">
<!-- Debug output
<property name="test.class.path" refid="test.class.path"/>
<echo message="${test.class.path}"/>
-->
<testng classpathref="test.class.path" outputDir="${testng.out}" haltOnFailure="${haltonfailure}" verbose="2" parallel="methods" threadcount="50">
<classfileset dir="${out}/test/${ant.project.name}" includes="**/*.class"/>
</testng>
</target>
<target name="exploded" description="create exploded deployment" depends="testng-test">
<copy todir="${exploded.classes}">
<fileset dir="${production.classes}"/>
</copy>
<copy todir="${exploded.lib}">
<fileset dir="${production.lib}"/>
</copy>
</target>
<target name="package" description="create package file" depends="exploded">
<jar destfile="${out}/${ant.project.name}.jar" basedir="${production.classes}" includes="**/*.class"/>
</target>
</project>
虽然maven乍一看可能有点吓人,但我认为它确实是一个非常好的工具。是的,它确实有缺点,但总的来说,我认为它是java的最佳选择之一。您真正需要了解的关于maven的"事情"是,一切都基于约定。如果您以传统的maven方式完成所有工作,那么您的maven通常非常小,并且一切都"正常工作"。还有很多入门信息可以向您展示什么是"maven约定"。(是的,你通常可以按照自己的方式做事,但通常不值得这样做,除非你被现有的项目困住了,很难重新组织)
除了Ant/Ivy或Maven XML之外,还有其他值得一试的选择。它们使用真正的编程语言而不是XML DSL。
- http://buildr.apache.org/(基于Ruby)
- http://www.gradle.org/index.php(基于Groovy)
仅在Eclipse中工作时可以使用的一种简化方法是让Eclipse负责编译。给定的项目可以有自己的简单Ant脚本,该脚本将bin
所需的任何内容打包到jar中。