Compare commits
23 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
2224e0d212 | ||
|
1a6006fe36 | ||
|
12c3b5fdb5 | ||
|
f065faf4e7 | ||
|
b4e277fcc0 | ||
|
bbc1e24b87 | ||
|
34c1a9e53a | ||
|
f6a855b6bf | ||
|
4377d19e2f | ||
|
820f2806ec | ||
|
808bdf6079 | ||
|
eac1d8d055 | ||
|
d5a88b8be4 | ||
|
fd5d554171 | ||
|
6d18be3df8 | ||
|
869824e2bb | ||
|
84573be584 | ||
|
3cc94a5b32 | ||
|
3a4f4f4495 | ||
|
d250b431a4 | ||
|
5e133e7de5 | ||
|
ef39843a25 | ||
|
0e5f3b7249 |
21
README.md
21
README.md
@@ -2,15 +2,25 @@
|
||||
|
||||
Enjoy 是基于 Java 语言的极轻量极魔板引擎。极轻量级仅 171K 并且不依赖任何第三方。极简设计仅 if、for、switch、set、define、include、render 七个核心指令,让学习成本低到极致。独创 DKFF(Dynamic Key Feature Forward) 词法分析算法与 DLRD (Double Layer Recursive Descent)语法分析算法,避免使用 javacc、antlr、jflex 生成器,令代码量少到极致。
|
||||
|
||||
#### Maven 坐标
|
||||
|
||||
```java
|
||||
<dependency>
|
||||
<groupId>com.jfinal</groupId>
|
||||
<artifactId>enjoy</artifactId>
|
||||
<version>4.2</version>
|
||||
</dependency>
|
||||
```
|
||||
|
||||
#### Enjoy 主要特点
|
||||
- 消灭传统模板引擎中大量繁杂概念,仅七个核心指令,学习成本极低
|
||||
- 独创 DKFF 词法分析算法与 DLRD 语法分析算法,避免使用 javacc、antlr
|
||||
- 功能强大,极为简单覆盖掉 freemarker、velocity 的核心功能
|
||||
- 扩展性强,支持多种扩展方式,且是唯一支持指令级扩展的模板引擎
|
||||
- 与 java 打通式设计,在模板中与 java 交互极为方便
|
||||
- 贴近 java 使用直觉,为 java 开发者量身打造
|
||||
- 功能强大,极为简单覆盖掉 freemarker、velocity 的核心功能
|
||||
- 扩展性强,支持多种扩展方式,且是唯一支持指令级扩展的模板引擎
|
||||
- 回归模板引擎渲染 View 数据的本质,采用指令式设计,避免 view 层表达复杂逻辑
|
||||
- 体积小,仅 171K,且不依赖于任何第三方
|
||||
- 体积小,仅 227 KB,且不依赖于任何第三方
|
||||
|
||||
|
||||
#### 简单示例:
|
||||
@@ -34,9 +44,10 @@ Enjoy 是基于 Java 语言的极轻量极魔板引擎。极轻量级仅 171K
|
||||
```
|
||||
|
||||
**2.详细使用方法见 jfinal 手册**
|
||||
read me 正在补充,详细使用文档请下载 jfinal.com 官网的 jfinal 手册[http://www.jfinal.com](http://www.jfinal.com)
|
||||
|
||||
**JFinal 官方网站:[http://www.jfinal.com](http://www.jfinal.com)**
|
||||
read me 正在补充,详细使用文档见官网:[https://www.jfinal.com/doc/6-1](https://www.jfinal.com/doc/6-1)
|
||||
|
||||
**JFinal Enjoy 官方文档:[https://www.jfinal.com/doc/6-1](https://www.jfinal.com/doc/6-1)**
|
||||
|
||||
|
||||
|
||||
|
2
pom.xml
2
pom.xml
@@ -4,7 +4,7 @@
|
||||
<artifactId>enjoy</artifactId>
|
||||
<packaging>jar</packaging>
|
||||
<name>enjoy</name>
|
||||
<version>3.6</version>
|
||||
<version>4.2</version>
|
||||
<url>http://www.jfinal.com</url>
|
||||
<description>Enjoy is a simple, light, rapid, independent, extensible Java Template Engine.</description>
|
||||
|
||||
|
@@ -30,11 +30,6 @@ import java.util.Map;
|
||||
@SuppressWarnings({"serial", "rawtypes", "unchecked"})
|
||||
public class Kv extends HashMap {
|
||||
|
||||
@Deprecated
|
||||
private static final String STATE_OK = "isOk";
|
||||
@Deprecated
|
||||
private static final String STATE_FAIL = "isFail";
|
||||
|
||||
public Kv() {
|
||||
}
|
||||
|
||||
@@ -46,68 +41,6 @@ public class Kv extends HashMap {
|
||||
return new Kv();
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public static Kv ok() {
|
||||
return new Kv().setOk();
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public static Kv ok(Object key, Object value) {
|
||||
return ok().set(key, value);
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public static Kv fail() {
|
||||
return new Kv().setFail();
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public static Kv fail(Object key, Object value) {
|
||||
return fail().set(key, value);
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public Kv setOk() {
|
||||
super.put(STATE_OK, Boolean.TRUE);
|
||||
super.put(STATE_FAIL, Boolean.FALSE);
|
||||
return this;
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public Kv setFail() {
|
||||
super.put(STATE_FAIL, Boolean.TRUE);
|
||||
super.put(STATE_OK, Boolean.FALSE);
|
||||
return this;
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public boolean isOk() {
|
||||
Boolean isOk = (Boolean)get(STATE_OK);
|
||||
if (isOk != null) {
|
||||
return isOk;
|
||||
}
|
||||
Boolean isFail = (Boolean)get(STATE_FAIL);
|
||||
if (isFail != null) {
|
||||
return !isFail;
|
||||
}
|
||||
|
||||
throw new IllegalStateException("调用 isOk() 之前,必须先调用 ok()、fail() 或者 setOk()、setFail() 方法");
|
||||
}
|
||||
|
||||
@Deprecated
|
||||
public boolean isFail() {
|
||||
Boolean isFail = (Boolean)get(STATE_FAIL);
|
||||
if (isFail != null) {
|
||||
return isFail;
|
||||
}
|
||||
Boolean isOk = (Boolean)get(STATE_OK);
|
||||
if (isOk != null) {
|
||||
return !isOk;
|
||||
}
|
||||
|
||||
throw new IllegalStateException("调用 isFail() 之前,必须先调用 ok()、fail() 或者 setOk()、setFail() 方法");
|
||||
}
|
||||
|
||||
public Kv set(Object key, Object value) {
|
||||
super.put(key, value);
|
||||
return this;
|
||||
|
@@ -16,6 +16,9 @@
|
||||
|
||||
package com.jfinal.kit;
|
||||
|
||||
import java.lang.reflect.Method;
|
||||
import java.lang.reflect.Parameter;
|
||||
|
||||
/**
|
||||
* 反射工具类
|
||||
*/
|
||||
@@ -29,6 +32,45 @@ public class ReflectKit {
|
||||
}
|
||||
}
|
||||
|
||||
public static String getMethodSignature(Method method) {
|
||||
StringBuilder ret = new StringBuilder()
|
||||
.append(method.getDeclaringClass().getName())
|
||||
.append(".")
|
||||
.append(method.getName())
|
||||
.append("(");
|
||||
|
||||
int index = 0;
|
||||
Parameter[] paras = method.getParameters();
|
||||
for (Parameter p : paras) {
|
||||
if (index++ > 0) {
|
||||
ret.append(", ");
|
||||
}
|
||||
ret.append(p.getParameterizedType().getTypeName());
|
||||
}
|
||||
|
||||
return ret.append(")").toString();
|
||||
}
|
||||
|
||||
/*
|
||||
public static String getMethodSignature(Method method) {
|
||||
StringBuilder ret = new StringBuilder()
|
||||
.append(method.getDeclaringClass().getName())
|
||||
.append(".")
|
||||
.append(method.getName())
|
||||
.append("(");
|
||||
|
||||
int index = 0;
|
||||
java.lang.reflect.Type[] paraTypes = method.getGenericParameterTypes();
|
||||
for (java.lang.reflect.Type type : paraTypes) {
|
||||
if (index++ > 0) {
|
||||
ret.append(", ");
|
||||
}
|
||||
ret.append(type.getTypeName());
|
||||
}
|
||||
|
||||
return ret.append(")").toString();
|
||||
}*/
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
@@ -26,6 +26,7 @@ import com.jfinal.template.expr.ast.FieldGetter;
|
||||
import com.jfinal.template.expr.ast.FieldKeyBuilder;
|
||||
import com.jfinal.template.expr.ast.FieldKit;
|
||||
import com.jfinal.template.expr.ast.MethodKit;
|
||||
import com.jfinal.template.io.EncoderFactory;
|
||||
import com.jfinal.template.source.ClassPathSourceFactory;
|
||||
import com.jfinal.template.source.ISource;
|
||||
import com.jfinal.template.source.ISourceFactory;
|
||||
@@ -471,6 +472,17 @@ public class Engine {
|
||||
return config.getEncoding();
|
||||
}
|
||||
|
||||
/**
|
||||
* Enjoy 模板引擎对 UTF-8 的 encoding 做过性能优化,某些偏门字符在
|
||||
* 被编码为 UTF-8 时会出现异常,此时可以通过继承扩展 EncoderFactory
|
||||
* 来解决编码异常,具体用法参考:
|
||||
* http://www.jfinal.com/feedback/5340
|
||||
*/
|
||||
public Engine setEncoderFactory(EncoderFactory encoderFactory) {
|
||||
config.setEncoderFactory(encoderFactory);
|
||||
return this;
|
||||
}
|
||||
|
||||
public Engine setWriterBufferSize(int bufferSize) {
|
||||
config.setWriterBufferSize(bufferSize);
|
||||
return this;
|
||||
|
@@ -17,7 +17,6 @@
|
||||
package com.jfinal.template;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
@@ -117,16 +116,10 @@ public class Template {
|
||||
* 适用于代码生成器类似应用场景
|
||||
*/
|
||||
public void render(Map<?, ?> data, File file) {
|
||||
FileOutputStream fos = null;
|
||||
try {
|
||||
fos = new FileOutputStream(file);
|
||||
try (FileOutputStream fos = new FileOutputStream(file)) {
|
||||
render(data, fos);
|
||||
} catch (FileNotFoundException e) {
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
} finally {
|
||||
if (fos != null) {
|
||||
try {fos.close();} catch (IOException e) {e.printStackTrace(System.err);}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -28,8 +28,8 @@ public class TemplateException extends RuntimeException {
|
||||
super(loc != null ? msg + loc : msg);
|
||||
}
|
||||
|
||||
public TemplateException(String msg, Location loc, Throwable t) {
|
||||
super(loc != null ? msg + loc : msg, t);
|
||||
public TemplateException(String msg, Location loc, Throwable cause) {
|
||||
super(loc != null ? msg + loc : msg, cause);
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -28,10 +28,10 @@ import com.jfinal.template.stat.Scope;
|
||||
*
|
||||
* field 表达式取值优先次序,以 user.name 为例
|
||||
* 1:假如 user.getName() 存在,则优先调用
|
||||
* 2:假如 user 为 Model 子类,则调用 user.get("name")
|
||||
* 3:假如 user 为 Record,则调用 user.get("name")
|
||||
* 4:假如 user 为 Map,则调用 user.get("name")
|
||||
* 5:假如 user 具有 public name 属性,则取 user.name 属性值
|
||||
* 2:假如 user 具有 public name 属性,则取 user.name 属性值
|
||||
* 3:假如 user 为 Model 子类,则调用 user.get("name")
|
||||
* 4:假如 user 为 Record,则调用 user.get("name")
|
||||
* 5:假如 user 为 Map,则调用 user.get("name")
|
||||
*/
|
||||
public class Field extends Expr {
|
||||
|
||||
|
@@ -44,10 +44,13 @@ public class FieldKit {
|
||||
LinkedList<FieldGetter> ret = new LinkedList<FieldGetter>();
|
||||
|
||||
ret.addLast(new GetterMethodFieldGetter(null));
|
||||
ret.addLast(new RealFieldGetter(null));
|
||||
// ret.addLast(new ModelFieldGetter());
|
||||
// ret.addLast(new RecordFieldGetter());
|
||||
ret.addLast(new MapFieldGetter());
|
||||
ret.addLast(new RealFieldGetter(null));
|
||||
|
||||
// 挪到第二的位置,addSharedObject(..., modelObj) 用法可以获取到 model 中的 public 属性
|
||||
// ret.addLast(new RealFieldGetter(null));
|
||||
ret.addLast(new ArrayLengthGetter());
|
||||
// ret.addLast(new IsMethodFieldGetter());
|
||||
|
||||
@@ -133,6 +136,10 @@ public class FieldKit {
|
||||
|
||||
getters = ret.toArray(new FieldGetter[ret.size()]);
|
||||
}
|
||||
|
||||
public static void clearCache() {
|
||||
fieldGetterCache.clear();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
@@ -115,6 +115,10 @@ public class MethodKit {
|
||||
forbiddenMethods.remove(methodName);
|
||||
}
|
||||
|
||||
public static void clearCache() {
|
||||
methodCache.clear();
|
||||
}
|
||||
|
||||
public static MethodInfo getMethod(Class<?> targetClass, String methodName, Object[] argValues) {
|
||||
Class<?>[] argTypes = getArgTypes(argValues);
|
||||
Long key = getMethodKey(targetClass, methodName, argTypes);
|
||||
|
@@ -18,7 +18,6 @@ package com.jfinal.template.source;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.net.URL;
|
||||
@@ -64,7 +63,7 @@ public class ClassPathSource implements ISource {
|
||||
this.classLoader = getClassLoader();
|
||||
this.url = classLoader.getResource(finalFileName);
|
||||
if (url == null) {
|
||||
throw new IllegalArgumentException("File not found : \"" + finalFileName + "\"");
|
||||
throw new IllegalArgumentException("File not found in CLASSPATH or JAR : \"" + finalFileName + "\"");
|
||||
}
|
||||
|
||||
processIsInJarAndlastModified();
|
||||
@@ -141,9 +140,8 @@ public class ClassPathSource implements ISource {
|
||||
|
||||
public static StringBuilder loadFile(InputStream inputStream, String encoding) {
|
||||
StringBuilder ret = new StringBuilder();
|
||||
BufferedReader br = null;
|
||||
try {
|
||||
br = new BufferedReader(new InputStreamReader(inputStream, encoding));
|
||||
|
||||
try (BufferedReader br = new BufferedReader(new InputStreamReader(inputStream, encoding))) {
|
||||
// br = new BufferedReader(new FileReader(fileName));
|
||||
String line = br.readLine();
|
||||
if (line != null) {
|
||||
@@ -159,16 +157,6 @@ public class ClassPathSource implements ISource {
|
||||
} catch (Exception e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
finally {
|
||||
if (br != null) {
|
||||
try {
|
||||
br.close();
|
||||
} catch (IOException e) {
|
||||
// com.jfinal.kit.LogKit.error(e.getMessage(), e);
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public String toString() {
|
||||
|
@@ -19,7 +19,6 @@ package com.jfinal.template.source;
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStreamReader;
|
||||
import com.jfinal.template.EngineConfig;
|
||||
|
||||
@@ -92,9 +91,8 @@ public class FileSource implements ISource {
|
||||
|
||||
public static StringBuilder loadFile(File file, String encoding) {
|
||||
StringBuilder ret = new StringBuilder((int)file.length() + 3);
|
||||
BufferedReader br = null;
|
||||
try {
|
||||
br = new BufferedReader(new InputStreamReader(new FileInputStream(file), encoding));
|
||||
|
||||
try (BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream(file), encoding))) {
|
||||
// br = new BufferedReader(new FileReader(fileName));
|
||||
String line = br.readLine();
|
||||
if (line != null) {
|
||||
@@ -110,16 +108,6 @@ public class FileSource implements ISource {
|
||||
} catch (Exception e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
finally {
|
||||
if (br != null) {
|
||||
try {
|
||||
br.close();
|
||||
} catch (IOException e) {
|
||||
// com.jfinal.kit.LogKit.error(e.getMessage(), e);
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public String toString() {
|
||||
|
@@ -69,7 +69,7 @@ public class Include extends Stat {
|
||||
Expr expr = exprList.getExpr(0);
|
||||
if (expr instanceof Const && ((Const)expr).isStr()) {
|
||||
} else {
|
||||
throw new ParseException("The first parameter of #include directive must be String", location);
|
||||
throw new ParseException("The first parameter of #include directive must be String, or use the #render directive", location);
|
||||
}
|
||||
// 其它参数必须为赋值表达式
|
||||
if (len > 1) {
|
||||
|
Reference in New Issue
Block a user