日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

如何以及为什么使用Spoon分析,生成和转换Java代码

發布時間:2023/12/3 java 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 如何以及为什么使用Spoon分析,生成和转换Java代码 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

Spoon是分析,生成和轉換Java代碼的工具。

在本文中,我們將看到通過使用以編程方式處理代碼的技術可以實現什么。 我認為這些技術不是很廣為人知或使用,這很遺憾,因為它們可能非常有用。 誰知道,即使您不想使用Spoon甚至不處理Java代碼,而是使用C#,Python,Kotlin或其他語言,某些想法對于您當前的項目也可能有用。 讓我們學習如何以更智能的方式編程。

Spoon具有一些與JavaParser重疊的功能, JavaParser是我貢獻的框架。 對于某些任務,Spoon可能是更好的選擇,而對于另一些任務,JavaParser具有明顯的優勢。 稍后,我們將深入探討這些工具之間的差異。

本文與隨附的存儲庫與所有代碼配對: ftomassetti / spoon-examples

使用代碼處理技術可以實現什么?

勺子和一般的代碼處理工具可用于:

  • 代碼分析
    • 計算源代碼指標,例如找出多少類具有比一定數量的方法更多的類
  • 代碼生成
    • 以編程方式生成重復代碼。
  • 代碼轉換
    • 自動重構,例如在構造函數中指定的字段中轉換幾種方法中使用的參數

這三大家族與我們與代碼交互的方式大致不同:

  • 在代碼分析中,代碼是我們用來產生非代碼輸出的輸入
  • 在代碼生成中,我們使用一些通常不是代碼的輸入,或者使用與我們輸出的語言相同的代碼。 輸出是代碼
  • 在代碼轉換中,相同的代碼庫是輸入和輸出

設置湯匙

要設置湯匙,您需要提供:

  • 要分析的代碼
  • 所有依賴關系(當然還有依賴關系的依賴關系)

利用此信息,Spoon可以構建代碼模型。 在該模型上,您可以執行相當高級的分析。 這與JavaParser的工作方式不同。 如果需要,在JavaParser中,您可以構建代碼的輕量級模型,而無需考慮依賴關系。 當您沒有可用的依賴項或需要執行簡單快速的操作時,這將很有用。 您也可以通過啟用符號解析來進行更高級的分析,但這是可選的,并且在僅某些依賴項可用時也可以使用。

我喜歡Spoon的一件事是支持從Maven進行配置。 我認為這是一個非常有用的功能。 但是,我只想得到Gradle的支持。

在我們的示例中,我們不使用Maven配置,我們僅指定一個包含代碼的目錄。 在我們的例子中,我們正在檢查JavaParser的核心模塊,該模塊的依賴項為零,因此我們無需指定任何JAR即可構建代碼模型。

fun main(args: Array<String>) {val launcher = Launcher()launcher.addInputResource("codebases/jp/javaparser-core/src/main/java")launcher.environment.noClasspath = trueval model = launcher.buildModel()... }

現在我們有了一個模型,讓我們看看如何使用它。

順便說一句,示例是用Kotlin編寫的,因為它是一種簡潔明了的語言,我認為它非常適合教程。 你同意嗎?

使用Spoon執行代碼分析

讓我們開始使用20多種方法打印類列表:

fun examineClassesWithManyMethods(ctModel: CtModel, threshold: Int = 20) {val classes = ctModel.filterChildren<CtClass<*>> {it is CtClass<*> && it.methods.size > threshold}.list<CtClass<*>>()printTitle("Classes with more than $threshold methods")printList(classes.asSequence().sortedByDescending { it.methods.size }.map { "${it.qualifiedName} (${it.methods.size})"})println() }fun main(args: Array<String>) {val launcher = Launcher()launcher.addInputResource("codebases/jp/javaparser-core/src/main/java")launcher.environment.noClasspath = trueval model = launcher.buildModel()examineClassesWithManyMethods(model) }

在本示例中,我們在主要函數中設置模型,然后在inspectClassesWithManyMethods中 ,按方法數量過濾類,然后使用幾個實用程序函數來打印這些類的列表(printTitle , printList )。

運行此代碼,我們獲得以下輸出:

===================================== | Classes with more than 20 methods | =====================================* com.github.javaparser.ast.expr.Expression (141)* com.github.javaparser.printer.PrettyPrintVisitor (105)* com.github.javaparser.ast.visitor.EqualsVisitor (100)* com.github.javaparser.ast.visitor.NoCommentEqualsVisitor (98)* com.github.javaparser.ast.visitor.CloneVisitor (95)* com.github.javaparser.ast.visitor.GenericVisitorWithDefaults (94)* com.github.javaparser.ast.visitor.ModifierVisitor (94)* com.github.javaparser.ast.visitor.VoidVisitorWithDefaults (94)* com.github.javaparser.ast.visitor.HashCodeVisitor (93)* com.github.javaparser.ast.visitor.NoCommentHashCodeVisitor (93)* com.github.javaparser.ast.visitor.ObjectIdentityEqualsVisitor (93)* com.github.javaparser.ast.visitor.ObjectIdentityHashCodeVisitor (93)* com.github.javaparser.ast.stmt.Statement (92)* com.github.javaparser.ast.visitor.GenericListVisitorAdapter (92)* com.github.javaparser.ast.visitor.GenericVisitorAdapter (92)* com.github.javaparser.ast.visitor.VoidVisitorAdapter (92)* com.github.javaparser.ast.Node (62)* com.github.javaparser.ast.NodeList (62)* com.github.javaparser.ast.type.Type (55)* com.github.javaparser.ast.body.BodyDeclaration (50)* com.github.javaparser.ast.modules.ModuleDirective (44)* com.github.javaparser.ast.CompilationUnit (44)* com.github.javaparser.JavaParser (39)* com.github.javaparser.resolution.types.ResolvedReferenceType (37)* com.github.javaparser.utils.SourceRoot (34)* com.github.javaparser.ast.body.CallableDeclaration (29)* com.github.javaparser.ast.body.MethodDeclaration (28)* com.github.javaparser.printer.PrettyPrinterConfiguration (27)* com.github.javaparser.metamodel.PropertyMetaModel (26)* com.github.javaparser.ast.type.WildcardType (25)* com.github.javaparser.ast.expr.ObjectCreationExpr (24)* com.github.javaparser.ast.type.PrimitiveType (24)* com.github.javaparser.printer.lexicalpreservation.NodeText (24)* com.github.javaparser.utils.VisitorList (24)* com.github.javaparser.printer.lexicalpreservation.Difference (23)* com.github.javaparser.ast.comments.Comment (22)* com.github.javaparser.ast.expr.FieldAccessExpr (22)* com.github.javaparser.ast.type.ClassOrInterfaceType (22)* com.github.javaparser.utils.Utils (22)* com.github.javaparser.JavaToken (22)* com.github.javaparser.ast.body.ClassOrInterfaceDeclaration (21)* com.github.javaparser.ast.body.FieldDeclaration (21)* com.github.javaparser.ast.expr.MethodCallExpr (21)* com.github.javaparser.ast.stmt.ExplicitConstructorInvocationStmt (21)* com.github.javaparser.ast.stmt.IfStmt (21)* com.github.javaparser.ParserConfiguration (21)

現在讓我們嘗試其他的東西。 讓我們嘗試查找所有測試類,并確保其名稱以“ Test”結尾。 測試類將是至少具有用org.unit.Test注釋的方法的類。

fun CtClass<*>.isTestClass() = this.methods.any { it.annotations.any { it.annotationType.qualifiedName == "org.junit.Test" } }fun verifyTestClassesHaveProperName(ctModel: CtModel) {val testClasses = ctModel.filterChildren<CtClass<*>> { it is CtClass<*> && it.isTestClass() }.list<CtClass<*>>()val testClassesNamedCorrectly = testClasses.filter { it.simpleName.endsWith("Test") }val testClassesNotNamedCorrectly = testClasses.filter { it !in testClassesNamedCorrectly }printTitle("Test classes named correctly")println("N Classes named correctly: ${testClassesNamedCorrectly.size}")println("N Classes not named correctly: ${testClassesNotNamedCorrectly.size}")printList(testClassesNotNamedCorrectly.asSequence().sortedBy { it.qualifiedName }.map { it.qualifiedName }) }fun main(args: Array<String>) {val launcher = Launcher()launcher.addInputResource("codebases/jp/javaparser-core/src/main/java")launcher.addInputResource("codebases/jp/javaparser-core-testing/src/test/java")launcher.addInputResource("libs/junit-vintage-engine-4.12.3.jar")launcher.environment.noClasspath = trueval model = launcher.buildModel()verifyTestClassesHaveProperName(model) }

構建模型與以前幾乎相同,我們只是添加了更多的源目錄和JAR,作為測試模塊對JUnit的依賴。

在verifyTestClassesHaveProperName中,我們:

  • 過濾所有屬于測試類的類 (它們至少具有一個用org.junit.Test注釋的方法)
  • 查找所有名稱以Test結尾的測試類,以及所有不包含
  • 我們打印要修復的類的列表以及有關它們的一些統計信息

讓我們運行這段代碼,我們得到以下結果:

================================ | Test classes named correctly | ================================ N Classes named correctly: 124 N Classes not named correctly: 2* com.github.javaparser.wiki_samples.CreatingACompilationUnitFromScratch* com.github.javaparser.wiki_samples.removenode.RemoveDeleteNodeFromAst

當然,這些只是相當簡單的示例,但希望它們足以顯示Spoon和代碼分析的潛力。 處理代表您的代碼的模型,提取有趣的信息以及驗證是否遵守某些語義規則是相當容易的。

有關更高級的用法,您還可以查看有關使用Spoon進行體系結構實施的本文。

使用Spoon執行代碼生成

讓我們來看一個考慮一個常見任務的代碼生成示例:JSON的代碼序列化和反序列化。 我們將從采用JSON模式開始,然后我們將生成表示JSON模式描述的實體的類。

這是一個相當高級的示例,我花了一些時間熟悉Spoon才能編寫它。 我還必須向他們的團隊提出一些問題,以解決一些問題。 的確,這段代碼絕非易事,但是我認為我們應該認為這是一個非常復雜的功能,因此對我來說聽起來很公平。

好的,現在讓我們進入代碼。

這是一個JSON模式:

{"$id": "https://example.com/arrays.schema.json","$schema": "http://json-schema.org/draft-07/schema#","description": "A representation of a person, company, organization, or place","type": "object","properties": {"fruits": {"type": "array","items": {"type": "string"}},"vegetables": {"type": "array","items": { "$ref": "#/definitions/veggie" }}},"definitions": {"veggie": {"type": "object","required": [ "veggieName", "veggieLike" ],"properties": {"veggieName": {"type": "string","description": "The name of the vegetable."},"veggieLike": {"type": "boolean","description": "Do I like this vegetable?"}}}} }

在頂層,我們可以看到整個架構所代表的實體。 我們知道它將被表示為一個對象并具有兩個屬性:

  • 水果 :字符串數組
  • 蔬菜 :一組蔬菜 ,其中蔬菜是下面描述的另一個對象,在“定義”部分中

在定義部分,我們可以看到素食是具有兩個屬性的對象:

  • veggieName :字符串
  • veggieLike :布爾值

我們應該得到什么

我們想要得到的是兩個java類:一個代表整個模式,一個代表單個蔬菜。 這兩個類應允許讀取和寫入單個字段,將實例序列化為JSON以及從JSON反序列化實例。

我們的代碼應生成兩個類:

package com.thefruit.company;public class FruitThing implements com.strumenta.json.JsonSerializable {private java.util.List<java.lang.String> fruits;public java.util.List<java.lang.String> getFruits() {return fruits;}public void setFruits(java.util.List<java.lang.String> fruits) {this.fruits = fruits;}private java.util.List<com.thefruit.company.Veggie> vegetables;public java.util.List<com.thefruit.company.Veggie> getVegetables() {return vegetables;}public void setVegetables(java.util.List<com.thefruit.company.Veggie> vegetables) {this.vegetables = vegetables;}public com.google.gson.JsonObject serialize() {com.google.gson.JsonObject res = new com.google.gson.JsonObject();res.add("fruits", com.strumenta.json.SerializationUtils.serialize(fruits));res.add("vegetables", com.strumenta.json.SerializationUtils.serialize(vegetables));return res;}public static com.thefruit.company.FruitThing unserialize(com.google.gson.JsonObject json) {com.thefruit.company.FruitThing res = new com.thefruit.company.FruitThing();res.setFruits((java.util.List) com.strumenta.json.SerializationUtils.unserialize(json.get("fruits"), com.google.gson.reflect.TypeToken.getParameterized(java.util.List.class, java.lang.String.class)));res.setVegetables((java.util.List) com.strumenta.json.SerializationUtils.unserialize(json.get("vegetables"), com.google.gson.reflect.TypeToken.getParameterized(java.util.List.class, com.thefruit.company.Veggie.class)));return res;} }

和:

package com.thefruit.company;public class Veggie implements com.strumenta.json.JsonSerializable {private java.lang.String veggieName;public java.lang.String getVeggieName() {return veggieName;}public void setVeggieName(java.lang.String veggieName) {this.veggieName = veggieName;}private boolean veggieLike;public boolean getVeggieLike() {return veggieLike;}public void setVeggieLike(boolean veggieLike) {this.veggieLike = veggieLike;}public com.google.gson.JsonObject serialize() {com.google.gson.JsonObject res = new com.google.gson.JsonObject();res.add("veggieName", com.strumenta.json.SerializationUtils.serialize(veggieName));res.add("veggieLike", com.strumenta.json.SerializationUtils.serialize(veggieLike));return res;}public static com.thefruit.company.Veggie unserialize(com.google.gson.JsonObject json) {com.thefruit.company.Veggie res = new com.thefruit.company.Veggie();res.setVeggieName((java.lang.String) com.strumenta.json.SerializationUtils.unserialize(json.get("veggieName"), com.google.gson.reflect.TypeToken.get(java.lang.String.class)));res.setVeggieLike((boolean) com.strumenta.json.SerializationUtils.unserialize(json.get("veggieLike"), com.google.gson.reflect.TypeToken.get(boolean.class)));return res;} }

這是我們如何使用這兩個類的示例:

package com.thefruit.company;import com.google.gson.Gson; import com.google.gson.GsonBuilder; import com.google.gson.JsonElement;import java.util.Arrays;public class Example {public static void main(String[] args) {FruitThing ft = new FruitThing();ft.setFruits(Arrays.asList("Banana", "Pear", "Apple"));Veggie cucumber = new Veggie();cucumber.setVeggieLike(false);cucumber.setVeggieName("Cucumber");Veggie carrot = new Veggie();carrot.setVeggieLike(true);carrot.setVeggieName("Carrot");ft.setVegetables(Arrays.asList(cucumber, carrot));Gson gson = new GsonBuilder().setPrettyPrinting().create();System.out.println(gson.toJson(ft.serialize()));JsonElement serialized = ft.serialize();FruitThing unserializedFt = FruitThing.unserialize(serialized.getAsJsonObject());System.out.println("Fruits: " + unserializedFt.getFruits());} }

在該示例中,我們構建了FruitThing和幾個Veggies的實例。 然后,我們對它們進行序列化并反序列化它們,以便我們可以證明序列化和反序列化都可以工作。

生成過程:一般組織

生成過程將生成一組GeneratedJavaFile實例,每個實例都有自己的文件名和代碼。 以后我們可以將它們寫入文件或在內存中進行編譯。

在程序的主要功能中,我們將讀取JSON模式并將其傳遞給函數generateJsonSchema 。 我們將其與兩個參數一起傳遞:首先在其中生成我們的類的包的名稱,然后是代表整個架構的類的名稱。

一旦我們獲得了生成的類,我們就將它們打印在屏幕上以快速瀏覽。

data class GeneratedJavaFile(val filename: String, val code: String)fun main(args: Array<String>) {Dummy::class.java.getResourceAsStream("/a_json_schema.json").use {val generatedClasses = generateJsonSchema(it, "com.thefruit.company", "FruitThing")generatedClasses.forEach {println("*".repeat(it.filename.length))println(it.filename)println("*".repeat(it.filename.length))println(it.code)}} }

好的,所以魔術發生在generateJsonSchema中,對嗎?

fun generateJsonSchema(jsonSchema: InputStream, packageName: String, rootClassName: String) : List<GeneratedJavaFile> {val rawSchema = JSONObject(JSONTokener(jsonSchema))val schema = SchemaLoader.load(rawSchema) as ObjectSchemaval cus = generateClasses(schema, packageName, rootClassName)val pp = DefaultJavaPrettyPrinter(StandardEnvironment())return cus.map { cu ->pp.calculate(cu, cu.declaredTypes)val filename = cu.declaredTypes[0].qualifiedName.replace('.', File.separatorChar) + ".java"GeneratedJavaFile(filename, pp.result)} }

在generateJsonSchema中,我們解析提供模式的InputStream,并調用generateClasses ,這將返回一堆CompilationUnits。 基本上,每個CompilationUnit都是單個Java文件的抽象語法樹。

一旦獲得了這些編譯單元,就將它們打印為Java代碼。 我們還計算適當的文件名并實例化GeneratedJavaFile實例。

因此,看來我們現在來看一下generateClasses 。

fun generateClasses(schema: ObjectSchema, packageName: String, rootClassName: String) : List<CompilationUnit> {// First we create the classesval pack = CtPackageImpl()pack.setSimpleName<CtPackage>(packageName)val classProvider = ClassProvider(pack)schema.generateClassRecursively(classProvider, rootClassName)// Then we put them in compilation units and we generate themreturn classProvider.classesForObjectSchemas.map {val cu = CompilationUnitImpl()cu.isAutoImport = truecu.declaredPackage = packcu.declaredTypes = listOf(it.value)cu}.toList() }

在generateClasses中,我們首先創建包( CtPackageImpl類)。 我們將使用它來生成所有類。 我們將其保留在ClassProvider類中。 它將用于生成和跟蹤我們將生成的類。 然后,我們調用添加到架構的擴展方法,稱為generateClassRecursively 。

最后,我們將從classProvider中獲取類,并將其放入CompilationUnits中。

private fun Schema.generateClassRecursively(classProvider: ClassProvider, name: String? = null) {when (this) {is ObjectSchema -> {classProvider.register(this, this.generateClass(classProvider, name))this.propertySchemas.forEach { it.value.generateClassRecursively(classProvider) }}is ArraySchema -> this.allItemSchema.generateClassRecursively(classProvider)is StringSchema, is BooleanSchema -> nullis ReferenceSchema -> this.referredSchema.generateClassRecursively(classProvider)else -> TODO("not implemented: ${this.javaClass}")} }

generateClassRecursively會發生什么? 基本上,我們尋找定義對象的模式,并為每個對象生成一個類。 我們還對模式進行爬網以查找屬性,以查看它們是否間接定義或使用了我們可能要為其生成類的其他對象模式。

在ObjectSchema的擴展方法generateClass中生成一個類。 當它產生一個類時,我們將其傳遞給classProvider以便對其進行記錄。

private fun ObjectSchema.generateClass(classProvider: ClassProvider, name: String? = null): CtClass<Any> {return CtClassImpl<Any>().let { ctClass ->val packag = classProvider.packpackag.types.add(ctClass)ctClass.setParent(packag)ctClass.setVisibility<CtModifiable>(ModifierKind.PUBLIC)ctClass.setSimpleName<CtClass<Any>>(name ?: this.schemaLocation.split("/").last().capitalize())ctClass.setSuperInterfaces<CtType<Any>>(setOf(createTypeReference(JsonSerializable::class.java)))this.propertySchemas.forEach {ctClass.addProperty(it.key, it.value, classProvider)}addSerializeMethod(ctClass, this, classProvider)addUnserializeMethod(ctClass, this, classProvider)ctClass} }

到目前為止,我們已經設置了對架構進行爬網并確定生成內容的邏輯,但是我們還沒有看到很多Spoon特定的API。 這在generateClass中發生了變化。

在這里,我們首先實例化CtClassImpl,然后:

  • 設置適當的包(從classProvider獲得)
  • 將課程設為公開
  • 指定類的名稱:如果類代表整個模式,我們可以將其作為參數接收,否則我們可以從模式本身派生它
  • 查看單個屬性并在addProperty中處理它們
  • 調用addSerializeMethod添加一個序列化方法,我們將使用該方法從此類的實例生成JSON

那么,我們如何添加屬性?

fun CtClass<*>.addProperty(name: String, schema: Schema, classProvider: ClassProvider) {val field = CtFieldImpl<Any>().let {it.setSimpleName<CtField<Any>>(name)it.setType<CtField<Any>>(schema.toType(classProvider))it.setVisibility<CtField<Any>>(ModifierKind.PRIVATE)}this.addField<Any, Nothing>(field)addGetter(this, field)addSetter(this, field) }

我們只需添加一個字段( CtField )。 我們設置正確的名稱,類型和可見性,并將其添加到類中。 目前我們還沒有生成getter或setter。

生成過程:序列化

在本節中,我們將看到如何生成類的serialize方法。 對于我們的兩個類,它們看起來像這樣:

public class FruitThing implements com.strumenta.json.JsonSerializable {...public com.google.gson.JsonObject serialize() {com.google.gson.JsonObject res = new com.google.gson.JsonObject();res.add("fruits", com.strumenta.json.SerializationUtils.serialize(fruits));res.add("vegetables", com.strumenta.json.SerializationUtils.serialize(vegetables));return res;}... }public class Veggie implements com.strumenta.json.JsonSerializable {...public com.google.gson.JsonObject serialize() {com.google.gson.JsonObject res = new com.google.gson.JsonObject();res.add("veggieName", com.strumenta.json.SerializationUtils.serialize(veggieName));res.add("veggieLike", com.strumenta.json.SerializationUtils.serialize(veggieLike));return res;}... }

這是生成這種方法的切入點:

fun addSerializeMethod(ctClass: CtClassImpl<Any>, objectSchema: ObjectSchema, classProvider: ClassProvider) {val method = CtMethodImpl<Any>().let {it.setVisibility<CtModifiable>(ModifierKind.PUBLIC)it.setType<CtTypedElement<Any>>(jsonObjectType)it.setSimpleName<CtMethod<Any>>("serialize")val statements = LinkedList<CtStatement>()statements.add(createLocalVar("res", jsonObjectType, objectInstance(jsonObjectType)))objectSchema.propertySchemas.forEach { statements.addAll(addSerializeStmts(it, classProvider)) }statements.add(returnStmt(localVarRef("res")))it.setBodyBlock(statements)it}ctClass.addMethod<Any, CtType<Any>>(method) }

我們實例化CtMethodImpl然后:

  • 我們設置方法的可見性
  • 我們將返回類型設置為JSONObject
  • 我們將名稱設置為序列化
  • 我們創建JSONObject類型的res變量
  • 對于每個屬性,我們將生成序列化語句,以將該屬性的值添加到res中
  • 最后,我們添加一個return語句并將此塊設置為方法的主體

在這里,我們使用了一堆實用程序方法來簡化代碼,因為Spoon API非常冗長。

例如,我們使用createLocalVar和objectInstance ,如下所示:

fun createLocalVar(name: String, type: CtTypeReference<Any>, value: CtExpression<Any>? = null) : CtLocalVariable<Any> {return CtLocalVariableImpl<Any>().let {it.setSimpleName<CtNamedElement>(name)it.setType<CtTypedElement<Any>>(type)if (value != null) {it.setAssignment<CtRHSReceiver<Any>>(value)}it} }fun objectInstance(type: CtTypeReference<Any>) : CtConstructorCall<Any> {return CtConstructorCallImpl<Any>().let {it.setType<CtTypedElement<Any>>(type)it} }

現在,我們來看看如何為特定屬性生成序列化方法的語句。

un addSerializeStmts(entry: Map.Entry<String, Schema>,classProvider: ClassProvider): Collection<CtStatement> {return listOf(instanceMethodCall("add", listOf(stringLiteral(entry.key),staticMethodCall("serialize",listOf(fieldRef(entry.key)),createTypeReference(SerializationUtils::class.java))), target= localVarRef("res"))) }

基本上,我們委托給SerializationUtils.serialize 。 該方法將包含在運行時庫中,以與我們生成的代碼一起使用。

它是這樣的:

public class SerializationUtils {public static JsonElement serialize(Object value) {if (value instanceof JsonSerializable) {return ((JsonSerializable) value).serialize();}if (value instanceof Iterable<?>) {com.google.gson.JsonArray jsonArray = new com.google.gson.JsonArray();for (Object element : (Iterable<?>)value) {jsonArray.add(com.strumenta.json.SerializationUtils.serialize(element));}return jsonArray;}if (value instanceof Boolean) {return new JsonPrimitive((Boolean)value);}if (value instanceof String) {return new JsonPrimitive((String)value);}throw new UnsupportedOperationException("Value: " + value + " (" + value.getClass().getCanonicalName() + ")");}public static Object unserialize(JsonElement json, TypeToken<?> expectedType) {...to be discussed later...} }

我們序列化某個屬性的方式取決于其類型。 簡單值(字符串和布爾值)很容易,而數組則比較棘手。 對于任何可通過JsonSerializable進行調用的對象,我們都調用相應的serialize方法。 我們為什么要這樣做? 這樣我們就可以使用為類( FruitThing和Veggie )生成的序列化方法。

生成過程:反序列化

讓我們看看我們應該能夠生成的反序列化方法:

public class FruitThing implements com.strumenta.json.JsonSerializable {...public static com.thefruit.company.FruitThing unserialize(com.google.gson.JsonObject json) {com.thefruit.company.FruitThing res = new com.thefruit.company.FruitThing();res.setFruits((java.util.List) com.strumenta.json.SerializationUtils.unserialize(json.get("fruits"), com.google.gson.reflect.TypeToken.getParameterized(java.util.List.class, java.lang.String.class)));res.setVegetables((java.util.List) com.strumenta.json.SerializationUtils.unserialize(json.get("vegetables"), com.google.gson.reflect.TypeToken.getParameterized(java.util.List.class, com.thefruit.company.Veggie.class)));return res;}... }public class Veggie implements com.strumenta.json.JsonSerializable {...public static com.thefruit.company.Veggie unserialize(com.google.gson.JsonObject json) {com.thefruit.company.Veggie res = new com.thefruit.company.Veggie();res.setVeggieName((java.lang.String) com.strumenta.json.SerializationUtils.unserialize(json.get("veggieName"), com.google.gson.reflect.TypeToken.get(java.lang.String.class)));res.setVeggieLike((boolean) com.strumenta.json.SerializationUtils.unserialize(json.get("veggieLike"), com.google.gson.reflect.TypeToken.get(boolean.class)));return res;}... }

負責生成此類方法的代碼是哪一部分? 毫不奇怪,它稱為addUnserializeMethod :

fun addUnserializeMethod(ctClass: CtClassImpl<Any>, objectSchema: ObjectSchema, classProvider: ClassProvider) {val method = CtMethodImpl<Any>().let {it.setType<CtTypedElement<Any>>(createTypeReference(ctClass))it.setModifiers<CtModifiable>(setOf(ModifierKind.STATIC, ModifierKind.PUBLIC))it.setSimpleName<CtMethod<Any>>("unserialize")it.setParameters<CtExecutable<Any>>(listOf(CtParameterImpl<Any>().let {it.setSimpleName<CtNamedElement>("json")it.setType<CtTypedElement<Any>>(jsonObjectType)it}))val thisClass = createTypeReference(ctClass.qualifiedName)val statements = LinkedList<CtStatement>()statements.add(createLocalVar("res", thisClass, objectInstance(thisClass)))objectSchema.propertySchemas.forEach { statements.addAll(addUnserializeStmts(it, classProvider)) }statements.add(returnStmt(localVarRef("res")))it.setBodyBlock(statements)it}ctClass.addMethod<Any, CtType<Any>>(method) }

結構與我們之前所見非常相似。 當然,這里涉及的是對addUnserializeStmts的調用。

fun addUnserializeStmts(entry: Map.Entry<String, Schema>,classProvider: ClassProvider): Collection<CtStatement> {// call to get the field, e.g. `json.get("veggieName")`val getField = instanceMethodCall("get",listOf(stringLiteral(entry.key)),target = localVarRef("json"))// call to create the TypeToken, e.g., `TypeToken.get(String.class)`// or `TypeToken.getParameterized(List.class, String.class)`val ctFieldType = entry.value.toType(classProvider)val createTypeToken = if (ctFieldType is CtTypeReference<Any> && ctFieldType.actualTypeArguments.isNotEmpty()) {staticMethodCall("getParameterized",(listOf(classField(ctFieldType)) + ctFieldType.actualTypeArguments.map { classField(it) }).toList() as List<CtExpression<Any>>,createTypeReference(TypeToken::class.java))} else {staticMethodCall("get",listOf(classField(ctFieldType)),createTypeReference(TypeToken::class.java))}val callToUnserialize = staticMethodCall("unserialize",listOf(getField, createTypeToken),createTypeReference("com.strumenta.json.SerializationUtils"))val castedCallToUnserialize = cast(callToUnserialize, entry.value.toType(classProvider))return listOf(instanceMethodCall("set" + entry.key.capitalize(), listOf(castedCallToUnserialize), target= localVarRef("res"))) }

現在,事情變得復雜了。 我們基本上必須為每個屬性調用設置方法。 給設置器,我們將使用適當的類型轉換將反序列化的結果傳遞給屬性類型。 要調用反序列化,我們需要一個TypeToken,用于引導反序列化過程。 我們想以不同的方式對同一值進行反序列化,具體取決于我們是否要獲取整數或字符串:類型標記告訴我們要獲取的內容。

生成過程:注釋

要構建此示例,我們必須編寫許多實用程序方法。 我們在本文中未顯示的整個示例的某些部分,但是您可以在協同存儲庫中找到所有這些代碼。

還要注意,我們可以將代碼保存到文件中,并使用編譯器API進行編程編譯。 如果需要,我們甚至可以在內存中編譯它。 在實際情況下,我建議這樣做,而不是像我在本教程中所做的那樣,將代碼手動復制粘貼到文件中。

使用Spoon執行代碼轉換

在使用大型代碼庫或防止重復性任務出現人為錯誤時,代碼轉換可能非常有用。

例如,假設您決定更改必須實施特定模式的方式。 假設您在代碼庫中使用了數十次單例模式,并且您想確保每次懶惰地創建實例(即,僅在第一次需要時)。 您可以自動執行此轉換。

或者假設您正在更新正在使用的庫,并且您所依賴的特定方法已重命名,或者其參數順序已更改。 同樣,您可以通過使用代碼轉換來解決此問題。

對于我們的示例,我們將采取一些簡單的措施。 我們將重構一個類。 在此類中,我們有幾種方法可以接收特定的參數。 鑒于基本上每個操作都需要此參數,因此我們決定將其移至構造函數并將其另存為字段實例。 然后,我們要轉換獲取該參數的所有方法,以使它們不再需要它,而是訪問相應的字段。

讓我們看一下轉換的樣子:

// original code class MyClass {MyClass() {} void foo(MyParam param, String otherParam) {param.doSomething();}int bar(MyParam param) {return param.count();}}// transformed code class MyClass {MyParam param;MyClass(MyParam param) {this.param = param;} void foo(String otherParam) {this.param.doSomething();} int bar() { return this.param.count(); }}

在這個例子中,我們只轉換定義方法的類; 在實際情況下,我們可能還希望轉換這些方法的調用。

我們如何實現此代碼轉換

讓我們先看一下代碼轉換示例的主要方法,以便我們可以看到常規結構:

fun main(args: Array<String>) {val originalCode = """class MyClass {MyClass() {}void foo(MyParam param, String otherParam) {param.doSomething();}int bar(MyParam param) {return param.count();}}"""val parsedClass = Launcher.parseClass(originalCode)ParamToFieldRefactoring("param", createTypeReference("com.strumenta.MyParam")).refactor(parsedClass)println(parsedClass.toCode()) }

如您所見,我們:

  • 解析代碼
  • 應用我們的類ParamToFieldRefactoring中定義的重構
  • 我們打印結果代碼

有趣的地方當然是ParamToFieldRefactoring

class ParamToFieldRefactoring(val paramName: String, val paramType: CtTypeReference<Any>) {fun refactor(clazz: CtClass<*>) {// Add field to the classclazz.addField<Any, Nothing>(CtFieldImpl<Any>().let {it.setSimpleName<CtNamedElement>(paramName)it.setType<CtTypedElement<Any>>(paramType)it})// Receive the value for the field in each constructorclazz.constructors.forEach {it.addParameter<Nothing>(CtParameterImpl<Any>().let {it.setSimpleName<CtNamedElement>(paramName)it.setType<CtTypedElement<Any>>(paramType)it})it.body.statements.add(CtAssignmentImpl<Any, Any>().let {it.setAssigned<CtAssignment<Any, Any>>(qualifiedFieldAccess(paramName, clazz.qualifiedName))it.setAssignment<CtRHSReceiver<Any>>(localVarRef(paramName))it})}clazz.methods.filter { findParamToChange(it) != null }.forEach {val param = findParamToChange(it)!!CtIterator(it).forEach {if (it is CtParameterReference<*> && it.simpleName == paramName) {val cfr = CtFieldReferenceImpl<Any>()cfr.setSimpleName<CtReference>(paramName)cfr.setDeclaringType<CtFieldReference<Any>>(createTypeReference(clazz.qualifiedName))it.replace(cfr)}}param.delete()}}fun findParamToChange(method: CtMethod<*>) : CtParameter<*>? {return method.parameters.find { it.simpleName == paramName }} }

首先,我們將新字段添加到類中:

clazz.addField<Any, Nothing>(CtFieldImpl<Any>().let {it.setSimpleName<CtNamedElement>(paramName)it.setType<CtTypedElement<Any>>(paramType)it})

然后,向所有構造函數添加一個參數,以便我們可以接收該值并將其分配給該字段:

// Receive the value for the field in each constructorclazz.constructors.forEach {it.addParameter<Nothing>(CtParameterImpl<Any>().let {it.setSimpleName<CtNamedElement>(paramName)it.setType<CtTypedElement<Any>>(paramType)it})it.body.statements.add(CtAssignmentImpl<Any, Any>().let {it.setAssigned<CtAssignment<Any, Any>>(qualifiedFieldAccess(paramName, clazz.qualifiedName))it.setAssignment<CtRHSReceiver<Any>>(localVarRef(paramName))it})}

請注意,在實際的應用程序中,我們可能還需要考慮該類過去僅具有默認構造函數的情況,并添加一個采用將單個值分配給字段的全新構造函數。 為了簡單起見,我們在示例中忽略了這一點。

最后,我們要修改所有方法。 如果他們使用的參數名稱被考慮,我們將刪除該參數。 我們還將查找對該參數的所有引用,并將其替換為對新字段的引用:

clazz.methods.filter { findParamToChange(it) != null }.forEach {val param = findParamToChange(it)!!CtIterator(it).forEach {if (it is CtParameterReference<*> && it.simpleName == paramName) {val cfr = CtFieldReferenceImpl<Any>()cfr.setSimpleName<CtReference>(paramName)cfr.setDeclaringType<CtFieldReference<Any>>(createTypeReference(clazz.qualifiedName))it.replace(cfr)}}param.delete()}

就是這樣! 現在,我們應該只打印代碼,我們就完成了。

我們如何打印代碼? 通過一個名為toCode的擴展方法:

fun CtClass<*>.toCode() : String {val pp = DefaultJavaPrettyPrinter(StandardEnvironment())val cu = CompilationUnitImpl()pp.calculate(cu, listOf(this))return pp.result }

有關代碼轉換的更多信息

如果您想了解有關使用Spoon進行代碼轉換的更多信息,請看以下內容:

  • CocoSpoon ,用于檢測Java代碼以計算代碼覆蓋率的工具
  • Trebuchet ,一個概念證明,展示如何使用Spoon將Java代碼轉換為C ++。

這篇文章是如何誕生的

Spoon是處理Java代碼的工具。 在某種程度上,它可以看作是JavaParser的競爭對手。 我一直想研究了很長一段時間,但我有許多事情一大堆 ,我想看看和勺子從未到列表的頂部。 然后,JavaParser的一些用戶指出了關于Spoon項目的討論,討論了JavaParser和Spoon之間的區別。 在我看來,存在一些誤解,Spoon的貢獻者賣出的JavaParser有點短……在成千上萬的開發人員和知名公司都使用JavaParser并對此感到非常滿意之后。 另外,JavaParser可能是最著名的Java解析器。 因此,我開始與Spoon的貢獻者進行討論,這引發了撰寫本文的想法。

當這篇文章是在Spoon的貢獻者的幫助下寫的時,我是這篇文章的作者,而且我還是JavaParser的貢獻者,所以這是我的“偏見警報”!

比較Spoon和JavaParser

Spoon是JavaParser的學術替代品。 雖然JavaParser本身實現符號解析(這是最難的部分),但Spoon卻充當Eclipse Java編譯器的包裝,然后在其之上構建一些高級API。 那么,這種選擇會有什么后果呢?

  • Eclipse Java編譯器已經成熟,盡管并非沒有錯誤,但它相當可靠
  • Eclipse Java編譯器是一個大型野獸,它具有依賴性和復雜的配置
  • Eclipse Java編譯器是……編譯器,它不是用于符號解析的庫,因此它不如我們在JavaParser上擁有的本地解決方案靈活。

就個人而言,我對成為JavaParser的貢獻者有很大的偏見。 我已經習慣了JavaParser,Spoon的某些行為對我來說似乎是不自然的。 例如,對片段表達式的類型強制轉換似乎不起作用。 類訪問(例如,“ String.class”)不是由特定表達式表示,而是由字段訪問表示。 但是,某些功能確實很有用,我們也應該在JavaParser中獲得它們。

總而言之,它們是不同的工具,具有不同的功能集,而且我認為也有不同的理念,如下所述。

關于文檔,對于JavaParser來說似乎更好一些:我們有一本書,可以免費下載,并下載了數千次,還擁有一套教程。

不同的哲學

現在,Spoon是在學術環境和法國創立的。 以我的經驗,法國工程師非常有才華,但他們傾向于以“瘋狂的方式”重新發明事物。 以該項目采用的許可證為例:那是Apache許可證嗎? GPL? LGPL? Eclipse許可證? 不,這是CeCILL-C免費軟件許可協議 。 我從未聽說過的許可證,專門為遵守某些法國法規而創建。 現在,這可能是有史以來最偉大的許可證,但是對于想要采用該項目的公司,他們需要仔細研究一下,弄清楚這意味著什么,意味著什么,如果它與他們正在使用的其他許可證兼容,并且以此類推。 我認為,如果他們只是選擇一個現有許可證,事情可能會簡單得多。 因為存在現實 ,在這種現實中,公司不想僅使用Spoon就必須學習此許可。 這與我們非常務實的 JavaParser中的方法截然不同。 我們與公司討論并確定了他們需要哪些許可證,然后我們努力為用戶提供雙重許可證(Apache許可證或LGPL)。 為什么? 因為他們是他們熟悉的選擇。

總的來說,當我與Spoon的家伙交談時,我有不同的哲學感受。 他們清楚地意識到他們的產品要好得多,并且坦率地說,JavaParser如此受歡迎,讓他們有些失望。 我們討論了一些合作的可能性,但在我看來,這是從我們正確的角度出發。 在JavaParser中,我們不認為我們是對的。 我們只是聽取用戶意見,在彼此之間進行討論,然后再努力向前邁進,使用戶的生活更加輕松。 一個很大的優勢就是我們會收到很多反饋,因此當我們出錯時,用戶可以幫助我們糾正方向。

關于依賴關系,到目前為止,在JavaParser上,我們一直在努力保持核心模塊沒有任何依賴關系。 我們將來可能會放寬此約束,但總的來說,我們將依賴管理視為重要方面。 相反,在Spoon中,您需要添加一個Maven存儲庫以使用甚至不在Maven Central或任何知名Maven存儲庫上的庫。 為什么? 為什么要讓用戶的生活變得更加艱難?

結論

我認為代碼處理功能非常強大:它允許使用我們開發人員的技能來自動化部分工作,從而減少工作量和錯誤。 如果您使用大型代碼庫,那么它是在工具箱中的好工具。 至少我認為,更多的開發人員應該意識到它提供的可能性。

在Java代碼上執行代碼處理時,Spoon是有效的解決方案。 因此,我邀請您熟悉并考慮使用它,我想您會幫自己一個忙。

翻譯自: https://www.javacodegeeks.com/2019/03/analyze-generate-transform-java-spoon.html

創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎

總結

以上是生活随笔為你收集整理的如何以及为什么使用Spoon分析,生成和转换Java代码的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

91福利免费 | 中文国产成人精品久久一 | 日本精品在线视频 | 亚洲国产精品电影 | 久久精品999| 中文字幕网址 | 国产在线色视频 | 日韩av片无码一区二区不卡电影 | 国产午夜精品免费一区二区三区视频 | 91视频免费国产 | 久久精品第一页 | 天无日天天操天天干 | 五月婷婷激情综合网 | 免费av网址大全 | 国内精品久久久久久中文字幕 | 国产亚洲永久域名 | 96精品视频 | 91精品福利在线 | 国产精品美女999 | 不卡电影免费在线播放一区 | 精品久久久久久久久中文字幕 | 亚洲aⅴ久久精品 | 成人在线视频论坛 | 午夜色场 | 日韩精品最新在线观看 | 国产小视频在线 | 国产生活一级片 | 西西人体4444www高清视频 | 欧美一区二区在线免费观看 | 啪一啪在线| 草免费视频 | 久久免费观看少妇a级毛片 久久久久成人免费 | 国产成人三级三级三级97 | 成人av在线看 | 天天干天天操天天操 | 久久精品美女视频网站 | 成人宗合网 | 日韩高清免费观看 | 成 人 免费 黄 色 视频 | 香蕉影院在线播放 | 亚洲三级av| 国产亚洲一区 | 99久国产| 在线看91| 亚洲精品一区二区三区新线路 | 精壮的侍卫呻吟h | 四虎最新域名 | 久久艹中文字幕 | 日韩一区二区三免费高清在线观看 | 久久电影色| 久久在线影院 | 不卡av在线 | 久久久综合精品 | 在线亚洲激情 | 日韩av电影网站在线观看 | 日批在线看 | 高清不卡一区二区三区 | 天天综合在线观看 | 亚洲天堂网视频在线观看 | 亚洲另类视频在线观看 | 96久久欧美麻豆网站 | 99视频在线观看一区三区 | 观看免费av | 波多野结衣电影一区 | 久久久久久久久久久久久9999 | 婷婷丁香九月 | 91精品国产一区二区在线观看 | 欧美色综合天天久久综合精品 | 婷婷 综合 色 | 日本爽妇网| 日韩激情第一页 | 青青久草在线 | www.xxx.性狂虐 | 午夜久久电影网 | 色妞久久福利网 | 欧美日韩国产在线 | 特级毛片网站 | 成人黄色免费观看 | 久久人人97超碰国产公开结果 | 在线成人国产 | 国产亚洲精品久久久久久无几年桃 | 中文字幕中文字幕 | 免费在线看成人av | 中文字幕 在线看 | 免费在线观看不卡av | 欧美日韩国产mv | 99中文视频在线 | 四虎在线永久免费观看 | 日韩视频www | 欧美一级视频在线观看 | 国产精品18久久久久vr手机版特色 | 九色琪琪久久综合网天天 | 手机在线永久免费观看av片 | 91精品国产麻豆国产自产影视 | 在线va网站 | 国产精品乱码高清在线看 | 最新国产精品视频 | 国产正在播放 | 久久久久久久综合色一本 | 国产精品第一页在线观看 | 一区二区三区在线看 | 中文字幕在线字幕中文 | av在线免费在线 | 日韩欧美在线高清 | 日日爽天天 | 国产97碰免费视频 | 欧美久久综合 | 国产正在播放 | 免费看的视频 | 蜜臀av性久久久久av蜜臀三区 | 免费av观看网站 | 激情av资源 | 国产精品久久久久久久久久久不卡 | 日日婷婷夜日日天干 | 欧美 日韩 成人 | 亚洲天堂激情 | 丝袜+亚洲+另类+欧美+变态 | 日韩一区二区免费在线观看 | 激情在线五月天 | 天天搞天天干天天色 | 日本aa在线 | 国产日韩中文字幕 | 国产精品久久久久久久av大片 | 国产超碰在线 | 国产麻豆视频 | 久久久久久久久久久高潮一区二区 | 成年人在线看片 | 蜜桃视频色 | 色偷偷88888欧美精品久久久 | 欧美xxxxx在线视频 | 免费av在线网站 | 中文字幕韩在线第一页 | 在线免费观看国产视频 | 久久精品国产亚洲a | 蜜桃麻豆www久久囤产精品 | 六月丁香在线观看 | 国产老妇av | 一区二区三区在线观看免费 | 国产香蕉av | 国产精品人成电影在线观看 | 九九九热 | 天天干天天干天天干天天干天天干天天干 | 国精产品一二三线999 | 在线中文字幕av观看 | 国产精品国产三级国产aⅴ9色 | 91av综合| 中文字幕在线观看一区二区 | 97在线免费 | 久久成人欧美 | 91麻豆国产福利在线观看 | 国产一级免费视频 | 久久天| 久久免费精品一区二区三区 | 99视频在线免费播放 | 国产精品对白一区二区三区 | 午夜视频在线观看欧美 | 人人爽人人搞 | 国产精品 日韩 欧美 | 黄色的片子| 国产精在线| 超碰激情在线 | 在线 视频 一区二区 | 久久精品欧美一区二区三区麻豆 | 波多野结衣久久精品 | 一区二区视频欧美 | 欧美在线视频一区二区三区 | 亚洲女欲精品久久久久久久18 | 日韩电影在线观看一区二区三区 | 久久人人爽视频 | 国产成人精品一区二区三区 | 久久久精品欧美一区二区免费 | 午夜视频在线瓜伦 | 丁香婷婷深情五月亚洲 | 国产一区二区在线观看免费 | 成人黄色av免费在线观看 | 国产精品午夜久久 | 欧美人牲 | 综合网天天色 | 最新高清无码专区 | 91av原创 | 国产精品白浆 | 国产999精品久久久久久 | 91重口视频| 欧美精品一二三 | 美女免费电影 | 久久在线影院 | 亚洲精选在线观看 | 色视频 在线 | 国产91精品一区二区麻豆网站 | 国产一区在线免费 | 婷婷精品国产欧美精品亚洲人人爽 | 日韩av一区二区三区在线观看 | 国产黄在线 | 国产精品黄色在线观看 | 九九久久久久久久久激情 | 中文一区二区三区在线观看 | 午夜国产福利视频 | 一区二区欧美激情 | 国产一级视频免费看 | 亚洲人xxx | 日韩高清黄色 | 国产在线一线 | 成人av电影在线 | 91插插视频 | 玖草在线观看 | 久久久午夜剧场 | 日韩精品一区二区三区视频播放 | 亚洲人成人天堂h久久 | a级一a一级在线观看 | 欧美另类高潮 | 91福利视频久久久久 | 国产精品18videosex性欧美 | 国产精品久久久久9999吃药 | 免费a v在线 | 久久精品国产亚洲a | 日韩色视频在线观看 | 国产原创在线观看 | 亚洲精品玖玖玖av在线看 | 国产.精品.日韩.另类.中文.在线.播放 | 国产精品久99 | a天堂最新版中文在线地址 久久99久久精品国产 | 久久精品二区 | 日本精a在线观看 | 在线观看亚洲成人 | 欧美性网站 | 成人一区二区在线观看 | 在线观看国产 | 成人h视频 | 国产日韩精品一区二区 | 国产在线a免费观看 | 国产区免费| 久章草在线观看 | 久久视| 天天干天天在线 | 日韩亚洲在线视频 | 国产精品美女久久久久久久久 | 日韩精品视频免费看 | 97成人超碰| 日韩午夜电影网 | 波多野结衣在线中文字幕 | 亚洲色图av | 丁香激情五月 | 国产视频在线一区二区 | 日韩精品免费一区二区 | 日韩啪啪小视频 | 久久久久亚洲精品 | 国内小视频 | 500部大龄熟乱视频 欧美日本三级 | 国产成人三级在线观看 | 久草视频网 | av韩国在线 | 激情久久久久久久久久久久久久久久 | 一区二区网 | 精品一区二区免费 | 夜夜爽www| 夜色.com | 色综合中文综合网 | 国产中文在线字幕 | 国产成本人视频在线观看 | www久久国产| 欧美国产91 | 97国产一区二区 | 69精品视频在线观看 | 久久九九网站 | www.天天射 | 国产精品原创在线 | 在线免费观看黄网站 | 国产日韩中文字幕在线 | 人人爱在线视频 | 一区二区视频在线播放 | 日韩欧美一区二区三区视频 | 丝袜美腿一区 | 97超级碰碰碰碰久久久久 | 成年人黄色免费看 | 亚洲国产播放 | 午夜婷婷在线观看 | 91看片在线看片 | 中文字幕观看视频 | 婷婷丁香五 | www国产在线 | av资源在线观看 | 久久99久国产精品黄毛片入口 | 日韩精品久久久久久中文字幕8 | 欧美日韩精品免费观看视频 | 亚洲精品一区二区久 | 日韩精品久久久久久 | 偷拍区另类综合在线 | 337p西西人体大胆瓣开下部 | 久久电影国产免费久久电影 | 日韩在线电影 | 国产一区二区三区久久久 | 黄色一区二区在线观看 | 亚洲在线视频免费 | 亚洲mv大片欧洲mv大片免费 | 久久99精品国产麻豆宅宅 | 热久久免费视频 | 国产精品久久久久久久久久99 | 亚洲最大激情中文字幕 | 在线视频中文字幕一区 | 国产精品久久久久久久久久久久久 | 欧美日韩成人一区 | 天天看天天操 | 天天操狠狠干 | 国产一级二级av | 97国产精品| 日韩,精品电影 | 久久国产精品99精国产 | 91香蕉国产在线观看软件 | 国产 日韩 在线 亚洲 字幕 中文 | 探花视频在线版播放免费观看 | av动图| 久久这里只精品 | 三级av黄色| 五月激情亚洲 | 九九热在线观看 | 天天操天天色综合 | 国产一二区精品 | 亚洲免费在线看 | 成人久久久久 | 欧美日本一二三 | 欧美天天综合 | 欧美精品久久人人躁人人爽 | 久久久久一区 | 在线观看av中文字幕 | 干综合网 | 夜色.com| 99久久精品国产欧美主题曲 | 午夜视频在线观看一区二区三区 | 福利视频 | 99热超碰在线 | 日韩黄色大片在线观看 | 日韩欧美在线观看一区二区三区 | 超碰免费公开 | 99re热精品视频 | 国产原创在线 | 亚洲国产中文在线 | 99精品国产一区二区 | 久久综合久久综合久久 | 月下香电影 | 黄色一级动作片 | 黄色大片日本 | 中文字幕黄色网址 | 日本在线观看黄色 | 在线香蕉视频 | 亚洲国产精品一区二区久久hs | 久久艹国产视频 | 国产婷婷在线观看 | 久久国语露脸国产精品电影 | 超碰成人网 | 成人久久国产 | 中文字幕永久在线 | 天天草天天操 | 日韩在线观看第一页 | 操操操av | 天天插天天操天天干 | 97超碰人人澡 | 黄色片亚洲 | 亚洲精品美女在线观看播放 | 亚洲国产欧美一区二区三区丁香婷 | www.色午夜 | 国产色综合天天综合网 | 少妇超碰在线 | 国产精品久久久久婷婷 | 欧美日韩精品久久久 | 欧美久久精品 | 欧美黑人性猛交 | 91精品视频一区 | 久久国产午夜精品理论片最新版本 | 51久久成人国产精品麻豆 | 九九视频免费在线观看 | 日韩久久久久久久久久久久 | www.操.com| 9在线观看免费高清完整版 玖玖爱免费视频 | 在线国产视频 | 日韩精品一区二区三区丰满 | 日韩欧美在线播放 | 久久久.com | 久久精品3 | 久久久久久久久久久久国产精品 | 成人久久综合 | 免费福利视频网 | 欧美日韩视频在线观看一区二区 | 国产精品ⅴa有声小说 | 91中文字幕在线 | 亚洲精品视频免费在线 | 精品成人在线 | 国产无限资源在线观看 | 久草视频中文 | 欧美一区二区三区特黄 | 天天色中文 | 91久久久久久久一区二区 | 中文资源在线官网 | 国产精品 999 | 97在线观看免费观看高清 | 一区二区三区高清 | 亚洲国产精品va在线看黑人动漫 | 国产美女免费看 | 国产精品正在播放 | 精品久久久久久久久久久久久久久久久久 | 五月婷香蕉久色在线看 | 亚洲一区精品人人爽人人躁 | 婷婷激情在线观看 | 韩国av电影网 | 狠狠色狠狠色综合系列 | 免费日韩一级片 | 手机在线视频福利 | 九九久久免费视频 | av一级在线观看 | 91视频88av | 99国产精品视频免费观看一公开 | 久久国产精品精品国产色婷婷 | 欧美日韩一区二区久久 | 国产欧美最新羞羞视频在线观看 | 中文字幕在线播放第一页 | 国产美女精品人人做人人爽 | 中文欧美字幕免费 | 久久久久久久久久久久电影 | 在线观看深夜福利 | 免费色视频网址 | 欧美日韩国产高清视频 | 欧美日韩69 | 又爽又黄又无遮挡网站动态图 | 亚洲电影久久 | 黄色精品久久久 | 亚洲最新视频在线播放 | 99精品乱码国产在线观看 | 久久久久国产a免费观看rela | a天堂中文在线 | 精品一区二区综合 | 亚洲精品美女在线观看播放 | 日韩无在线 | 一区中文字幕在线观看 | 久久999久久 | 国产精品亚洲综合久久 | 久久久久免费精品国产小说色大师 | www日韩在线 | 天天操天天射天天 | 麻花豆传媒mv在线观看网站 | 色偷偷88888欧美精品久久久 | 国产剧情亚洲 | 国产成人一区二区在线观看 | 成人av电影免费在线播放 | 婷婷色在线播放 | 久久久受www免费人成 | 91精品系列| 国产精品欧美久久 | 伊人久久婷婷 | 国产玖玖在线 | 国产一级性生活视频 | 欧美做受69 | 丁香五香天综合情 | 久久中文字幕在线视频 | 97精产国品一二三产区在线 | 免费看三级 | 欧美一区视频 | 国产精品a久久久久 | 在线观看成人一级片 | 欧美日韩一二三四区 | 日韩高清不卡一区二区三区 | 91视频免费看| 亚洲视频,欧洲视频 | 国产精品久久久视频 | www.天天干.com | av五月婷婷 | 国产日韩精品一区二区在线观看播放 | 久久国产露脸精品国产 | 国产一区二区在线观看免费 | 欧美日韩高清一区二区 | 又色又爽又黄高潮的免费视频 | 激情xxxx | 欧洲在线免费视频 | 深夜免费网站 | 精品久久久久久久久久 | 国产在线视频一区二区 | 青春草视频在线播放 | 亚洲久久视频 | 夜添久久精品亚洲国产精品 | 国产一级精品视频 | 又黄又爽又无遮挡免费的网站 | 韩日成人av | 97碰在线 | 亚洲天天做 | 日韩中字在线 | www.国产高清 | 国产麻豆成人传媒免费观看 | 色欧美88888久久久久久影院 | 五月天精品视频 | 国产999视频 | 国产精品 日韩 | 欧美a在线免费观看 | 久草视频在线免费看 | 黄色大全在线观看 | 日本一区二区三区免费观看 | 精品国产乱码久久久久久浪潮 | 国产精品日韩欧美 | 精品久久久久亚洲 | 在线观看国产区 | 久艹在线观看视频 | 午夜精品一区二区三区四区 | 久久久99精品免费观看 | 久久亚洲综合色 | 欧美一级电影片 | 成人三级网址 | 美女视频黄在线 | 免费观看一级特黄欧美大片 | 超黄视频网站 | 精品福利在线 | 蜜臀av网址 | 欧美91精品国产自产 | 一级黄色在线免费观看 | 国产精品久久久久久久久久尿 | 国产夫妻自拍av | 夜色资源网 | 99在线观看免费视频精品观看 | 99精品久久久久久久久久综合 | 射久久久 | 亚洲毛片视频 | 国产无限资源在线观看 | 天堂av高清| 伊人五月天.com | 天天综合天天做 | 成人免费xxx在线观看 | 日韩在线二区 | 日韩电影一区二区三区在线观看 | 在线观看视频国产一区 | 黄色aaaaa| 国产精品免费视频一区二区 | 月下香电影 | 91精品办公室少妇高潮对白 | 国产麻豆果冻传媒在线观看 | 久久免费视频播放 | 麻豆一区在线观看 | 91网在线观看 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 欧美成年网站 | 成人黄色中文字幕 | 国产九色在线播放九色 | 国产精品综合久久久 | 日韩网站在线免费观看 | 一区二区日韩av | 精品亚洲网 | 国产一区二区视频在线播放 | 亚洲三级国产 | 国内精品久久久久影院男同志 | 国产精品va在线 | 日本 在线 视频 中文 有码 | 成人资源在线 | 精品在线二区 | 超碰免费97| 成人在线视频一区 | 久久久高清一区二区三区 | 91大神精品视频在线观看 | 少妇性aaaaaaaaa视频 | 最新av在线网址 | 久草在线资源观看 | 一级黄色av| 久久精品国产一区 | 国产一二三区在线观看 | 免费在线成人av电影 | 欧美在线观看视频 | 一区二区三区四区五区在线视频 | 国产玖玖精品视频 | 国产精品久久久免费 | 国产中文字幕亚洲 | 婷婷综合影院 | 日韩av电影网站在线观看 | 91视频在线免费看 | 99在线精品观看 | 99视频久 | 亚洲三级国产 | 免费看在线看www777 | 精品久久影院 | 国产黄色一级大片 | 亚洲午夜精品久久久久久久久 | 丁香资源影视免费观看 | 波多野结衣在线中文字幕 | 国产看片免费 | 国产99久久 | 国产资源在线免费观看 | 欧美日韩视频 | 国产美女精品视频 | 97成人超碰 | 五月激情片 | 国产在线日本 | 亚洲精品国产综合久久 | 国产欧美精品在线观看 | 91九色国产 | 天天插日日射 | 91麻豆文化传媒在线观看 | 久久精品一区 | 黄色av大片| 成人在线观看资源 | 最新婷婷色 | av片一区二区| 国产999免费视频 | 国产精品久久久区三区天天噜 | 91麻豆精品国产91久久久无需广告 | 国产亚洲精品久久久久5区 成人h电影在线观看 | 免费看的黄色小视频 | 五月婷婷影视 | 日韩国产在线观看 | 黄色福利网 | 日本精品免费看 | 日韩高清一区在线 | 国产高清视频在线观看 | 亚洲综合涩 | 亚洲天堂毛片 | 国产精品久久久久av免费 | 欧美日韩色婷婷 | www.狠狠色.com| 黄色片网站免费 | 亚洲精品午夜久久久 | 91av99| 国产一级二级视频 | 欧美日韩国产mv | 日韩中文在线字幕 | 欧美精品三级 | 亚洲综合色视频在线观看 | 在线国产高清 | 天天色欧美 | 亚洲最大av| 国产高清免费观看 | 精品久久久久久久久久久久久久久久久久 | 亚洲乱码在线观看 | 久草男人天堂 | 久久99精品久久久久久三级 | 正在播放五月婷婷狠狠干 | 日本xxxx裸体xxxx17 | 欧美日韩一区三区 | 狠狠操导航 | 天天撸夜夜操 | 国产精品理论视频 | 婷婷丁香在线 | av在线精品| 热久久国产 | 色99网| 999在线视频 | 日韩黄色在线观看 | 久久精品国产免费看久久精品 | 超碰在线99 | 欧美 日韩 成人 | 天天色天天操综合网 | 麻豆视频在线免费 | 天天躁日日躁狠狠躁 | www.黄色片网站 | 国产精品久久免费看 | 视频国产在线观看18 | 国产精品av免费在线观看 | 麻豆va一区二区三区久久浪 | 911香蕉视频 | 天天综合网天天综合色 | 亚洲综合欧美激情 | 91视频专区| 在线观看不卡视频 | 国产夫妻av在线 | 免费日韩在线 | 国产xxxx | 亚洲精品视频免费 | 在线成人国产 | 中文av日韩 | 国产五月| 精品国产一区二区三区四区vr | 亚洲视频第一页 | 91中文字幕网 | 日韩免 | 毛片99 | 亚洲专区路线二 | 美女视频黄免费网站 | 米奇四色影视 | 69精品视频在线观看 | 青青草国产精品 | 在线国产能看的 | 欧美日韩一区二区免费在线观看 | 欧美激情视频在线免费观看 | 精品日韩在线 | 久久精品—区二区三区 | 久久草草影视免费网 | 97电影院网 | 国产美女久久久 | 久久免费视频这里只有精品 | 丁香av在线 | 韩日电影在线 | 亚洲成人av一区二区 | 伊人网综合在线观看 | 不卡电影免费在线播放一区 | 亚洲国产欧洲综合997久久, | 高清一区二区 | 亚洲免费在线观看视频 | 久久久久国产精品免费 | 欧美黑人巨大xxxxx | 99亚洲精品视频 | 亚洲 在线 | 国内精品久久久久 | 中国一级片在线观看 | 天天爽人人爽 | 久久y | 在线视频观看成人 | 99re国产视频| 日本丰满少妇免费一区 | 美女国内精品自产拍在线播放 | 国产91综合一区在线观看 | av不卡免费在线观看 | av在线免费播放网站 | 在线成人观看 | 色综合色综合久久综合频道88 | 欧美人体xx | 国产精品一区二区电影 | 8x成人免费视频 | 又黄又爽又无遮挡免费的网站 | 一二区av | 免费看黄色91 | 深爱激情开心 | 9在线观看免费 | 日韩精品2区 | 久久激情片 | 久久精品理论 | 天堂av免费在线 | 五月天综合激情网 | 中文字幕 第二区 | 亚洲va欧美va国产va黑人 | 日本久久久久久科技有限公司 | 婷婷久久网站 | 国产精品久久久久久久久费观看 | 视频国产在线观看18 | 麻豆国产精品永久免费视频 | 97国产在线视频 | 91精品在线麻豆 | 国产精品久久久视频 | 国产高清视频 | 日韩高清成人 | 日韩午夜精品福利 | 成人久久18免费网站图片 | 在线天堂8√| 日本性xxx | 国产精品剧情 | 激情久久久久久久久久久久久久久久 | 69精品视频 | 欧美一级久久久 | 97精品国自产拍在线观看 | 欧洲一区二区三区精品 | 国产精品国内免费一区二区三区 | 91一区二区在线 | 狠狠操夜夜操 | 在线观看国产福利片 | 最近字幕在线观看第一季 | 91视频免费看网站 | 国产一区二区在线免费视频 | 狠狠干夜夜操天天爽 | 九色精品免费永久在线 | 国产91精品高清一区二区三区 | 草久草久 | 97在线观看免费 | 欧美日韩精品免费观看 | 日本久久久久久科技有限公司 | 日韩专区在线播放 | 精品国产自在精品国产精野外直播 | 国产91影院 | 亚洲黄色在线观看 | 亚洲精品动漫成人3d无尽在线 | 欧美在线观看禁18 | 日本3级在线观看 | 国产精品久久久久影视 | 在线不卡视频 | 六月婷婷久香在线视频 | 99久久久国产精品 | av片子在线观看 | 在线精品一区二区 | 欧美性做爰猛烈叫床潮 | 九九综合久久 | 婷婷在线综合 | 奇米影视777影音先锋 | 久久深爱网 | 国产黄色在线观看 | 九色琪琪久久综合网天天 | av不卡免费看 | 精品国产一区二区三区久久久 | 亚洲丝袜一区二区 | 99久久婷婷国产 | 久久久999| 亚洲精品乱码久久久久久蜜桃91 | 免费情缘| 日韩av看片 | 久久久久久伊人 | 免费特级黄毛片 | 国产精品久久久久久爽爽爽 | 丁香花在线观看免费完整版视频 | 午夜在线免费观看 | 国产精品剧情在线亚洲 | 日本中文字幕观看 | 免费a视频 | 国产高清视频免费最新在线 | 夜夜操网 | 91看片在线免费观看 | 91精品国产一区二区在线观看 | 久久福利综合 | 国产老太婆免费交性大片 | 99久久精品久久亚洲精品 | 欧美一区二区在线免费看 | 狠狠干夜夜操 | 99热这里精品 | 国产日韩欧美综合在线 | 国产精品中文字幕在线 | 久久婷婷综合激情 | 婷婷综合伊人 | 精品国产一区二区三区日日嗨 | 91热视频 | 国产精品免费大片视频 | 曰本免费av | 中国一级片免费看 | 国产精品久久久久免费 | 手机av片| 操操操操网| 欧美精品天堂 | 色综合久久久久综合体桃花网 | 天天干天天玩天天操 | 91九色在线观看 | 欧美精品一区二区三区四区在线 | 国产精品18久久久久久首页狼 | 在线直播av | 九九免费精品视频 | 激情影院在线观看 | 97视频久久久 | 久射网 | 亚洲精品乱码白浆高清久久久久久 | 国外成人在线视频网站 | 在线观看黄色大片 | 国产一区二区在线播放视频 | 精品福利视频在线观看 | 91麻豆国产福利在线观看 | 久久久久中文字幕 | 亚洲成年人免费网站 | 美女黄久久 | 狠狠干综合网 | 91精选在线观看 | 久久久精品影视 | 97热在线观看 | 98福利在线| 又色又爽又黄 | 黄色片网站av | 天天玩天天干天天操 | 亚洲日本欧美在线 | 国产成人三级一区二区在线观看一 | 丁香在线观看完整电影视频 | 久久视频中文字幕 | 精品国产一区二区三区久久久久久 | 视频一区亚洲 | 美女搞黄国产视频网站 | 狠狠狠色丁香婷婷综合久久五月 | 在线观看国产91 | 人人爱人人舔 | 成人免费视频免费观看 | 久久丁香 | 国产资源网 | 国产亚洲一级高清 | 日日碰狠狠添天天爽超碰97久久 | 日日成人网 | 精品国产精品国产偷麻豆 | 亚洲精选视频免费看 | 国产999精品久久久久久绿帽 | 麻豆视频免费网站 | 韩国av在线播放 | 五月婷婷视频在线 | 国产亚洲一区 | 97网在线观看 | 福利一区二区 | 丁香激情综合久久伊人久久 | 久久av中文字幕片 | 日韩精品首页 | 欧美日韩国产xxx | 国产成人精品久久久久 | 午夜精品av| 国产精品毛片久久久久久久 | 日韩精品免费一区二区三区 | 永久精品视频 | 欧美日韩国产二区 | 天天摸天天弄 | 亚洲 欧美 成人 | 精品视频免费久久久看 | 日韩欧美精品一区 | 婷婷色中文网 | 日韩在线视频免费看 | 久久综合久久鬼 | 天干啦夜天干天干在线线 | 日本久久电影网 | 国产黄a三级三级三级三级三级 | 婷婷狠狠操 | 99热这里只有精品免费 | 五月开心六月伊人色婷婷 | 精品国产乱码久久久久久浪潮 | 奇米影视8888在线观看大全免费 | 婷婷丁香在线 | 最近免费观看的电影完整版 | 国产综合片 | 欧美a性 | 久久精品视频国产 | 丝袜美腿一区 | 色综合天天在线 | 亚洲情婷婷 | 91九色最新地址 | 欧美日本高清视频 | 欧美韩国日本在线 | 色久av| 黄色av一区二区三区 | 欧美精品一二 | 狠狠色丁香婷婷综合久久片 | 国产精品永久久久久久久久久 | 久草精品视频 | 日韩 在线观看 | 美女网站久久 | 欧美日韩精品在线免费观看 | 亚洲在线成人精品 | 亚州中文av | 公开超碰在线 | 国产黄色免费在线观看 | 国产精品久久免费看 | 九九在线视频免费观看 | 一级一片免费观看 | 欧美激情视频在线免费观看 | 国产高清小视频 | 国产黄色片一级三级 | 中文字幕在线日 | 精品999在线观看 | 国产资源网站 | 色a网 | 国产激情久久久 | 午夜99| 在线观看免费av片 | 日韩美一区二区三区 | 中文字幕在线观看一区二区 | 丁香激情网 | 久久久久| 欧美精品一区在线 | 国内精品中文字幕 | 在线有码中文 | 91污在线| 中文字幕精品在线 | 91av亚洲| 欧美日韩免费一区 | 999成人精品 | 天天色天天操天天爽 | 91人人爱| 激情五月婷婷丁香 | 日本中文字幕在线免费观看 | 免费在线一区二区 | 99这里只有精品视频 | 丁香六月婷婷激情 | 国产精品免费一区二区三区 | 欧美在线91 | 免费电影一区二区三区 | 日日夜夜婷婷 | 免费在线一区二区三区 | 国产免费xvideos视频入口 | 国产伦理一区二区三区 | 中文字幕在线专区 | 国产亚洲欧美在线视频 | 日韩在线观看av | 国产日韩精品一区二区三区在线 | a√国产免费a | 一区二区 不卡 | 狂野欧美激情性xxxx欧美 | 中文字幕在线影视资源 | 中文字幕在线视频国产 | 久久国产视频网站 | 亚洲专区视频在线观看 | 久久久国产精品免费 | 青青草国产精品视频 | 国产区网址 | 色婷婷伊人 | 久久草| 18女毛片| 波多野结衣亚洲一区二区 | 久草.com | 中文电影网| 999热线在线观看 | 国产精品乱码一区二区视频 | 中文字幕乱在线伦视频中文字幕乱码在线 | 国产无区一区二区三麻豆 | 黄色毛片观看 | 精品嫩模福利一区二区蜜臀 | 91女神的呻吟细腰翘臀美女 | 久久一区二区三区日韩 | 亚洲精品视频免费在线 | 国产在线国偷精品产拍免费yy | 久久久美女 | 综合在线亚洲 | 色综合久久综合网 | 欧美粗又大 | 99性视频 | 五月天六月婷婷 | 婷婷亚洲五月 | 日批视频在线 | 国产在线理论片 | 91精品久久久久久久久久入口 | 狠狠狠色丁香婷婷综合激情 | 欧美国产日韩在线视频 | 天天射天天干天天操 | 免费看一级片 | 国产一区视频免费在线观看 |