使用Schema类型在Java中创建SQL查询 – Kilo框架介绍

3,404次阅读
没有评论

共计 4716 个字符,预计需要花费 12 分钟才能阅读完成。

了解如何利用 Kilo 框架创建和消费 RESTful web 服务,以及与关系数据库交互和处理常见文件格式。阅读本文了解如何优化应用程序的占用空间和提高代码可读性。

Java,SQL,Schema

Kilo(以前称为 HTTP-RPC)是一个用于在 Java 中创建和消费 RESTful 和类似 REST 的 Web 服务的开源框架。它非常轻量级,只需要 Java 运行时环境和一个 Servlet 容器。整个框架的大小不到 150KB,因此在需要最小化占用空间的应用程序中非常理想。

本文首先介绍了两个基本的 Kilo 类,QueryBuilder 和 ResultSetAdapter。然后讨论了 Kilo 对“模式类型”的支持,这使得可以通过 Java 语言构造而不是纯文本来表达 SQL 查询。

换句话说,它们为查询提供了一种“领域特定语言”(DSL)。

获取 Kilo

Kilo 通过 Maven Central 进行分发:

  • org.httprpc:kilo-client – 包括支持消费 Web 服务,与关系数据库交互以及处理常见文件格式(要求 Java 17 或更高版本)(repo1.maven.org/maven2/org/httprpc/kilo-client/)

  • org.httprpc:kilo-server – 依赖于客户端;包括创建 Web 服务的支持(要求 Jakarta Servlet 规范 5.0 或更高版本)(repo1.maven.org/maven2/org/httprpc/kilo-server/)

QueryBuilder 和 ResultSetAdapter

Kilo 的 QueryBuilder 类提供了对程序化构建和执行 SQL 查询的支持。例如,给定以下 MySQL 示例数据库中的表:

create table pet (name varchar(20),
  owner varchar(20),
  species varchar(20),
  sex char(1),
  birth date,
  death date
);

这段代码可以用于创建一个查询,返回与特定所有者关联的所有行:

var queryBuilder = new QueryBuilder();

queryBuilder.append("select * from pet where owner = :owner");

冒号字符将“owner”标识为参数或变量。参数值可以作为参数传递给 QueryBuilder 的 executeQuery()方法,如下所示:

try (var statement = queryBuilder.prepare(getConnection());
    var results = new ResultSetAdapter(queryBuilder.executeQuery(statement, mapOf(entry("owner", owner)
    )))) {...}

Kilo 的 ResultSetAdapter 类通过 Iterable 接口提供对 JDBC 结果集内容的访问。适配器的迭代器生成 Map 实例来表示单个行。结果可以序列化为 JSON 或 CSV,也可以用作模板文档的数据字典。或者,它们可以映射到 Pet 实例的列表,并返回给调用者:

public interface Pet {String getName();
    String getOwner();
    String getSpecies();
    String getSex();
    Date getBirth();
    Date getDeath();}
return results.stream().map(result -> BeanAdapter.coerce(result, Pet.class)).toList();

模式 (Schema) 类型

QueryBuilder 还支持使用“模式类型”进行更结构化的查询构建方法。例如,上述查询可以使用模式类型编写如下:

public interface Pet {@Table("pet")
    enum Schema implements SchemaElement {@Column("name")
        NAME,
        @Column("owner")
        OWNER,
        @Column("species")
        SPECIES,
        @Column("sex")
        SEX,
        @Column("birth")
        BIRTH,
        @Column("death")
        DEATH
    }

    ...
}
var queryBuilder = QueryBuilder.selectAll()
    .from(Pet.Schema.class)
    .where(OWNER.eq("owner"));

模式类型是实现 SchemaElement 接口的枚举类型。它们分别使用 Table 和 Column 注解与表名和列名相关联。相比于字符串字面值,它们通常更可取,因为它们促进了代码重用,并对重构更具弹性。它们还可以提高可读性,特别是在处理复杂查询时。在上面的示例中,已经静态导入了 OWNER 常量以减少冗余。

还支持插入、更新和删除操作。例如,给定以下类型(表示假设产品目录中的一项):

@Description("Represents an item in the catalog.")
public interface Item {@Table("item")
    enum Schema implements SchemaElement {@Column("id")
        ID,
        @Column("description")
        DESCRIPTION,
        @Column("price")
        PRICE
    }

    @Key("id")
    @Description("The item's ID.")
    Integer getID();
    void setID(Integer id);

    @Description("The item's description.")
    @Required
    String getDescription();
    void setDescription(String description);

    @Description("The item's price.")
    @Required
    Double getPrice();
    void setPrice(Double price);
}

此服务方法可用于向 item 表添加一行:

@RequestMethod("POST")
@ResourcePath("items")
@Description("Adds an item to the catalog.")
@Creates
public Item addItem(@Description("The item to add.") Item item
) throws SQLException {var queryBuilder = QueryBuilder.insertInto(Item.Schema.class, DESCRIPTION, PRICE)
        .values("description", "price");

    try (var statement = queryBuilder.prepare(getConnection())) {queryBuilder.executeUpdate(statement, new BeanAdapter(item));
    }

    return getItem(BeanAdapter.coerce(queryBuilder.getGeneratedKeys().get(0), Integer.class));
}

类似地,以下两个方法可用于更新或删除现有项:

@RequestMethod("PUT")
@ResourcePath("items/?")
@Description("Updates an item.")
public void updateItem(@Description("The item ID.") Integer itemID,
    @Description("The updated item.") Item item
) throws SQLException {item.setID(itemID);

    var queryBuilder = QueryBuilder.update(Item.Schema.class, DESCRIPTION, PRICE)
        .set("description", "price")
        .where(ID.eq("id"));

    try (var statement = queryBuilder.prepare(getConnection())) {queryBuilder.executeUpdate(statement, new BeanAdapter(item));
    }
}
@RequestMethod("DELETE")
@ResourcePath("items/?")
@Description("Deletes an item.")
public void deleteItem(@Description("The item ID.") Integer itemID
) throws SQLException {var queryBuilder = QueryBuilder.deleteFrom(Item.Schema.class).where(ID.eq("id"));

    try (var statement = queryBuilder.prepare(getConnection())) {
        queryBuilder.executeUpdate(statement, mapOf(entry("id", itemID)
        ));
    }
}

更多相关信息,可以查看右边链接,项目文件(github.com/HTTP-RPC/Kilo/blob/master/README.md)文章来源地址 https://www.toymoban.com/diary/java/687.html

到此这篇关于使用 Schema 类型在 Java 中创建 SQL 查询 – Kilo 框架介绍的文章就介绍到这了, 更多相关内容可以在右上角搜索或继续浏览下面的相关文章,希望大家以后多多支持 TOY 模板网!

原文地址:https://www.toymoban.com/diary/java/687.html

如若转载,请注明出处:如若内容造成侵权 / 违法违规 / 事实不符,请联系站长进行投诉反馈,一经查实,立即删除!

    正文完
     0
    Yojack
    版权声明:本篇文章由 Yojack 于1970-01-01发表,共计4716字。
    转载说明:
    1 本网站名称:优杰开发笔记
    2 本站永久网址:https://yojack.cn
    3 本网站的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,请联系站长进行删除处理。
    4 本站一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
    5 本站所有内容均可转载及分享, 但请注明出处
    6 我们始终尊重原创作者的版权,所有文章在发布时,均尽可能注明出处与作者。
    7 站长邮箱:laylwenl@gmail.com
    评论(没有评论)