TAC是一个基于java的微服务容器,提供从业务代码编写、编译、发布、jar动态加载、运行等一系列经常使用开发流程的支持,是天猫App在服务端开发模式下的新尝试。TAC和客户端框架Tangram结合,极大提升开发效率;TAC目前在天猫App、手机淘宝特价版普遍使用;java
天猫App(如下简称猫客)首页从2015年的坑位运营走向2016年的全面个性化,当时猫客首页的个性化业务多大50多处。以首页为例,这个过程当中除了接入导购链路的二方服务以外,接入了大量的三方服务;同时咱们发现:android
当时微服务的概念已经很火,同时阿里内部也开始大力推广docker(如今大部分应用都跑在docker上)。咱们想过将庞大的首页应用拆分红微服务,可是和基础服务不一样,前台业务变化快,修改频繁,拆分以后开发同窗依然要面临各类服务接入等体力劳动,同时须要维护拆分以后的多个应用,反而增长了劳动成本,所以拆分红微服务的方式治标不治本;ios
在此背景下TAC孕育而生,它提供低成本开发与发布流程、低成本搭建与维护开发环境、高稳定性保障;TAC经过热部署的方式使得研发同窗够从苦力劳动中解放出来,回归到业务开发中去,同时一个基础服务接入以后,可以提供给多个业务使用;在此模式下,业务可以进行更细粒度的拆分,且故障隔离业务A的改动不会影响业务B;git
在TAC的帮助下,频繁修改的新业务可快速上线,不会出现由于修改一个字段、几行代码就须要从新发布整个应用的状况;同时与tangram结合,实现页面卡片、坑位的快速调整;github
通过近三年的沉淀,咱们今日放出了开源版本,将集团版本的TAC剥离与阿里相关的中间件、网络、协议、部署环境等,保留其核心功能;开源版本提供了编译、热加载、运行的基础能力;redis
上图是tac的类加载器结构,每一个线上的微服务实例都经过一个新的classloader加载,同时为了方便用户扩展新的数据源,在AppClassLoader上扩展了一个classloader以加载第三方数据源(固然也能够直接在代码中扩展,见tac-infrastructure);docker
java -jar tac-container.jar
复制代码
java -jar tac-console.jar --admin
复制代码
http://localhost:7001/#/tacMs/list
复制代码
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>tac-sdk</artifactId>
<version>${project.version}</version>
</dependency>
复制代码
public class HelloWorldTac implements TacHandler<Object> {
/**
* 引入日志服务
*/
private TacLogger tacLogger = TacServiceFactory.getLogger();
/**
* 编写一个实现TacHandler接口的类
*
* @param context
* @return
* @throws Exception
*/
@Override
public TacResult<Object> execute(Context context) throws Exception {
// 执行逻辑
tacLogger.info("Hello World");
Map<String, Object> data = new HashMap<>();
data.put("name", "hellotac");
data.put("platform", "iPhone");
data.put("clientVersion", "7.0.2");
data.put("userName", "tac-userName");
return TacResult.newResult(data);
}
}
复制代码
本地编译、打包json
发布及测试api
正式发布bash
线上验证
curl http://localhost:8001/api/tac/execute/helloworld -s|json
复制代码
{
"success": true,
"msgCode": null,
"msgInfo": null,
"data": {
"helloworld": {
"data": {
"name": "hellotac",
"clientVersion": "7.0.2",
"userName": "tac-userName",
"platform": "iPhone"
},
"success": true,
"msCode": "helloworld"
}
},
"hasMore": null,
"ip": "127.0.0.1"
}
复制代码