【问题标题】:Optaplanner - NullPointerException when creating jar fileOptaplanner - 创建 jar 文件时出现 NullPointerException
【发布时间】:2017-09-13 13:50:27
【问题描述】:

我的程序在我的 IDE (IntelliJ) 上运行良好,但由于某种原因,当我尝试创建 jar 文件时,从终端运行程序时出现以下错误:

org.optaplanner.core.config.score.director.ScoreDirectorFactoryConfig.buildDroolsScoreDirectorFactory(ScoreDirectorFactoryConfig.java:461) 的线程“main”java.lang.NullPointerException 中的异常

org.optaplanner.core.config.score.director.ScoreDirectorFactoryConfig.buildScoreDirectorFactory(ScoreDirectorFactoryConfig.java:331)

org.optaplanner.core.config.solver.SolverConfig.buildSolver(SolverConfig.java:220)

org.optaplanner.core.impl.solver.AbstractSolverFactory.buildSolver(AbstractSolverFactory.java:57)

org.optaplanner.EmployeeRoster.main(EmployeeRoster.java:31)

这是我在 EmployeeRoster 中的第 31 行:

Solver solver = SolverFactory.createFromXmlResource(SOLVER_CONFIG_XML).buildSolver();

SOLVER_CONFIG_XML 是一个字符串,其中包含我的 XML 求解器配置路径, 看起来像这样

<?xml version="1.0" encoding="UTF-8"?>
<solver>
	<solutionClass>org.optaplanner.solver.Roster</solutionClass>
	<entityClass>org.optaplanner.domain.Assignment</entityClass>
	<scoreDirectorFactory>
		<scoreDrl>org/optaplanner/solver/employeeShiftsScoreRules.drl</scoreDrl>
	</scoreDirectorFactory>
	<localSearch>
		<termination>
			<secondsSpentLimit>5</secondsSpentLimit>
			<bestScoreLimit>0hard/0medium/0soft</bestScoreLimit>
		</termination>
		<!--<termination>
			<unimprovedStepCountLimit>5</unimprovedStepCountLimit>
		</termination>-->
		<acceptor>
			<entityTabuSize>7</entityTabuSize>
		</acceptor>
		<forager>
			<acceptedCountLimit>1000</acceptedCountLimit>
		</forager>
	</localSearch>
</solver>

如果应该相关的话,这也是我的 pom.xml 文件:

<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>org.btrg.dfb</groupId>
<artifactId>optaplanner</artifactId>
<version>0.0.1-SNAPSHOT</version>
<dependencies>
    <dependency>
        <groupId>org.optaplanner</groupId>
        <artifactId>optaplanner-core</artifactId>
        <version>7.3.0.Final</version>
    </dependency>

    <dependency>
        <groupId>com.thoughtworks.xstream</groupId>
        <artifactId>xstream</artifactId>
        <version>1.4.8</version>
    </dependency>

    <dependency>
        <groupId>com.google.code.gson</groupId>
        <artifactId>gson</artifactId>
        <version>2.8.1</version>
    </dependency>

    <dependency>
        <groupId>com.jolira</groupId>
        <artifactId>onejar-maven-plugin</artifactId>
        <version>1.4.4</version>
    </dependency>

</dependencies>

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
                <source>1.8</source>
                <target>1.8</target>
            </configuration>
        </plugin>

        <plugin>
            <!-- Build an executable JAR -->
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-assembly-plugin</artifactId>
            <configuration>
                <archive>
                    <manifest>
                        <addClasspath>true</addClasspath>
                        <mainClass>org.avalin.optaplanner.EmployeeRoster</mainClass>
                    </manifest>
                </archive>
                <descriptorRefs>
                    <descriptorRef>jar-with-dependencies</descriptorRef>
                </descriptorRefs>
            </configuration>
            <executions>
                <execution>
                    <id>make-my-jar-with-dependencies</id>
                    <phase>package</phase>
                    <goals>
                        <goal>single</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>

        <plugin>
            <groupId>com.jolira</groupId>
            <artifactId>onejar-maven-plugin</artifactId>
            <version>1.4.4</version>
            <executions>
                <execution>
                    <goals>
                        <goal>one-jar</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

我可能做错了什么?

【问题讨论】:

  • 我推断,当我到达 .buildSolver() 时,会出现空指针:/ 它确实在 ide 和终端中都找到了 XML 文件,但由于某种原因仅在终端中, buildSolver() 返回 null
  • 听起来类路径中缺少 kie 或 drools jar?
  • 谢谢杰弗里!你是对的,我只是将 drools-core.jar、drools-compiler.jar、kie-api.jar 和 kie-internal.jar 写到我的清单文件中。起初它不起作用,因为我使用逗号将它们彼此分隔,但是当我修复它时它完美地工作。你是救生员!
  • Related jira 链接到其他 2 个类似问题。

标签: java jar nullpointerexception optaplanner


【解决方案1】:

对我来说,问题必须处理我想如何运行 jar (java -jar),以及我如何构建 jar。当我从 6.4.0 升级到 optaplanner-core/7.4.1 时出现 NullPointerException。当我还在使用 6.4.0 时,此问题不存在。

例外:

java.lang.NullPointerException 在 org.kie.internal.io.ResourceFactory.newByteArrayResource(ResourceFactory.java:66) 在 org.drools.compiler.kie.builder.impl.AbstractKieModule.getResource(AbstractKieModule.java:299) 在 org.drools.compiler.kie.builder.impl.AbstractKieModule.addResourceToCompiler(AbstractKieModule.java:264) 在 org.drools.compiler.kie.builder.impl.AbstractKieModule.addResourceToCompiler(AbstractKieModule.java:259) 在 org.drools.compiler.kie.builder.impl.AbstractKieProject.buildKnowledgePackages(AbstractKieProject.java:243) 在 org.drools.compiler.kie.builder.impl.AbstractKieProject.verify(AbstractKieProject.java:74) 在 org.drools.compiler.kie.builder.impl.KieBuilderImpl.buildKieProject(KieBuilderImpl.java:250) 在 org.drools.compiler.kie.builder.impl.KieBuilderImpl.buildAll(KieBuilderImpl.java:218) 在 org.drools.compiler.kie.builder.impl.KieBuilderImpl.buildAll(KieBuilderImpl.java:176) 在 org.optaplanner.core.config.score.director.ScoreDirectorFactoryConfig.buildDroolsScoreDirectorFactory(ScoreDirectorFactoryConfig.java:503)

以下是我为解决 NullPointerException 并让应用程序运行而进行的临时解决方法

  1. 使用 IntelliJ 的 gradle 构建一个包含所有依赖项的大 jar。
  2. 解压jar包,例如解压到unzippedJar/目录。
  3. 修改unzippedJar/META-INF/kie.conf
  4. 重新压缩文件。
  5. 用 java 运行 jar。

步骤 1. 构建 jar

dependencies {
    ...
    compile group: 'org.optaplanner', name: 'optaplanner-core', version:'7.4.1.Final'
    compile group: 'org.optaplanner', name: 'optaplanner-benchmark', version:'7.4.1.Final'
    ...
}
task fatJar(type: Jar) {
    manifest {
        attributes 'Implementation-Title': 'Self contained jar with all dependencies',
        'Implementation-Version': version,
        'Main-Class': 'path.to.class.with.main.method'
    }
    baseName = 'fatJar'
    from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
    with jar
}

第 2 步。解压缩

unzip fatJar.jar -d unzippedJar/

第三步,修改kie.conf

“fatJar”任务扁平化了依赖关系 - jar 中允许重复的文件名。这导致了四个 unzippedJar/META-INF/kie.conf 文件,其中只使用了一个。无论使用哪个 kie.conf 文件,这都是我的最终 kie.conf。

    org.kie.api.internal.assembler.KieAssemblers = +org.optaplanner.core.impl.solver.kie.KieSolverAssemblerService
    org.kie.api.internal.assembler.KieAssemblers = org.kie.internal.services.KieAssemblersImpl
    org.kie.api.internal.runtime.KieRuntimes = org.kie.internal.services.KieRuntimesImpl
    org.kie.api.internal.weaver.KieWeavers = org.kie.internal.services.KieWeaversImpl
    org.kie.api.internal.runtime.beliefs.KieBeliefs = org.kie.internal.services.KieBeliefsImpl
    org.kie.api.io.KieResources = org.drools.core.io.impl.ResourceFactoryServiceImpl
    org.kie.api.marshalling.KieMarshallers = org.drools.core.marshalling.impl.MarshallerProviderImpl
    org.kie.api.concurrent.KieExecutors = org.drools.core.concurrent.ExecutorProviderImpl
    org.kie.api.KieServices = org.drools.compiler.kie.builder.impl.KieServicesImpl
    org.kie.internal.builder.KnowledgeBuilderFactoryService = org.drools.compiler.builder.impl.KnowledgeBuilderFactoryServiceImpl

第 4 步。重新安装

无论出于何种原因,指定 MANIFEST.MF 文件对我没有任何帮助,因此我将其省略了。

jar cf rejard.jar .

步骤 5. 运行 jar

java -cp rejard.jar path.to.class.with.main.method

【讨论】:

  • 这在 Optaplanner 7.8.0.Final 中对我有用,尽管我必须在 rejaring 时指定清单文件,如下所示:jar cfm rejard.jar META-INF/MANIFEST.MF .。然后我可以用java -jar rejard.jar 正常运行jar文件
【解决方案2】:

根据@Arturo W 的回答,我可以建议使用maven's assembly plugin 进行另一个修复 做一个胖罐子。

documentation中所述:

"如果两个或多个元素(例如,文件、文件集)选择不同的源 归档相同的文件时,只会归档其中一个源文件。 [...] 阶段的顺序如下:1) FileItem 2) FileSets 3) ModuleSet 4) DependencySet 和 5) Repository 元素。"

META-INF/kie.conforg.optaplanner.core 的存档中提供,因此 maven 将 optaplanner 的版本打包在您定义的自定义版本之上 存储库。

要包含它,我们会在解包过程中排除所有 META-INF/kie.conf 文件 并使用&lt;file&gt;&lt;/file&gt; 手动复制我们的版本 自定义描述符文件中的选项。

我们是这样做的:

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">
    <!-- ... -->
    <build>
        <plugins>
            <!-- ... -->

            <!-- fat jar -->
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
                <version>3.1.0</version>
                <configuration>
                    <descriptors>
                        <descriptor>src/assembly/distribution.xml</descriptor>
                    </descriptors>
                    <archive>
                        <manifest>
                            <addClasspath>true</addClasspath>
                            <mainClass>com.veryseriouscompany.veryseriousproject.app.Main</mainClass>
                        </manifest>
                    </archive>

                </configuration>
                <executions>
                    <execution>
                        <id>assemble-all</id>
                        <phase>package</phase>
                        <goals>
                            <goal>single</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

    <!-- ... -->

要修复包含kie.conf,我们必须编写一个自定义描述符, 在src/assembly/distribution.xml 中定义。我们从一个预定义的 描述符: jar-with-dependencies。 这是一个很好的起点。

src/assembly/distribution.xml

<assembly xmlns="http://maven.apache.org/ASSEMBLY/2.0.0"
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://maven.apache.org/ASSEMBLY/2.0.0 http://maven.apache.org/xsd/assembly-2.0.0.xsd">
    <id>epigno-jar-with-dependencies</id>
    <formats>
        <format>jar</format>
    </formats>
    <includeBaseDirectory>false</includeBaseDirectory>
    <dependencySets>
        <dependencySet>
            <outputDirectory>/</outputDirectory>
            <useProjectArtifact>true</useProjectArtifact>
            <unpack>true</unpack>
            <scope>runtime</scope>
            <!-- Exclude all META-INF/kie.conf during unpacking -->
            <unpackOptions>
                <excludes>
                    <exclude>META-INF/kie.conf</exclude>
                </excludes>
            </unpackOptions>
        </dependencySet>
    </dependencySets>
    <files>
        <file>
            <!-- Manually copy your custom kie configuration file from your repository. -->
            <!-- Please replace this path by whatever path is relevant for your project. -->
            <source>src/main/resources/META-INF/kie.conf</source>
            <outputDirectory>META-INF</outputDirectory>
        </file>
    </files>
</assembly>

完成后,运行

mvn package

希望这会将正确的kie.conf 打包到您的胖罐中!

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2011-11-14
    • 1970-01-01
    • 2013-01-03
    • 1970-01-01
    • 2012-05-21
    • 2012-03-23
    • 1970-01-01
    • 2018-02-08
    相关资源
    最近更新 更多