# MJplus
**Repository Path**: gzdjava/MJplus
## Basic Information
- **Project Name**: MJplus
- **Description**: 模块化的jplus,打造一个最合适的J2EE框架
- **Primary Language**: Java
- **License**: Not specified
- **Default Branch**: master
- **Homepage**: http://139.196.18.60
- **GVP Project**: No
## Statistics
- **Stars**: 0
- **Forks**: 4
- **Created**: 2017-07-10
- **Last Updated**: 2020-12-18
## Categories & Tags
**Categories**: Uncategorized
**Tags**: None
## README
## Jplus-core说明 ##
### 声明 ###
jplus 是一个java框架,web只是他的一个模块,他并非重复造轮子,初衷是希望把所有好的合适的框架集合到一起,能发挥出对开发者更大的作用。
框架思想主要学习黄勇 的smart,推荐大家去看看;
### 关于Jplus ###
1. 支持零配置,所有配置可由.properties或代码管理。
2. 提供可以媲美Spring的 IOC、AOP、MVC,及各种强大的第三方定制插件。
3. 快速开发,框架~200Kb,所有插件可插拔。
### jplus 简介 ###

> **项目启动顺序:**
> 初始化 -->CoreLoader -->BeanHandle -->ActionHandle -->PluginHandle -->AopHandle -->IocHandle -->完成
>
> **关于Bean管理**
> 项目启动后CoreLoader 加载 BeanHandle,同时BeanHandle 初始化ConfigHandle, 获取一系列相应的用户配置信息properties。
> 然后扫描 ${app.scan.pkg} 包中所有带@Component注解的class,完成bean的发现。
> *PS:在项目中运行时获取 bean 对象可以使用:BeanHandle.getBean(Classcls);*
----------
### 1. 快速创建一个jplusMvc项目 ###
1. 首先我们创建一个simple Maven Project
然后第一步创建一个**HelloAction.java**用于接收用户请求:
```java
package com.demo.mvc.action;
import com.jplus.core.mvc.WebUtil;
import com.jplus.core.mvc.annotation.Controller;
import com.jplus.core.mvc.annotation.Request;
@Controller
public class HelloAction {
@Request.All("{msg}")
public void index(String msg) {
System.err.println("== This is Restful Action ===========");
WebUtil.writeHTML(msg);
}
}
```
2. 然后我们看见,包名是com.demo开始的,现在配置注解扫描配置,在project的classpath目录下,添加文件:**app.properties**
```java
app.scan.pkg=com.demo
```
3. [附加]其实到第二步,项目就可以启动了,但为了debug方便,添加日志配置,在classpath目录下,添加:**log4j.xml**
```xml
```
4. 最后添加Tomcat容器,启动访问:[localhost:8080/hello](http://localhost:8080/hello)试试吧~
### 2. 关于JplusMvc的相关配置 ###
1. 关于MVC注解
- 注解:**@Controller**
使用说明:用于定义Action配置。类上加上此注解,该类将被定义为Action处理的接收类。
使用方式:直接在请求处理类上加上此注解即可。
- 注解:**@Request**
使用说明:用于定义请求的UrlMapping,对用户的请求根据路径做路由分发。
使用方式:见下方代码,默认提供六种配置方式,对应于不同的http请求:
@Request("xxx") => 用于公共请求前缀,作用于类上
@Request.All("xxx")=> 用于接受所有匹配的请求,作用于方法上
@Request.Get("xxx")=> 用于接受Http get请求,作用于方法上
@Request.Post("xxx")=> 用于接受Http post请求,作用于方法上
@Request.Put("xxx")=> 用于接受Http put请求,作用于方法上
@Request.Delete("xxx")=> 用于接受Http delete请求,作用于方法上
```java
package com.demo.mvc.action;
import com.jplus.core.mvc.WebUtil;
import com.jplus.core.mvc.annotation.Controller;
import com.jplus.core.mvc.annotation.Request;
@Controller
@Request("user/")
public class UserAction {
@Request.All("info")
private void info() {
WebUtil.writeHTML("This is Page info");
}
@Request.Get("{id}")
private void getUser(int id) {
WebUtil.writeHTML("This is a get user info by id,id:"+id);
}
@Request.Post("add")
private void addUser() {
WebUtil.writeHTML("This is add Page");
}
@Request.Put("edit")
private void editUser() {
WebUtil.writeHTML("This is edit Page");
}
@Request.Delete("{id}")
private void delUser(int id) {
WebUtil.writeHTML("This is del Page,user id is :" + id);
}
}
```
- 注解:**@Param**
使用说明:用于设置请求入参,通过入参名称,获取restful/get/post形式的参数。
使用方式:见下方代码:
```java
/**
* 发现和上面getUser的方法的区别了么,
* 上面直接取值,使用的下标取值,取的restful请求中的占位符{xx}中的值,然后按下标依次取,支持类型:int/long/Double/BigDecimal/String
* 下面通过@Param取值,使用名称key取值,可以取restful/get/post...等各种方式提交的值,支持类型:int/long/Double/BigDecimal/String
*/
@Request.Get("{id}")
private void getUser2(@Param("id") int userId) {
WebUtil.writeHTML("This is a get user info by id,id:"+userId);
}
```
- 注解:**@Form**
使用说明:用于标示POJO实体对象的请求入参,作用于实体类上。
使用方式:当一个pojo实体对象为入参时(form表单提交),框架会自动将入参填充到对象中去。见下方代码:
```java
==>UserForm.java
@Form
public class User {
private int id;
private String name;
private String age;
/** ... omit get/set/toString method ... */
}
--------------------------------------------------------------
==> UserAction.java
@Request.Post("add")
private void addUser(User user) {
System.err.println("I can get form param:"+ user.toString());
WebUtil.writeHTML("This is add Page");
}
```
- 注解:**@Service**
使用说明:用于标记实现类,供框架扫描,作为IOC注入到其他类中,作用于类上,一般用于Service层。又一个默认参数[名称],非必填。
使用方式:@Service用于标记扫描,等待其他类使用@Autowired 注入,
@Service可以添加参数名 @Service("xxx"),然后注入的地方使用@Autowired("xxx")进行注入,见下方代码:
```java
==> UserService.java
@Service
public class UserService {
public Result add(User form) {
Result res = new Result().OK();
System.err.println("Add Service :" + form.toString());
return res;
}
}
--------------------------------------------------------------
==> UserAction.java
@Controller
@Request("user/")
public class UserAction {
private @Autowired UserService userService;
@Request.Post("add")
private Result addUser(User form) {
return userService.add(form);
}
}
```
2. 关于MVC上下文
- **DataContext.java**
DataContext为整个MVC的上下文容器,生产周期为当前请求。
里面封装了**Cookie,HttpServletRequest,HttpServletResponse,ServletContext,Session**等相关对象的处理。
- **WebUtil.java**
Web 操作工具类,依赖于DataContext,提供了取参,输出,转发请求等实用功能。
- **视图输出:View.java**
视图输出,将请求执行完毕,返回到JSP页面进行渲染,然后返回也客户端页面。
默认视图地址[在properties中配置]:*app.view.path=/WEB-INF/jsp/*
```java
@Controller
@Request("hello/")
public class HelloAction {
@Request.All("page")
public View page() {
System.err.println("== This is page Action ===========");
//跳转到 : /WEB-INF/jsp/hello/page.jsp
return new View();
}
}
```
- **JSON输出:Result.java**
JSON输出,将请求执行完毕,返回到前端JSON字符串。
推荐使用Result作为方法返回值。用户也可以**自定义任何对象**输出,默认都为JSON格式。
```java
@Request.Get("JSON")
public Result getJSON() {
System.err.println("== This is page Action ===========");
// 返回JSON : {"code":1,"obj":"请求成功"}
return new Result().OK().DATA("请求成功");
}
```
### 3. 关于JplusAop的相关配置 ###
> 说明:使用jdk,Cglib动态代理技术实现,采用责任链模式,递归代理。
> 默认AOP提供三种代理:
1. **AspectProxy.java** 切面代理,提供6种切点方法:
```java
//开始
public void begin()
//拦截[根据boolean判断是否继续执行方法]
public boolean intercept(Class> cls, Method method, Object[] params)
//方法执行前
public void before(Class> cls, Method method, Object[] params) throws Throwable
//方法执行后
public void after(Class> cls, Method method, Object[] params, Object result)
//抛出异常
public void error(Class> cls, Method method, Object[] params, Throwable e)
//结束
public void end()
```
- **使用方式:**
对自定义切面类继承AspectProxy抽象类,并重写目标方法, 然后对该类添加@Aspect注解。
- **注解:@Aspect** *(该注解有三个属性值:)*:
**pkg**:想要被代理类所在的包名。
**cls**:想要被代理的具体类,和pkg二选一。
**annotation**:想要被代理的类是有指定注解的类
- **注解:@AspectOrder** :
该注解是@Aspect的补充,对链式AOP做排序,若有@AspectOrder注解,则优先比较(序号的值越小越靠前)
- **代码示例:**
```java
package com.demo.mvc.aop;
import java.lang.reflect.Method;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.jplus.core.aop.AspectProxy;
import com.jplus.core.aop.annotation.Aspect;
import com.jplus.core.mvc.DataContext;
import com.jplus.core.mvc.annotation.Controller;
/**
* AOP拦截器演示
* 测试拦截指定包名下所有的Action
* @author Yuanqy
*/
@Aspect(pkg = "com.demo.mvc.action", annotation = Controller.class)
public class ActionInterceptor extends AspectProxy {
private Logger logger = LoggerFactory.getLogger(getClass());
ThreadLocal curTimes = new ThreadLocal();
@Override
public void before(Class> cls, Method method, Object[] params) throws Throwable {
curTimes.set(System.currentTimeMillis());
logger.info("===Is new RequestURI:" + DataContext.getRequest().getRequestURI());
logger.info("===Action:{}.{}", cls.getName(), method.getName());
}
@Override
public void after(Class> cls, Method method, Object[] params, Object result) throws Throwable {
logger.info("===Action is complete,time:{}ms \n", (System.currentTimeMillis() - curTimes.get()));
curTimes.remove();
}
}
```
1. **TransactionProxy.java** 事务代理:
- **使用方式:**
在想要被事务代理的类、或类的方法上添加注解:**@Transaction**即可。
- **注解:@Transaction**:
类似于Spring的事务,支持事务传播,支持配置事务隔离级别(默认level=4)
作用于类上,表示当前类中所有方法开启事务。
作用于方法上,表示当前方法开启事务。
- **代码示例:**
```java
@Service
public class UserService {
@Transaction
public Result add(User form) {
Result res = new Result().OK();
// TODO something...
return res;
}
}
```
1. **PluginProxy.java** 插件代理:
- **使用方式:**
他是对于上诉两种代理的补充,可以用于扩展任意规则、任意场景代理实现。
该类仅有一个获取代理链方法:
```java
/**
* 具体代理被执行时,会执行实现了Proxy接口的doProxy()方法.
* 使用方式:请参考:com.jplus.j2cache.proxy.J2CachePluginProxy
* 原理:
* 项目启动时,先扫描缓存bean,当AopHandle运行时,执行插件代理的getTargetClassList方法。
* 该方法由开发人员自己定义筛选规则,将满足条件的对象bean集合返回,做AOP代理;
* 在项目运行期,代理的对象执行时,通过链式代理,找到当前类,执行doProxy()方法,完成代理操作。
*/
public abstract ListgetTargetClassList()
```
### 4. 关于JplusIOC的相关配置 ###
> 框架中,ioc初始化位于项目启动最后阶段。 之前的MVC,AOP,Plugin 已经是创建了所有项目中所需的bean对象, 在最后要做的就是把这些 对象 互相关联起来,建立原本的依赖关系。
- **注解:@Autowired**:
- **使用方式:**
```java
//通过接口注入,框架会找到容器中的第一个接口实现类进行注入,实现类必须>=1,否则异常
private @Autowired IIocDemo demo;// 通过接口注入
//通过指定接口实现注入,实现类上必须标明当前名称@Service("xxx")才能注入成功
private @Autowired("xxx") IIocDemo demo;// 通过指定接口实现注入
//这个没什么好说的,直接注入目标类即可
//PS.只要是bean容器管理了的对象,都可通过@Autowired注入。
private @Autowired IocDemoImpl demo;// 通过指定对象注入
```
- **注解:@Value**:
- **使用方式:**
```java
//方式1:@Value("xxx")
//方式2:@Value("${xxx}") 两种方式效果一致
private @Value("app.scan.pkg") String scanPkg;// 注入配置文件
```
- **注解:@Component**:
@Component注解是其他注解的元注解。jplus的扫描规则是扫描所有继承或使用Component注解的类,然后加入bean管理。
- **使用方式:**
```java
/**
* 属于Bean扫描注解,任何需要加入IOC的组件类,都 可以添加该注解。
* 该注解提供两个参数[都非必填项]:
* initMethod:填入当前类的方法名,项目启动后会执行该方法
* destroyMethod:填入当前类的方法名,项目注销前会执行该方法
*/
@Component(initMethod = "init", destroyMethod = "destroy")
public class InitComponent {
public void init() {
System.err.println("== 项目启动后会执行该方法 ==");
}
public void destroy() {
System.err.println("== 项目注销前会执行该方法 ==");
}
}
```
- **代码示例:**
```java
/**
* 接口类
*/
public interface IIocDemo {
Result addUser(User user);
Result getUser(int id);
}
```
---------
```java
/**
* 实现类
*/
//@Service
@Service("iocDemoImpl")
public class IocDemoImpl implements IIocDemo {
private Map map = new HashMap<>();
@Override
public Result addUser(User user) {
map.put(user.getId(), user);
return new Result().OK().DATA("新增成功");
}
@Override
public Result getUser(int id) {
if (map.containsKey(id))
return new Result().OK().DATA(map.get(id));
else
return new Result().FAIL().DATA("数据不存在");
}
}
```
---------
```java
/**
* 在Action中注入
*/
@Controller
@Request("ioc/")
public class IocAction {
private @Autowired IIocDemo ioc1; // 通过接口注入
private @Autowired("iocDemoImpl") IIocDemo ioc2; // 通过指定接口实现注入
private @Autowired IocDemoImpl ioc3; // 通过指定对象注入
private @Value("app.scan.pkg") String scanPkg;
@Request.Get("test")
private void iocTest() {
System.out.println(ioc1.addUser(new User(1, "张三")));
System.out.println(ioc2.getUser(1));
System.out.println(ioc3.getUser(1));
WebUtil.writeTEXT("OK");
}
}
```
### 5. 关于Jplus Junit的相关配置 ###
> jplus框架默认集成 **Junit4** 单元测试框架。
- **使用方式:**
测试类添加两个注解:@RunWith [Junit框架注解]、@ContextConfiguration [Jplus框架注解] 即可。
RunWith注解大家可以百度下使用方式,现在说明下ContextConfiguration注解。
- **注解:@ContextConfiguration**
ContextConfiguration用于配置配置文件路径。通过参数locations配置,说明如下:
"file:/app.properties" //系统根目录
"app.properties" //项目根目录
"/app.properties" //项目WEB-INF下
"classpath:app.properties" //项目ClassPath下
//如果有多个配置文件,可以用 ; 分号分割,如下:
"file:/db.properties;classpath:app.properties" //多个配置文件.一个文件目录下;一个classpath目录
- **代码示例:**
```java
/**
* JplusJunit4 测试框架IOC功能
*/
@RunWith(JplusJunit4Runner.class)
@ContextConfiguration(locations = "classpath:app.properties")
public class JunitIOC {
private @Autowired("iocDemoImpl") IIocDemo ioc3;
private @Autowired IocDemoImpl ioc2;
private @Value("app.scan.pkg") String scanPkg;
@Test
public void test() throws InterruptedException {
System.out.println("添加用户:" + ioc3.addUser(new User(1001, "Jplus")));
System.out.println("获取用户:" + ioc2.getUser(1001));
System.out.println("app.scan.pkg=" + scanPkg);
Thread.currentThread().join();
}
}
```
### 6. 关于JplusMVC的后端参数验证 ###
> 对于请求入参,JPlus可以通过拦截器做公共参数校验,对于form表单对象,可以使用工具类进行校验。
- **使用方式:**
对于form表单对象(可以理解为添加了@Form注解的pojo类),当该对象作为请求入参时,正常情况参数的校验都是固定的字段,固定的规则。而且开发人员也不想在业务代码里面进行入参判断。这时候就可以使用:**Result res= VerifyUtil.doVerify(form);**通过返回Result判断校验结果。
原理:对入参对象的字段进行处理,判断是否有@V注解,对@V注解的字段进行判断处理。
- **注解 @V:**
注解名|默认表达式|默认返回描述|是否可为空
:----|:-----|:-----|:-----:
IsNotNull|\\w+|字段{}不能为空|FALSE
IsCN|^[\u4e00-\u9fa5]+$|字段{}不是汉字|TRUE
IsDigits|^[0-9]*$|字段{}不是纯数字|TRUE
IsFloat|^[0-9]*$|字段{}不是小数|TRUE
IsEmail|^\\w+([-+.]\\w+)*@\\w+([-.]\\w+)*\\.\\w+([-.]\\w+)*$|邮箱格式不正确|TRUE
IsIdCardNo|^(\\d{6})()?(\\d{4})(\\d{2})(\\d{2})(\\d{3})(\\w)$|身份证号不正确|TRUE
IsPhone|^[1][0-9]{10}$|手机号码不正确|TRUE
IsPwd|^(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]{6,16}$|密码格式不正确|TRUE
IsTel|^(0\\d{2}-\\d{8}(-\\d{1,4})?)\|(0\\d{3}-\\d{7,8}(-\\d{1,4})?)$|电话号码不正确|TRUE
IsZipCode|^[0-9]{6}$|邮政编码不正确|TRUE
Custom| |未定义|TRUE
*PS:@V.Custom()用于给用户自定义校验规则及返回说明。还支持JPEL表达式。*
- **代码示例:**
```java
/**
* Form入参表单
*/
@Form
public class User {
@V.IsNotNull
@V.IsDigits
private int id; //表示id必须为数字,不能为空
private String name;
@V.Custom(regex = "el:{}>=18 && {}<30", retmsg = "年龄必须在18~30岁之间",isNull=false)
private String age; //表示 自定义用JPEL表达式判断年龄范围,不能为空
@V.Custom(regex="^[A-Za-z]+$",retmsg="请输入正确的URL")
private String url; //表示 自定义用正则表达式判断
@V.IsPhone
private String phone; //表示 必须为手机号,可为空
}
//======================================================
@Controller
@Request("user/")
public class UserAction {
private @Autowired UserService userService;
@Request.Post("add")
private Result addUser(User form) {
//执行入参验证
Result res = VerifyUtil.doVerify(form);
if (res.IsOK())
return userService.add(form);
return res;
}
}
```
### 7. 附加:关于Jplus的JPEL表达式的使用 ###
- **使用逆波兰算法实现,相比js引擎,效率提升>15倍,后续会逐渐优化,支持特性:**
- 算数运算表达式:
加(+)、减(-)、乘(*)、除(/)、求余(%)、幂(^)运算
- 关系表达式:
等于(==)、不等于(!=)、大于(>)、大于等于(>=)、小于(<)、小于等于(<=)
- 逻辑表达式:
且(&&)、或( || )、非( ! )、true、false
- 括号优先级表达式:
使用“(表达式)”构造,括号里的具有高优先级。
- 不支持三目运算,不支持文本字符串
- **使用方式:**
```java
//参见代码:com.jplus.core.util.JPEL.java
String calc = "1+2*(3+4)-5+(-2)*8/2+8 == 5*6/2-5 && 10%4*3>-1+2*3";
System.err.println("中缀表达式:" + calc);
// == 逆波兰 ========================
long t = System.currentTimeMillis();
System.out.println("逆波兰=" + excute(calc) + "\ttime:" + (System.currentTimeMillis() - t));
// == JS引擎==========================
t = System.currentTimeMillis();
System.out.println("JS引擎=" + getSE().eval(calc) + "\ttime:" + (System.currentTimeMillis() - t));
```
### 8. 附加:关于Jplus多数据源的配置 ###
> 多数据源可以支持**申明式配置**和**编程式配置**两种,具体方法参考jplus-orm说明文档。
- **编程式配置 实例:**
```java
/**
* JplusJunit4 测试Jplus-core编程式多数据源切换
*/
@RunWith(JplusJunit4Runner.class)
@ContextConfiguration(locations = "classpath:app.properties")
public class JplusDDS {
@Test
public void test() {
JdbcTemplate jt = null;
try {
// 1.创建N个数据源,可选用C3p0,Druid...都可以
DataSource ds1 = new C3p0Plugin("jdbc:mysql://139.196.18.60:3306", "root", "password", "com.mysql.jdbc.Driver").getDataSource();
DataSource ds2 = new C3p0Plugin("jdbc:mysql://192.168.1.16:3306", "root", "password", "com.mysql.jdbc.Driver").getDataSource();
// 2.将N个数据源放入DynamicDataSource中,并定义key名称
Map map = new HashMap();
map.put("readone", ds1);
map.put("readtwo", ds2);
DynamicDataSource dds = new DynamicDataSource(map);
// 3.创建一个JdbcTemplate,使用数据源为DynamicDataSource
jt = new JdbcTemplate(dds);
// 4.循环测试数据源切换[ps.我事先准备了两个不同版本的mysql]
for (int i = 0; i < 3; i++) {
if (i == 2)
jt.openTransaction(); // 最后一次开启事务
DataSourceHolder.setDbType("readone");
System.err.println("[切换1]:" + JSON.toJSONString(jt.query("SHOW VARIABLES LIKE 'version';", new Object[] {})));
DataSourceHolder.setDbType("readtwo");
System.err.println("[切换2]:" + JSON.toJSONString(jt.query("SHOW VARIABLES LIKE 'version';", new Object[] {})));
if (i == 2)
jt.commit(); // 提交事务
}
} catch (Exception e) {
e.printStackTrace();
if (jt != null)
jt.rollback(); // 回滚事务
}
}
}
```