官方的supercsv推土机示例不适用于Play - Play 2.1.1,Java

时间:2013-09-06 08:49:25

标签: java jar playframework-2.0 dozer supercsv

更新

现在我发现了一些非常奇怪的东西,当我直接编译测试类时它会起作用 进入“super-csv-dozer-2.1.0.jar”,其结构与svn repo相同。 但是只要我使用自己的包装就行不通。我总是使用整个路径 像这样:myPackage.csv.SurveyResponse.class 我错过了什么?为什么它在普通的Java项目中工作,但在Play项目中却没有? 我也在eclipse之外的另一个目录中尝试过它,并且没有任何结账 可能会混淆路径的svn回购。我尝试的另一件事是将Writing.class放入不同的包但没有帮助。

原始生活:

我试图在Play项目中使用带有Dozer的SuperCSV。 官方的supercsv示例在单独的Java项目中正常工作。 但是当我把代码和所需的SuperCSV Jars放在一个新创建的Play项目中时,我总是如此 在这行代码中获取SurveyResponse.class的ClassNotfoundException: beanWriter.configureBeanMapping(myPackage.csv.SurveyResponse.class,FIELD_MAPPING);

这是我项目结构的截图: http://oi44.tinypic.com/mrqp7s.jpg

我确保所有JARS都可用,我将它们放在/ lib文件夹中, 它们在eclipse中可用,编译期间不会出现错误。 我调试了代码并找到了SurveyResponse.class并初始化了beanWriter。 所以Play必须在后台做一些魔术来触发这个bug。 在后台可能会发生什么可能触发这种奇怪的行为? 我可以尝试做些什么来解决这个问题?

我对示例所做的更改,因此适用于Play: 我使用与官方supercsv示例完全相同的代码。 我做的唯一更改是删除Writing.main(..)方法并设置方法Writing.writeWithDozerCsvBeanWriter() 和Writing.partialWriteWithCsvDozerBeanWriter()为public,因此可以从Application控制器访问它。 当然,我更改了所有类中的包名称以打包myPackage.test;

示例链接: 在评论中看到,由于声誉低,我无法添加超过2个链接。

控制器:

public class Application extends Controller {
    public static Result index() throws Exception  {
        Writing.writeWithDozerCsvBeanWriter();
        Writing.partialWriteWithCsvDozerBeanWriter();
        return ok(index.render("Your new application is ready."));
    } 
}

在写入类中触发错误的代码:

ICsvDozerBeanWriter beanWriter = null;
    try {
        beanWriter = new CsvDozerBeanWriter(new FileWriter("target/writeWithCsvDozerBeanWriter.csv"),
            CsvPreference.STANDARD_PREFERENCE);

        // configure the mapping from the fields to the CSV columns
        //Here the exception occures:
        beanWriter.configureBeanMapping(myPackage.csv.SurveyResponse.class, FIELD_MAPPING);

堆栈追踪:

play.api.Application$$anon$1: Execution exception[[MappingException: java.lang.ClassNotFoundException: myPackage.test.SurveyResponse]]
    at play.api.Application$class.handleError(Application.scala:289) ~[play_2.10.jar:2.1.1]
    at play.api.DefaultApplication.handleError(Application.scala:383) [play_2.10.jar:2.1.1]
    at play.core.server.netty.PlayDefaultUpstreamHandler$$anon$2$$anonfun$handle$1.apply(PlayDefaultUpstreamHandler.scala:144) [play_2.10.jar:2.1.1]
    at play.core.server.netty.PlayDefaultUpstreamHandler$$anon$2$$anonfun$handle$1.apply(PlayDefaultUpstreamHandler.scala:140) [play_2.10.jar:2.1.1]
    at play.api.libs.concurrent.PlayPromise$$anonfun$extend1$1.apply(Promise.scala:113) [play_2.10.jar:2.1.1]
    at play.api.libs.concurrent.PlayPromise$$anonfun$extend1$1.apply(Promise.scala:113) [play_2.10.jar:2.1.1]
    at play.api.libs.concurrent.PlayPromise$$anonfun$extend$1$$anonfun$apply$1.apply(Promise.scala:104) [play_2.10.jar:2.1.1]
    at scala.concurrent.impl.Future$PromiseCompletingRunnable.liftedTree1$1(Future.scala:24) [scala-library.jar:na]
    at scala.concurrent.impl.Future$PromiseCompletingRunnable.run(Future.scala:24) [scala-library.jar:na]
    at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(Unknown Source) [na:1.6.0_37]
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source) [na:1.6.0_37]
    at java.lang.Thread.run(Unknown Source) [na:1.6.0_37]
org.dozer.MappingException: java.lang.ClassNotFoundException: myPackage.test.SurveyResponse
    at org.dozer.util.MappingUtils.throwMappingException(MappingUtils.java:82) ~[dozer-5.4.0.jar:na]
    at org.dozer.util.DefaultClassLoader.loadClass(DefaultClassLoader.java:38) ~[dozer-5.4.0.jar:na]
    at org.dozer.util.MappingUtils.loadClass(MappingUtils.java:224) ~[dozer-5.4.0.jar:na]
    at org.dozer.loader.DozerBuilder$MappingBuilder.classA(DozerBuilder.java:129) ~[dozer-5.4.0.jar:na]
    at org.dozer.loader.api.BeanMappingBuilder.mapping(BeanMappingBuilder.java:72) ~[dozer-5.4.0.jar:na]
    at org.dozer.loader.api.BeanMappingBuilder.mapping(BeanMappingBuilder.java:67) ~[dozer-5.4.0.jar:na]
    at org.supercsv.io.dozer.CsvDozerBeanWriter$MappingBuilder.configure(CsvDozerBeanWriter.java:178) ~[super-csv-dozer-2.1.0.jar:na]
    at org.dozer.loader.api.BeanMappingBuilder.build(BeanMappingBuilder.java:42) ~[dozer-5.4.0.jar:na]
    at org.dozer.DozerBeanMapper.addMapping(DozerBeanMapper.java:258) ~[dozer-5.4.0.jar:na]
    at org.supercsv.io.dozer.CsvDozerBeanWriter.configureBeanMapping(CsvDozerBeanWriter.java:91) ~[super-csv-dozer-2.1.0.jar:na]
    at myPackage.test.Writing.writeWithDozerCsvBeanWriter(Writing.java:88) ~[na:na]
    at controllers.Application.index(Application.java:12) ~[na:na]
    at Routes$$anonfun$routes$1$$anonfun$applyOrElse$1$$anonfun$apply$1.apply(routes_routing.scala:49) ~[na:na]
    at Routes$$anonfun$routes$1$$anonfun$applyOrElse$1$$anonfun$apply$1.apply(routes_routing.scala:49) ~[na:na]
    at play.core.Router$HandlerInvoker$$anon$6$$anon$2.invocation(Router.scala:164) ~[play_2.10.jar:2.1.1]
    at play.core.Router$Routes$$anon$1.invocation(Router.scala:345) ~[play_2.10.jar:2.1.1]
    at play.core.j.JavaAction$$anon$1.call(JavaAction.scala:31) ~[play_2.10.jar:2.1.1]
    at play.core.j.JavaAction$$anon$2.apply(JavaAction.scala:74) ~[play_2.10.jar:2.1.1]
    at play.core.j.JavaAction$$anon$2.apply(JavaAction.scala:73) ~[play_2.10.jar:2.1.1]
    at play.libs.F$Promise$PromiseActor.onReceive(F.java:420) ~[play_2.10.jar:2.1.1]
    at akka.actor.UntypedActor$$anonfun$receive$1.applyOrElse(UntypedActor.scala:159) ~[akka-actor_2.10.jar:na]
    at akka.actor.ActorCell.receiveMessage(ActorCell.scala:425) ~[akka-actor_2.10.jar:na]
    at akka.actor.ActorCell.invoke(ActorCell.scala:386) ~[akka-actor_2.10.jar:na]
    at akka.dispatch.Mailbox.processMailbox(Mailbox.scala:230) ~[akka-actor_2.10.jar:na]
    at akka.dispatch.Mailbox.run(Mailbox.scala:212) ~[akka-actor_2.10.jar:na]
    at akka.dispatch.ForkJoinExecutorConfigurator$MailboxExecutionTask.exec(AbstractDispatcher.scala:502) ~[akka-actor_2.10.jar:na]
    at scala.concurrent.forkjoin.ForkJoinTask.doExec(ForkJoinTask.java:262) ~[scala-library.jar:na]
    at scala.concurrent.forkjoin.ForkJoinPool$WorkQueue.runTask(ForkJoinPool.java:975) ~[scala-library.jar:na]
    at scala.concurrent.forkjoin.ForkJoinPool.runWorker(ForkJoinPool.java:1478) ~[scala-library.jar:na]
    at scala.concurrent.forkjoin.ForkJoinWorkerThread.run(ForkJoinWorkerThread.java:104) ~[scala-library.jar:na]
Caused by: java.lang.ClassNotFoundException: myPackage.test.SurveyResponse
    at java.net.URLClassLoader$1.run(Unknown Source) ~[na:1.6.0_37]
    at java.security.AccessController.doPrivileged(Native Method) ~[na:1.6.0_37]
    at java.net.URLClassLoader.findClass(Unknown Source) ~[na:1.6.0_37]
    at java.lang.ClassLoader.loadClass(Unknown Source) ~[na:1.6.0_37]
    at java.lang.ClassLoader.loadClass(Unknown Source) ~[na:1.6.0_37]
    at sbt.PlayCommands$$anonfun$53$$anonfun$55$$anon$2.loadClass(PlayCommands.scala:535) ~[na:na]
    at java.lang.Class.forName0(Native Method) ~[na:1.6.0_37]
    at java.lang.Class.forName(Unknown Source) ~[na:1.6.0_37]
    at org.apache.commons.lang3.ClassUtils.getClass(ClassUtils.java:823) ~[commons-lang3.jar:3.1]
    at org.apache.commons.lang3.ClassUtils.getClass(ClassUtils.java:889) ~[commons-lang3.jar:3.1]
    at org.apache.commons.lang3.ClassUtils.getClass(ClassUtils.java:872) ~[commons-lang3.jar:3.1]
    at org.dozer.util.DefaultClassLoader.loadClass(DefaultClassLoader.java:36) ~[dozer-5.4.0.jar:na]
    ... 28 common frames omitted

额外信息: 我仍然无法解决问题,但这里有一些额外的信息来缩小问题范围: 当我删除configureBeanMapping方法调用时,csv头成功写入文件(参见下面的代码)。 当我在一个填充的SurveyResponse对象上执行System.out.println()时,SurveyResponse类也正在工作(参见下面的代码)。 所以它不是包或类名的问题。

仅编写标题的修改代码:

//...more code
myPackage.csv.SurveyResponse response3 = new myPackage.csv.SurveyResponse(42, false, Arrays.asList(new Answer(1, null), new Answer(2,
    "Carl Sagan"), new Answer(3, "Star Wars")));
final List<myPackage.csv.SurveyResponse> surveyResponses = Arrays.asList(response1, response2, response3);

ICsvDozerBeanWriter beanWriter = null;
try {
    beanWriter = new CsvDozerBeanWriter(new FileWriter("target/writeWithCsvDozerBeanWriter.csv"),
        CsvPreference.STANDARD_PREFERENCE);

    // configure the mapping from the fields to the CSV columns
    //beanWriter.configureBeanMapping(myPackage.csv.SurveyResponse.class, FIELD_MAPPING);

    //Prints the value 42 successfully
    System.out.println(response3.getAge());

    // write the header
    beanWriter.writeHeader("age", "consentGiven", "questionNo1", "answer1", "questionNo2", "answer2","questionNo3", "answer3");

//...more code

2 个答案:

答案 0 :(得分:0)

根据我从错误中理解,Play正在尝试加载错误的类。看看这一行:

java.lang.ClassNotFoundException: org.supercsv.mock.dozer.SurveyResponse

我认为SurverResponse是您自己的项目类之一,而不是SuperCSV的一部分。尝试使用完整路径作为前缀,例如:

beanWriter.configureBeanMapping(whateverpackage.models.SurveyResponse.class, FIELD_MAPPING);

答案 1 :(得分:0)

SurveyResponseAnswer是<{1}}工件的 test 类 - 所以它们没有打包在发行版中(即它们不在Maven中)或者在SourceForge上的zip文件中)。

您可以查看测试来源online或结帐SVN repo(听起来您已经这样做过)并将其复制到Play项目中。

这两个文件都在以下目录中:

super-csv-dozer

我有点好奇为什么在尝试在代码中使用supercsv/super-csv-dozer/src/test/java/org/supercsv/mock/dozer/ 时没有收到编译错误。我的猜测是你已经检查了SurveyResponse项目并在你的IDE中打开,你的项目正在找到它。