SpringBoot-Velocity模板引擎-快速入门

Velocity-快速入门

一 介绍

Apache Velocity 是一个基于 Java 的模板引擎,它允许任何人使用简单而强大的模板语言来引用对象数据,并生成基于文本的输出。Velocity 最初是作为 WebMacro 项目的一部分开发的,后来成为一个独立的开源项目,并最终加入了 Apache 软件基金会。
在这里插入图片描述

1 主要特点

  • 简单易用:Velocity 拥有简单的语法,易于学习和使用,适合快速开发。
  • 灵活性高:它可以用于多种场景,如生成 HTML 页面、电子邮件、源代码、配置文件等。
  • 性能良好:Velocity 在处理大量数据和高并发请求时表现优秀。
  • 强大的模板语言:支持条件判断、循环、宏定义等功能,能够满足复杂的业务逻辑需求。
  • 丰富的工具集:提供了一系列的工具类,帮助开发者更高效地利用 Velocity 进行开发。

2 应用场景

  • Web 开发:与 Servlet 结合,可以用来动态生成 HTML 页面。
  • 邮件通知:根据用户信息定制个性化的邮件内容。
  • 代码生成:可以用于自动生成代码,比如从数据库表结构生成 DAO 层代码。
  • 报告生成:根据数据生成各种格式的报告文档。
  • 配置文件生成:可以根据不同的环境或参数生成相应的配置文件。

3 社区和支持

作为一个成熟的开源项目,Velocity 拥有一个活跃的社区,提供了大量的文档、教程和示例。如果你遇到任何问题,可以通过官方文档、论坛或者 Stack Overflow 等平台寻求帮助。

二 SpringBoot快速入门

1 创建入门案例Demo

在这里插入图片描述

2 引入依赖

 <!-- velocity代码生成使用模板 -->
<dependency>
<groupId>org.apache.velocity</groupId>
<artifactId>velocity-engine-core</artifactId>
<version>2.3</version>
</dependency>

3 引入工具类

这个是velocity工厂初始化工具,我直接抽成一个类使用

package com.demo.velocitydemo.utils;import java.util.Properties;
import org.apache.velocity.app.Velocity;/*** VelocityEngine工厂* * @author ruoyi*/
public class VelocityInitializer
{/*** 初始化vm方法*/public static void initVelocity(){Properties p = new Properties();try{// 加载classpath目录下的vm文件p.setProperty("resource.loader.file.class", "org.apache.velocity.runtime.resource.loader.ClasspathResourceLoader");// 定义字符集p.setProperty(Velocity.INPUT_ENCODING, "UTF-8");// 初始化Velocity引擎,指定配置PropertiesVelocity.init(p);}catch (Exception e){throw new RuntimeException(e);}}
}

4 创建模板

src/test/resources下创建vm目录用来存放模板

在这里插入图片描述

创建一个简单模板(后缀名是.vm):text.html.vm

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
<h1>${message} #*设置填充的字段*#
</h1>
</body>
</html>

5 创建测试类

package com.demo.velocitydemo;import com.demo.velocitydemo.utils.VelocityInitializer;
import org.apache.velocity.Template;
import org.apache.velocity.VelocityContext;
import org.apache.velocity.app.Velocity;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;import java.io.FileWriter;
import java.io.IOException;@SpringBootTest
class VelocityDemoApplicationTests {@Testvoid contextLoads() throws Exception {//1 初始化模板引擎VelocityInitializer.initVelocity();//2 准备数据模型VelocityContext velocityContext=new VelocityContext();velocityContext.put("message","holle world");//设置填充的字段内容//3 读取模板Template template = Velocity.getTemplate("vm/text.html.vm", "UTF-8");//模板的位置,设置编码UTF-8//4 渲染模板 (合并输出)//设置输出流,配置输出文件的位置FileWriter fileWriter = new FileWriter("E:\\Programming files\\java\\Project_files\\Velocity-Demo\\Velocity-Demo\\text1.html");template.merge(velocityContext,fileWriter);fileWriter.close();//关闭输出流}}

一定要关闭输出流,不然页面无法显示

6 查看结果

找到text.html.vm输出位置,发现${message}被填充成holle world

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
<h1>holle world
</h1>
</body>
</html>

浏览器打开,渲染成功

在这里插入图片描述

三 基本语法

1 变量

  • 在模板中定义变量,#set开头,比如#set($name="velocity")

  • 获取变量的值 $name或者${name}

  • Velocity 支持字符串、数字和布尔值字面量。

    String: "Hello"
    Number: 123
    Boolean: $true
    

$name或者${name}举例

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
## 定义变量
#set($name="Holle")## 使用变量
${name} <br>
$name   <br></body>
</html>

效果:

Holle
Holle

注意:$name不能用于字符串拼接例如:

<body>
${name} World <br>
$nameWorld  
</body>

效果:

Holle World 
$nameWorld   

如果变量是java对象:

User:

package com.demo.velocitydemo.pojos;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class User {private String name;private String sex;private int age;
}

测试类:

    @Testvoid contextLoads2() throws Exception {//1 初始化模板引擎VelocityInitializer.initVelocity();//2 准备数据模型VelocityContext velocityContext=new VelocityContext();//实体化对象UserUser user = new User("小明", "男", 18);//主要:第一个变量“User”要和vm模板的变量名对应。velocityContext.put("User",user);//3 读取模板Template template = Velocity.getTemplate("vm/text.html.vm", "UTF-8");//4 渲染模板 (合并输出)FileWriter fileWriter = new FileWriter("E:\\Programming_files\\Java\\Project_file\\Velocity-Demo\\text1.html");template.merge(velocityContext,fileWriter);fileWriter.close();//关闭输出流}

模板:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
## 获取Uesr对象
$User
## 变量明.属性名
姓名:$User.name   <br>
性别:$User.sex    <br>
年龄:$User.age    <br>
</body>
</html>

渲染效果:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
User(name=小明, sex=男, age=18) <br>
姓名:小明   <br>
性别:男    <br>
年龄:18    <br>
</body>
</html>

2 循环

  • 循环语法:#foreach(....) #end

模板:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>
## 定义集合
#set($list=["小猫","小狗","小牛","小鱼","小猪"])#foreach($item in $list)## $foreach.count 遍历的索引,从1开始,如果想从0开始 $foreach.index 。$item是集合中的元素$foreach.count. $item <br>
#end</body>
</html>

渲染效果:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>1. 小猫 <br>2. 小狗 <br>3. 小牛 <br>4. 小鱼 <br>5. 小猪 <br>
</body>
</html>

变量对象

测试方法:

    @Testvoid contextLoads3() throws Exception {//1 初始化模板引擎VelocityInitializer.initVelocity();//2 准备数据模型VelocityContext velocityContext=new VelocityContext();User user1 = new User("小明", "男", 18);User user2 = new User("小明", "男", 18);User user3= new User("小明", "男", 18);List<User> users = List.of(user1, user2, user3);velocityContext.put("Users",users);//3 读取模板Template template = Velocity.getTemplate("vm/text.html.vm", "UTF-8");//4 渲染模板 (合并输出)FileWriter fileWriter = new FileWriter("E:\\Programming_files\\Java\\Project_file\\Velocity-Demo\\text1.html");template.merge(velocityContext,fileWriter);fileWriter.close();//关闭输出流}

模板:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>#foreach($item in $Users)$item.name $item.sex $item.age <br>
#end</body>
</html>

渲染效果:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>小明 男 18 <br>小明 男 18 <br>小明 男 18 <br>
</body>
</html>

3 条件判断

  • 判断语法:#if(条件)...#elseif(条件)...#else...#end

模板:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>## 设置变量#set($age=19)#if($age>=18)成年#elseif($age<18)未成年#else未知#end
</body>
</html>

渲染效果:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>test</title>
</head>
<body>成年
</body>
</html>

如果要判断对象是否为空

不为空执行操作

#if($obj).....#end

为空执行操作

#if(!$obj).....#end

也支持||&&逻辑判断

5 注释

  • 单行注释使用 ##,多行注释使用 #* *#
## This is a single-line comment#* 
This is a
multi-line comment
*#

注释不会被渲染

6 宏定义

  • 宏(macro)可以定义可重用的代码块。

相当于函数

#macro( greeting $name )Hello, $name!
#end#set( $name = "Alice" )
#set( $name2 = "Bob" )#greeting( $name )
#greeting( $name2 )

7 字典(Map)

  • 可以创建和操作字典(Map)。
#set( $map = {"name": "Alice", "age": 30} )
Name: $map.name
Age: $map["age"]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/58834.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

UE5之5.4 第一人称示例代码阅读2 子弹发射逻辑

TP_WeaponComponent.h 看看头文件 暴露了attach weapon和fire给蓝图 这两个函数意义一看名字吧&#xff0c;就是捡起来枪的时候执行&#xff0c;一个就是发射子弹的时候执行 #pragma once#include "CoreMinimal.h" #include "Components/SkeletalMeshComponen…

智能合约分享

智能合约练习 一、solidity初学者经典示例代码&#xff1a; 1.存储和检索数据&#xff1a; // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // 声明 Solidity 编译器版本// 定义一个名为 SimpleStorage 的合约 contract SimpleStorage {// 声明一个公共状态变量 d…

绿色智慧冻结 专精深地空间:全国冻结法施工技术交流研讨会即将在京召开

2024年10月29日&#xff0c;由中国煤炭工业协会、中国煤炭建设协会、中国中煤能源集团有限公司主办&#xff0c;中煤建设集团有限公司、中煤邯郸特殊凿井有限公司承办的全国冻结法施工技术交流研讨会将在北京黄河京都会议中心隆重召开。研讨会将以“绿色智慧冻结 专精深地空间”…

CSGO: Content-Style Composition in Text-to-Image Generation(代码的复现)

文章目录 CSGO简介论文的代码部署需要下载的模型权重&#xff1a;复现中存在的一些问题 推理代码生成结果示意图 CSGO简介 CSGO: Content-Style Composition in Text-to-Image Generation&#xff08;风格迁移&#xff09; 本文是一篇风格迁移的论文&#xff1a;将内容参考图像…

[nssround#4 swpu]1zweb

能上传文件和查看文件 非预期:出题人没有对读取文件做限制&#xff0c;导致了目录穿越&#xff0c;可直接读取flag 预期解如下&#xff1b; 首先读取index.php与upload.php php <?php //index.php class LoveNss{ public $ljt; public $dky; public $cmd;…

Jmeter命令监控CPU等指标

JMeter 命令行执行脚本得到的报告中&#xff0c;是没有CPU、内存使用率等监控数据的&#xff0c;但是可以使用JMeter插件帮忙。 一、下载jmeter-plugins-manager.jar 下载后将文件放到jmeter安装包lib/ext目录下。打开Jmeter》菜单栏》选项》Plugins Manager 二、安装PerfMon…

江协科技STM32学习- P32 MPU6050

&#x1f680;write in front&#x1f680; &#x1f50e;大家好&#xff0c;我是黄桃罐头&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流 &#x1f381;欢迎各位→点赞&#x1f44d; 收藏⭐️ 留言&#x1f4dd;​…

原生html+js+css+php多图上传带预览可增删判断图片大小和后缀

原生htmljscssphp多图上传带预览可增删&#xff0c;前后端判断图片大小和后缀 源码来自AI&#xff0c;有改动&#xff0c;整合亲测可用 <?php // 设置允许的最大文件大小为 2MB $maxFileSize 2 * 1024 * 1024; $allowedExtensions [jpg, jpeg, png, gif];// 上传目录&am…

java设计模式之创建者模式(5种)

设计模式 软件设计模式&#xff0c;又称为设计模式&#xff0c;是一套被反复利用&#xff0c;代码设计经验的总结&#xff0c;他是在软件设计过程中的一些不断发生的问题&#xff0c;以及该问题的解决方案。 **创建者模式又分为以下五个模式&#xff1a;**用来描述怎么“将对象…

如何一键更换ppt模板?掌握这2个ppt技巧快速搞定!

每当要制作ppt&#xff0c;很多人会第一时间去搜刮各种ppt模板&#xff0c;有时我们找到了一份貌似符合需求的模板&#xff0c;等到了ppt制作环节&#xff0c;才发现离我们的预期相距甚远&#xff0c;做到一半的ppt如何换模板呢&#xff1f; 想要在中途更换ppt模板&#xff0c;…

操作系统笔记(五)信号量,经典的IPC问题(读写者问题...)

信号量 一个信号量是一个包含两部分内容的数据结构&#xff1a; (a) 一个整数计数器, COUNT (b) 一个记录阻塞进程ID的队列, Q 信号量有两个原子操作&#xff1a; UP(V操作&#xff09; 和 DOWN (P操作) DOWN(S): if (S.count > 0) S.count …

臻于智境 安全护航 亚信安全受邀出席新华三智算新品发布会

近日&#xff0c;紫光股份旗下新华三集团在北京隆重举办了主题为“乘势 进化 臻于智境”的新华三智算新品发布会。作为新华三集团的长期战略合作伙伴&#xff0c;亚信安全受邀参会&#xff0c;亚信安全CEO马红军出席发布仪式&#xff0c;并与来自各界的业界伙伴共同探讨智能化…

【解决】Ubuntu18.04 卸载python之后桌面异常且终端无法打开,重启后进入tty1,没有图形化界面

我因为python版本太过于混乱 &#xff08;都是为了学习os&#xff09; &#xff0c;3.6—3.9版本我都安装了&#xff0c;指向关系也很混乱&#xff0c;本着“重装是最不会乱”的原则&#xff0c;我把全部版本都卸载了。然后装了3.9 发现终端打不开了&#xff0c;火狐浏览器的图…

2-140 基于Solidworks和Matlab Simulink Simscape仿真的机器人手臂仿真

基于Solidworks和Matlab Simulink Simscape仿真的机器人手臂仿真&#xff0c;使用Simulink-Simscape Multibody模块&#xff0c;完成SW-Simscape-Multibody-Link-Plugin手臂仿真&#xff0c;通过调节关节实现手臂动作&#xff0c;并显示三坐标位置。程序已调通&#xff0c;可直…

流媒体协议.之(RTP,RTCP,RTSP,RTMP,HTTP)(一)

闲着没事做&#xff0c;记录一下开发项目用过的协议&#xff0c;项目中&#xff0c;大多是是实时显示播放的&#xff0c;通过私有协议&#xff0c;传输到上位机&#xff0c;实时播放&#xff0c;延时小于200ms&#xff0c;仿照这些协议&#xff0c;定义的数据格式。如果用这些协…

通过思维导图【脑图】梳理Java 开发技术栈

通过思维导图【脑图】梳理Java 开发技术栈 前言一、思维导图概述二、Java 开发技术栈1.整体脑图结构2.基础知识3.核心语法4.高级特性5.常用框架6.数据库7.算法8.设计模式9.最佳实践10.资源推荐 附思维导图原件总结我是将军我一直都在&#xff0c;。&#xff01; 前言 将军深刻…

Mac “屏幕保护程序启动或显示器关闭后需要密码“无效

屏幕保护程序启动或显示器关闭后需要密码只能选择“立即”的解决方法&#xff1a; 在 iPhone mirror中设置&#xff0c;每次询问权限。 参考&#xff1a;https://support.apple.com/en-us/120421

基于matlab的凸包(Convex Hull)算法理解与测试

基于matlab的凸包Convex Hull算法理解与测试 0 引言1 Graham扫描算法原理2 Graham扫描算法实现3 Graham扫描算法关键函数4 结语 0 引言 &#x1f4bb;&#x1f4bb;AI一下&#x1f4bb;&#x1f4bb; 凸包算法是计算给定点集的最小凸包的一种算法。凸包是包含给定点集中所有点…

Conmi的正确答案——在Kibana中搜索Elasticsearch的索引

Elasticsearch版本&#xff1a;7.17.25 Kibana版本&#xff1a;7.17.25 0、进入首页 1、点击空间名“默”&#xff08;我的默认空间名就是“默认”&#xff09;&#xff1b; 2、点击气泡弹窗的“管理空间”进入管理页面&#xff1b; 3、点击“索引模式”&#xff0c;进入索引模…

QT访问数据库:应用提示Driver not loaded

在QT中运行完全正确错误截图 解决办法1 我用的是MySQL。我把libmysql.dll复制到应用程序的目录下&#xff0c;即可正常访问数据库。 解决办法2 bool open_work_db() {QString info "support drivers:";for (int i0; i<QSqlDatabase::drivers().size(); i){inf…