Javaソースファイルを生成するフレームワークを探しています。
次のAPIのようなもの:
X clazz = Something.createClass("package name", "class name");
clazz.addSuperInterface("interface name");
clazz.addMethod("method name", returnType, argumentTypes, ...);
File targetDir = ...;
clazz.generate(targetDir);
次に、Javaソースファイルがターゲットディレクトリのサブディレクトリにあります。
誰もがそのようなフレームワークを知っていますか?
[〜#〜] edit [〜#〜]:
[〜#〜] solutions [〜#〜]
回答に基づいて2つの回答を投稿しました... CodeModelを使用 および Eclipse JDTを使用 .
私はソリューションで CodeModel を使用しました:-)
Sunは、APIを使用してJavaソースファイルを生成するためのCodeModelと呼ばれるAPIを提供します。
それを取得する最も簡単な方法は、JAXB 2 RIの一部としてです。XJCスキーマからJavaへのジェネレーターは、CodeModelを使用してJavaソースを生成し、XJC jarの一部です。 CodeModelにのみ使用できます。
http://codemodel.Java.net/ から入手してください
CodeModelで解決策が見つかりました
ありがとう、 スカッフマン 。
たとえば、次のコードでは:
JCodeModel cm = new JCodeModel();
JDefinedClass dc = cm._class("foo.Bar");
JMethod m = dc.method(0, int.class, "foo");
m.body()._return(JExpr.lit(5));
File file = new File("./target/classes");
file.mkdirs();
cm.build(file);
私はこの出力を得ることができます:
package foo;
public class Bar {
int foo() {
return 5;
}
}
Eclipse JDTのASTで解決策が見つかりました
ありがとう、 Giles 。
たとえば、次のコードでは:
AST ast = AST.newAST(AST.JLS3);
CompilationUnit cu = ast.newCompilationUnit();
PackageDeclaration p1 = ast.newPackageDeclaration();
p1.setName(ast.newSimpleName("foo"));
cu.setPackage(p1);
ImportDeclaration id = ast.newImportDeclaration();
id.setName(ast.newName(new String[] { "Java", "util", "Set" }));
cu.imports().add(id);
TypeDeclaration td = ast.newTypeDeclaration();
td.setName(ast.newSimpleName("Foo"));
TypeParameter tp = ast.newTypeParameter();
tp.setName(ast.newSimpleName("X"));
td.typeParameters().add(tp);
cu.types().add(td);
MethodDeclaration md = ast.newMethodDeclaration();
td.bodyDeclarations().add(md);
Block block = ast.newBlock();
md.setBody(block);
MethodInvocation mi = ast.newMethodInvocation();
mi.setName(ast.newSimpleName("x"));
ExpressionStatement e = ast.newExpressionStatement(mi);
block.statements().add(e);
System.out.println(cu);
私はこの出力を得ることができます:
package foo;
import Java.util.Set;
class Foo<X> {
void MISSING(){
x();
}
}
Roaster( https://github.com/forge/roaster )を使用してコード生成を行うことができます。
以下に例を示します。
JavaClassSource source = Roaster.create(JavaClassSource.class);
source.setName("MyClass").setPublic();
source.addMethod().setName("testMethod").setPrivate().setBody("return null;")
.setReturnType(String.class).addAnnotation(MyAnnotation.class);
System.out.println(source);
次の出力が表示されます。
public class MyClass {
private String testMethod() {
return null;
}
}
別の選択肢はEclipse JDTのASTです。これは、ソースコードを生成するだけでなく、任意のJavaソースコードを書き換える必要がある場合に適しています。 Eclipseから独立して使用されます)。
Eclipse JET プロジェクトは、ソース生成に使用できます。 APIがあなたが説明したものとまったく同じだとは思いませんが、プロジェクトがJavaソース生成を行っていると聞いたときはいつも、JETまたは自家製ツールを使用しています。
ライブラリについては知りませんが、必要なのは汎用テンプレートエンジンだけです。 それらの束 があります、私は個人的に FreeMarker で良い経験をしました
「sourcegen」と呼ばれる理論上のDSLに非常によく似たものを作成しましたが、技術的には私が書いたORMのutilプロジェクトではありません。 DSLは次のようになります。
@Test
public void testTwoMethods() {
GClass gc = new GClass("foo.bar.Foo");
GMethod hello = gc.getMethod("hello");
hello.arguments("String foo");
hello.setBody("return 'Hi' + foo;");
GMethod goodbye = gc.getMethod("goodbye");
goodbye.arguments("String foo");
goodbye.setBody("return 'Bye' + foo;");
Assert.assertEquals(
Join.lines(new Object[] {
"package foo.bar;",
"",
"public class Foo {",
"",
" public void hello(String foo) {",
" return \"Hi\" + foo;",
" }",
"",
" public void goodbye(String foo) {",
" return \"Bye\" + foo;",
" }",
"",
"}",
"" }),
gc.toCode());
}
https://github.com/stephenh/joist/blob/master/util/src/test/Java/joist/sourcegen/GClassTest.Java
また、パラメータ/戻り値型のFQCNの「インポートの自動整理」、このcodegenの実行で変更されなかった古いファイルの自動整理、内部クラスのインデントの適切化など、いくつかの適切な処理を行います。
生成されたコードは、コードの残りの部分と同じように、警告(未使用のインポートなど)なしで、見た目がきれいでなければなりません。生成されたコードの多くは読むのがugいです...恐ろしいです。
とにかく、多くのドキュメントはありませんが、APIは非常にシンプル/直感的だと思います。 Mavenリポジトリは here 誰かが興味があるなら。
本当にソースが必要な場合、ソースを生成するものは知りません。ただし、 [〜#〜] asm [〜#〜] または [〜#〜] cglib [〜#〜] を使用して、.classファイルを直接作成できます。
これらからソースを生成できるかもしれませんが、私はそれらをバイトコードの生成にのみ使用しました。
新しいプロジェクト write-it-once があります。テンプレートベースのコードジェネレータ。 Groovy を使用してカスタムテンプレートを記述し、Javaリフレクションに応じてファイルを生成します。これは、任意のファイルを生成する最も簡単な方法です。 AspectJファイル、JPAアノテーションに基づくSQL、enumに基づく挿入/更新など。
テンプレートの例:
package ${cls.package.name};
public class ${cls.shortName}Builder {
public static ${cls.name}Builder builder() {
return new ${cls.name}Builder();
}
<% for(field in cls.fields) {%>
private ${field.type.name} ${field.name};
<% } %>
<% for(field in cls.fields) {%>
public ${cls.name}Builder ${field.name}(${field.type.name} ${field.name}) {
this.${field.name} = ${field.name};
return this;
}
<% } %>
public ${cls.name} build() {
final ${cls.name} data = new ${cls.name}();
<% for(field in cls.fields) {%>
data.${field.setter.name}(this.${field.name});
<% } %>
return data;
}
}
StringTemplate もあります。 ANTLRの作成者によるもので、非常に強力です。
私は自分でモックジェネレータツールを作成していました。 Sunのフォーマットガイドラインに従う必要がある場合でも、非常に簡単な作業です。インターネットであなたの目標に合ったものを見つけた後、それを速くするコードを完成させるに違いない。
基本的に自分でAPIの概要を説明しました。今すぐ実際のコードを入力してください!
それは本当にあなたがやろうとしていることに依存します。コード生成自体はトピックです。特定のユースケースがなければ、速度コード生成/テンプレートライブラリを調べることをお勧めします。また、コード生成をオフラインで行う場合は、ArgoUMLのようなものを使用してUMLダイアグラム/オブジェクトモデルからJava code。
例:1 /
private JFieldVar generatedField;
2 /
String className = "class name";
/* package name */
JPackage jp = jCodeModel._package("package name ");
/* class name */
JDefinedClass jclass = jp._class(className);
/* add comment */
JDocComment jDocComment = jclass.javadoc();
jDocComment.add("By AUTOMAT D.I.T tools : " + new Date() +" => " + className);
// génération des getter & setter & attribues
// create attribue
this.generatedField = jclass.field(JMod.PRIVATE, Integer.class)
, "attribue name ");
// getter
JMethod getter = jclass.method(JMod.PUBLIC, Integer.class)
, "attribue name ");
getter.body()._return(this.generatedField);
// setter
JMethod setter = jclass.method(JMod.PUBLIC, Integer.class)
,"attribue name ");
// create setter paramétre
JVar setParam = setter.param(getTypeDetailsForCodeModel(Integer.class,"param name");
// affectation ( this.param = setParam )
setter.body().assign(JExpr._this().ref(this.generatedField), setParam);
jCodeModel.build(new File("path c://javaSrc//"));
以下は、面白そうなJSON-to-POJOプロジェクトです。