ServiceFramework作为Java Web框架都有哪些不错的设计

栏目: Java · 发布时间: 6年前

内容简介:最近需要开发一个纯API的项目,ServiceFramework算是一个古老的,基于Java的web框架了。我印象中应该是我11年的作品,那个时候应该是RubyOnRails正火的时候。我做了一段时间Rails程序员,后面转型做搜索,期间觉得没啥好用的Web框架,于是就开发了ServiceFramework。早年Java语言的笨拙一直是广受诟病的,业务还没两行,代码和配置就已经几百上千行了。首先我们不可能改变这门语言,那么如何做到极致简约呢?自动生成源码的套路肯定不行,用户就天天通过各种命令生成源码去了,

前言

最近需要开发一个纯API的项目, mlsql-cluster ,从无到有,到最后完整的proxy功能开发完毕,只花了四个小时不到,自己不尽小感叹了一把 ServiceFramework 的高效。

关于ServiceFramework的诞生

ServiceFramework算是一个古老的,基于 Java 的web框架了。我印象中应该是我11年的作品,那个时候应该是RubyOnRails正火的时候。我做了一段时间Rails程序员,后面转型做搜索,期间觉得没啥好用的Web框架,于是就开发了ServiceFramework。

极致简约的要求

早年Java语言的笨拙一直是广受诟病的,业务还没两行,代码和配置就已经几百上千行了。首先我们不可能改变这门语言,那么如何做到极致简约呢?自动生成源码的套路肯定不行,用户就天天通过各种命令生成源码去了,而且通常生成的源码又丑又难看,还不敢改,所以我们需要无声无息的为用户生成必要的代码, 并且还不能让用户看见,还需要兼顾IDE的代码提示。那么,应该怎么玩呢? 核心在于两个点,我们后续会展开讲:

运行时代码生成(codegen,功能增强)+ 父类方法签名(代码提示)

极致简约体现在哪

应用包含容器

早年几乎清一色的,代码都是跑在容器里的(weblogic,tomcat等)。在11年的时候,SF做出了一个重要的设计,就是http只是代码对外暴露的一个交互方式,和RPC一样,Web容器只是你运行代码里的一个组件而已。所以SF的启动是这样的(演示代码都是用Scala写的哈):

ServiceFramework作为Java Web框架都有哪些不错的设计

image.png

就是一个普通的Main方法。大家有没有发现现在大部分Web框架已经都这么干了。

配置文件精简

早年Java领域出现了一个潮流,就是能配置的坚决不写代码,配置可以更灵活,但是它们忘了配置本身也是一种代码(语法受限的语言),反倒增加了成本,所以后面引入了Annotation以及约定俗成。SF设计之初,就只有两个配置文件,一个application.yml,一个logging.yml文件。基本需要配置的很少。核心就是一个数据库配置信息,然后一个http端口。

自动读取数据库配置ORM

个人感觉对数据库的操作很难比SF更简化了(吹牛)。在SF中ORM是无任何配置文件的,唯一的信息就是在application.yml里的链接信息:

ServiceFramework作为Java Web框架都有哪些不错的设计

image.png

接着你按传统的方式在数据库里建好表,比如

CREATE TABLE backend
(
  id                   int(11) NOT NULL AUTO_INCREMENT,
  name                 varchar(255) DEFAULT NULL,
  url                  text,
  tag                  text,
  ecs_resource_pool_id int(11),
  PRIMARY KEY (id)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

创建了一张backend表。然后我要在代码中怎么操作呢?一行代码就是一个Model。

public class Backend extends Model {
}

什么都没有啊? 就这么个类,我们看看怎么操作数据库,首先是新建一条记录:

Map<String, Object> newParams = new HashMap<>();
newParams.put(......)
Backend backend = Backend.create(newParams);
backend.save();

找到并且删除一条记录:

Backend backend = Backend.where(map("name","jack")).fetchOne()
backend.delete()

那没有申明属性,怎么访问属性呢?

String name = backend.attr("name",String.class)

当然,因为我经常会用name属性,我们就申明一下,方便代码提示,不申明可以通过attr去定向拿:

public class Backend extends Model {
        private String name;
        public String getName() {
              return name;
         }
}

记住,这里的代码纯粹是为了做代码提示,不是必须的。

你可能会问,ORM里的关联咋办?

public class Backend extends Model {
       @OneToMany
       private List< Info>  infos;
       public Association infos(){
           throw new AutoGeneration();
        }
}

这就是所有,接着你就可以通过类似 backend.infos.where(....).fetch() 来操作。你不用写任何逻辑代码,ORM会根据你的数据库读取到的元数据自动帮你做关联,自动填充属性,自动提供查询语法(代码提示通过Model类已经写好的方法完成)

Web Contorller,一切只为便捷。

写一个controller 以及一个action,你只要这么做:

class BackendController extends ApplicationController {
   @At(path = Array("/backend/add"), types = Array(GET, POST))
  def backendAdd = {
    List("url", "tag", "name").foreach(item => require(hasParam(item), s"${item} is required"))
    Backend.newOne(params(), true)
    BackendService.refreshCache
    render(map("msg", "success"))
  }
}

继承ApplicationController,就是一个controller,@就定义了请求的endpoint。http参数怎么获取?使用param方法:

val name = param("name") 
paramAsInt("times",0) // 获取int类型参数,并且默认值设置为0
hasParam("name")//判断有没有
params() //拿到所有参数

比如前面的例子,我们鼓励直接在controller里使用模型类操作数据库,免去了service的麻烦,因为model已经具有足够的表达能力,很多业务逻辑也可以放在model里。

这不比通过定义方法的参数强很多?定义方法的参数你会说便于测试,我们看SF怎么做接口测试的:

@Test
  public void search() throws Exception {
      RestResponse response = get("/doc/blog/search", map(
              "tagNames", "_10,_9"
      ));
      Assert.assertTrue(response.status() == 200);
      Page page = (Page) response.originContent();
      Assert.assertTrue(page.getResult().size() > 0);
  }

So Easy.

基于HTTP协议的伪RPC协议

越来越多的人喜欢HTTP协议而非PPC, PRC无论测试还是复杂度其实都大于HTTP,但是每次调用HTTP接口还是很麻烦的,SF提供了一个对HTTP自动包装的接口(动态生成代理类的方式),你只要提供HTTP接口代码,就可以直接使用。比如:

trait BackendService {
  @At(path = Array("/run/script"), types = Array(GET, POST))
  def runScript(@Param("sql") sql: String): HttpTransportService.SResponse

  @At(path = Array("/run/script"), types = Array(GET, POST))
  def runScript(params: Map[String, String]): HttpTransportService.SResponse

  @At(path = Array("/run/sql"), types = Array(GET, POST))
  def runSQL(params: Map[String, String]): HttpTransportService.SResponse

  @At(path = Array("/instance/resource"), types = Array(GET, POST))
  def instanceResource(params: Map[String, String]): HttpTransportService.SResponse
}

这个接口是没有任何实现的,他对应的是后端一个服务的http接口。接着我们在SF里就可以这么调用了:

val instance = ClientProxy.get[BackendService]()
instance.runScript(params().asScala.toMap)

是不是很easy,很PRC?

后话

使用SF,你只需要几分钟就能搭建一个可以运行,具备部分业务逻辑功能的API服务。去掉尽量多的层,尽量让使用者可以用最简单的办法去完成对应的功能而不是去考虑一些设计的优雅性来完成一些功能特点。 大家可以查看 mlsql-cluster 获得更多使用范例,感受其魅力。

另外,我个人认为比较完美的一个组合是: Reactjs + ServiceFramework :grin: . React是一个我很让我有愿意去写代码的的前端框架。


以上所述就是小编给大家介绍的《ServiceFramework作为Java Web框架都有哪些不错的设计》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

编码的奥秘

编码的奥秘

Charles Petzold / 伍卫国、王宣政、孙燕妮 / 机械工业出版社 / 2000-9-1 / 24.00

渴望交流是大多数人的天性。在本书中,“编码”通常指一种在人和机器之间进行信息转换的系统。换句话说、编码即是交流。有时我们将编码看得很神秘,其实大多数编码并非都是这样。大多数的编码都需要被很好地理解,因为它们是人类交流的基础。――《编码的奥秘》 手电筒、英国人入侵、黑色的猫和跷跷板与计算机有什么必然联系?本书向我们展示了使用语言的一些直观方法并创造新的方法来进行相互之间的交流。此书使我们明白了......一起来看看 《编码的奥秘》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

MD5 加密
MD5 加密

MD5 加密工具

正则表达式在线测试
正则表达式在线测试

正则表达式在线测试