# 常见问题
# 如何不登录直接访问
方法1:在ShiroConfig.java
中设置filterChainDefinitionMap
配置url=anon
/admins/**=anon # 表示该 uri 可以匿名访问
/admins/**=auth # 表示该 uri 需要认证才能访问
/admins/**=authcBasic # 表示该 uri 需要 httpBasic 认证
/admins/**=perms[user:add:*] # 表示该 uri 需要认证用户拥有 user:add:* 权限才能访问
/admins/**=port[8080] # 表示该 uri 需要使用 8080 端口
/admins/**=roles[admin] # 表示该 uri 需要认证用户拥有 admin 角色才能访问
/admins/**=ssl # 表示该 uri 需要使用 https 协议
/admins/**=user # 表示该 uri 需要认证或通过记住我认证才能访问
/logout=logout # 表示注销,可以当作固定配置
注意:
anon,authcBasic,authc,user 是认证过滤器。
perms,roles,ssl,rest,port 是授权过滤器。
2
3
4
5
6
7
8
9
10
11
12
13
方法2:在对应的方法上面使用@Anonymous
注解。
// 方法定义匿名注解,作用于单独的方法
@Anonymous
@GetMapping("/list")
public List<SysXxxx> list(SysXxxx xxxx)
{
return xxxxList;
}
2
3
4
5
6
7
# 如何更换项目包路径
懒人可以使用若依框架包名修改器 (opens new window)一键替换。
1、更换目录名称
├── xxxxx
│ └── xxxxx-admin
│ └── xxxxx-common
│ └── xxxxx-framework
│ └── xxxxx-generator
│ └── xxxxx-quartz
│ └── xxxxx-system
│ └── pom.xml
2
3
4
5
6
7
8
2、更换顶级目录中的pom.xml
<modules>
<module>xxxxx-admin</module>
<module>xxxxx-framework</module>
<module>xxxxx-system</module>
<module>xxxxx-quartz</module>
<module>xxxxx-generator</module>
<module>xxxxx-common</module>
</modules>
2
3
4
5
6
7
8
3、更换项目所有包名称com.ruoyi.xxx
换成com.xxxxx.xxx
提示
DataSourceAspect.java
这个类@Pointcut
注解上面的包路径也需要替换com.xxxxx
CaptchaConfig.java
这个类验证码文本生成器参数KAPTCHA_TEXTPRODUCER_IMPL
的包路径也需要替换com.xxxxx
ApplicationConfig.java
这个类@MapperScan
注解上面的包路径也需要替换com.xxxxx
Constants.java
这个类com.ruoyi
需要替换com.xxxx
4、更换application.yml
指定要扫描的Mapper
类的包的路径typeAliasesPackage
包路径名称替换com.xxxxx
# MyBatis
mybatis:
# 搜索指定包别名
typeAliasesPackage: com.你的包名.**.domain
2
3
4
5、更换mapper
文件的namespace
包路径
ruoyi-system/resources/mapper/system/*
ruoyi-quartz/resources/mapper/quartz/*
ruoyi-generator/resources/mapper/generator/*
2
3
xml
包路径名称替换com.xxxxx
6、更换pom
文件内容
提示
以下pom.xml
文件中包含ruoyi
的关键字替换成xxxxx
├── xxxxx
│ └── xxxxx-admin pom.xml
│ └── xxxxx-common pom.xml
│ └── xxxxx-framework pom.xml
│ └── xxxxx-generator pom.xml
│ └── xxxxx-quartz pom.xml
│ └── xxxxx-system pom.xml
│ └── pom.xml
2
3
4
5
6
7
8
7、更换日志路径
- 更换
application.yml
文件logging
属性为com.xxxxx: debug
- 更换
logback.xml
文件为com.xxxxx
8、启动项目验证
提示
到此步骤如能正常启动,表示更换完成。剩余的小细节可以自行调整。
# 业务模块访问出现404
1、单应用检查
- 确认此用户是否已经配置菜单
- 确认此角色是否已经配置菜单权限
- 确认此菜单
url
是否和后台地址
一致
如参数管理,后台地址配置@RequestMapping("/system/config")
对应参数管理url
为/system/config
2、多模块检查(多了几个步骤)
pom.xml
引入了业务子系统ruoyi-admin
添加业务子模块的依赖ruoyi-xxxxx
新增业务模块pom检查配置是否正确
PS:IDEA可能存在缓存,需要清理下缓存在编译。
提示
如果业务模块和项目的包名不一致,需要在启动类上指定扫描包路径,如
@SpringBootApplication(exclude = { DataSourceAutoConfiguration.class }, scanBasePackages = { "com.ruoyi.*", "com.test.*" })
或者加上@ComponentScan({ "com.ruoyi.*", "com.test.*" })
# IDEA更改页面不重启
经常有小伙伴问到这个问题,为什么我的用IDEA修改html页面之后不实时生效呢?
1、修改IDEA设置
File
-> Settings
-> Build Execution Deployment
-> Build Project automatically
勾选
2、勾选Running
Ctrl
+ Shift
+ Alt
+ /
然后选择 Registry
,勾上 Compiler.autoMake.allow.when.app.running
PS:Eclipse开发工具无需任何配置。
# 如何使用多数据源
对于只有两个数据源直接配置slave
加入注解即可。
- 在
resources
目录下修改application-druid.yml
# 从库数据源
slave:
# 开启从库
enabled: true
url: 数据源
username: 用户名
password: 密码
2
3
4
5
6
7
- 在
Service
实现中添加DataSource
注解
@DataSource(value = DataSourceType.SLAVE)
public List<User> selectUserList()
{
return mapper.selectUserList();
}
2
3
4
5
如果涉及到两个以上数据源,参考配置 多数据源
# 如何更换主题皮肤
1、项目主页-个人信息中选择切换主题
2、修改主框架页-默认皮肤,在菜单参数设置
修改参数键名sys.index.skinName
支持如下几种皮肤
- 蓝色 skin-blue
- 绿色 skin-green
- 紫色 skin-purple
- 红色 skin-red
- 黄色 skin-yellow
3、修改主框架页-侧边栏主题,在菜单参数设置
修改参数键名sys.index.sideTheme
支持如下几种主题
- 深色主题theme-dark
- 浅色主题theme-light
注:如需新增修改皮肤主题可以在skins.css
中调整
提示
顶部默认主题颜色在skins.css
/** 蓝色主题 skin-blue **/
.navbar, .skin-blue .navbar {
background-color: #3c8dbc
}
2
3
4
左侧默认主题颜色在static\css\style.css
.navbar-static-side {
background: #2f4050;
}
nav .logo {
background-color: #367fa9;
}
2
3
4
5
6
7
# 如何使用横向菜单
默认的导航菜单都是在左侧,如果需要横向导航菜单可以做如下配置。
1、点击顶部最右侧个人中心头像,切换为横向菜单。(局部设置)
2、在参数管理设置主框架页-菜单导航显示风格
,键值为topnav
为顶部导航菜单。(全局设置)
# 如何获取用户登录信息
- 第一种方法
// 获取当前的用户信息
User currentUser = ShiroUtils.getSysUser();
// 获取当前的用户名称
String userName = currentUser.getUserName();
2
3
4
- 第二种方法(子模块可使用)
// 获取当前的用户名称
String userName = (String) PermissionUtils.getPrincipalProperty("userName");
2
3、界面获取当前用户信息(支持任意th标签)
<input th:value="${@permission.getPrincipalProperty('userName')}">
4、js中获取当前用户信息
var userName = [[${@permission.getPrincipalProperty('userName')}]];
# 如何防止请求重复提交
- 前端通过
js
控制
// 禁用按钮
$.modal.disable();
// 启用按钮
$.modal.enable();
2
3
4
- 后端通过
@RepeatSubmit
注解控制
/**
* 在对应方法添加注解 @RepeatSubmit
*/
@RepeatSubmit
public AjaxResult editSave()
2
3
4
5
# 如何配置允许跨域访问
现在开发的项目一般都是前后端分离的项目,所以跨域访问会经常使用。
1、单个控制器方法CORS
注解
@RestController
@RequestMapping("/system/test")
public class TestController {
@CrossOrigin
@GetMapping("/{id}")
public AjaxResult getUser(@PathVariable Integer userId) {
// ...
}
@DeleteMapping("/{userId}")
public AjaxResult delete(@PathVariable Integer userId) {
// ...
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2、整个控制器启用CORS
注解
@CrossOrigin(origins = "http://ruoyi.vip", maxAge = 3600)
@RestController
@RequestMapping("/system/test")
public class TestController {
@GetMapping("/{id}")
public AjaxResult getUser(@PathVariable Integer userId) {
// ...
}
@DeleteMapping("/{userId}")
public AjaxResult delete(@PathVariable Integer userId) {
// ...
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
3、全局CORS
配置(在ResourcesConfig
重写addCorsMappings
方法)
/**
* web跨域访问配置
*/
@Override
public void addCorsMappings(CorsRegistry registry)
{
// 设置允许跨域的路径
registry.addMapping("/**")
// 设置允许跨域请求的域名
.allowedOrigins("*")
// 是否允许证书
.allowCredentials(true)
// 设置允许的方法
.allowedMethods("GET", "POST", "DELETE", "PUT")
// 设置允许的header属性
.allowedHeaders("*")
// 跨域允许时间
.maxAge(3600);
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# 如何实现滑块验证码
# 日期插件精确到时分秒
1、界面设置时间格式data-format
,选择类型data-type
属性。
<!-- data-type="date"(年)| data-type="month(月)| data-type="date"(日)| data-type="time"(时、分、秒)| data-type="datetime"(年、月、日、时、分、秒) -->
<li class="select-time">
<label>创建时间: </label>
<input type="text" class="time-input" placeholder="开始时间" name="params[beginTime]" data-type="datetime" data-format="yyyy-MM-dd HH:mm:ss"/>
<span>-</span>
<input type="text" class="time-input" placeholder="结束时间" name="params[endTime]" data-type="month" data-format="yyyy-MM"/>
</li>
2
3
4
5
6
7
2、通过js函数设置
datetimepicker
日期控件可以设置format
$('.input-group.date').datetimepicker({
format: 'yyyy-mm-dd hh:ii:ss',
autoclose: true,
minView: 0,
minuteStep:1
});
2
3
4
5
6
laydate
日期控件可以设置common.js
配置type=datetime
layui.use('laydate', function() {
var laydate = layui.laydate;
var startDate = laydate.render({
elem: '#startTime',
max: $('#endTime').val(),
theme: 'molv',
trigger: 'click',
type : 'datetime',
done: function(value, date) {
// 结束时间大于开始时间
if (value !== '') {
endDate.config.min.year = date.year;
endDate.config.min.month = date.month - 1;
endDate.config.min.date = date.date;
} else {
endDate.config.min.year = '';
endDate.config.min.month = '';
endDate.config.min.date = '';
}
}
});
var endDate = laydate.render({
elem: '#endTime',
min: $('#startTime').val(),
theme: 'molv',
trigger: 'click',
type : 'datetime',
done: function(value, date) {
// 开始时间小于结束时间
if (value !== '') {
startDate.config.max.year = date.year;
startDate.config.max.month = date.month - 1;
startDate.config.max.date = date.date;
} else {
startDate.config.max.year = '';
startDate.config.max.month = '';
startDate.config.max.date = '';
}
}
});
});
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
# 代码生成不显示新建表
默认条件需要表注释,特殊情况可在GenMapper.xml
去除table_comment
条件
<select id="selectTableByName" parameterType="String" resultMap="TableInfoResult">
<include refid="selectGenVo"/>
where table_comment <> '' and table_schema = (select database())
</select>
2
3
4
提示
如果版本>=4.0不需要表注解,在代码生成页面导入即可。
# 提示您没有数据的权限
这种情况都属于权限标识配置不对在菜单管理
配置好权限标识(菜单&按钮)
- 确认此用户是否已经配置角色
- 确认此角色是否已经配置菜单权限
- 确认此菜单权限标识是否和后台代码一致
如参数管理
后台配置@RequiresPermissions("system:config:view")
对应参数管理权限标识为system:config:view
注:如需要角色权限,配置角色权限字符 使用@RequiresRoles("admin")
# 富文本编辑器文件上传
富文本控件采用的summernote
,图片上传处理需要设置callbacks
函数
$('.summernote').summernote({
height : '220px',
lang : 'zh-CN',
callbacks: {
onImageUpload: function(files, editor, $editable) {
var formData = new FormData();
formData.append("file", files[0]);
$.ajax({
type: "POST",
url: ctx + "common/upload",
data: data,
cache: false,
contentType: false,
processData: false,
dataType: 'json',
success: function(result) {
if (result.code == web_status.SUCCESS) {
$(obj).summernote('editor.insertImage', result.url, result.fileName);
} else {
$.modal.alertError(result.msg);
}
},
error: function(error) {
$.modal.alertWarning("图片上传失败。");
}
});
}
}
});
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
# 富文本编辑器底部回弹
富文本控件采用的summernote
,如果不需要底部回弹设置followingToolbar: false
$('.summernote').summernote({
placeholder: '请输入公告内容',
height : 192,
lang : 'zh-CN',
followingToolbar: false,
callbacks: {
onImageUpload: function (files) {
sendFile(files[0], this);
}
}
});
2
3
4
5
6
7
8
9
10
11
# 富文本对话框回弹顶部
富文本控件采用的summernote
,点击下方的各种按钮的弹框时,页面会回到顶部,滚到顶部会使用户体验很不好,如需要置于弹框的body
中,可以设置dialogsInBody: true
$('.summernote').summernote({
placeholder: '请输入公告内容',
height : 192,
lang : 'zh-CN',
dialogsInBody: true,
callbacks: {
onImageUpload: function (files) {
sendFile(files[0], this);
}
}
});
2
3
4
5
6
7
8
9
10
11
# 如何创建新的菜单页签
创建新的页签有以下两种方式(js&html)
// 方式1 打开新的选项卡
function dept() {
var url = ctx + "system/dept";
$.modal.openTab("部门管理", url);
// 如果需要打开并刷新 $.modal.openTab("部门管理", url, true);
}
// 方式2 选卡页同一页签打开
function dept() {
var url = ctx + "system/dept";
$.modal.parentTab("部门管理", url);
}
// 方式3 html创建
<a class="menuItem" href="/system/dept">部门管理</a>
// 如果需要打开并刷新
<a class="menuItem" data-refresh="true" href="/system/dept">部门管理</a>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# 表格数据进行汇总统计
对于某些数据需要对金额,数量等进行汇总,可以配置showFooter: true
表示尾部统计
// options 选项中添加尾部统计
showFooter: true,
// columns 中添加
{
field : 'balance',
title : '余额',
sortable: true,
footerFormatter:function (value) {
var sumBalance = 0;
for (var i in value) {
sumBalance += parseFloat(value[i].balance);
}
return "总金额:" + sumBalance;
}
},
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 表格设置行列单元格样式
1、options
参数中配置属性
rowStyle: rowStyle,
2、对应js添加响应方法(根据row
或index
定义规则)即可
function rowStyle(row, index) {
var style = { css: { 'color': '#ed5565' } };
return style;
}
2
3
4
# 表单提交大小限制配置
有时候请求的数据量非常大,数据到后台发现,controller
层对应的方法接收的数据不够,问题的原因:tomcat
的限制了post
的请求的大小和请求参数的个数。
配置文件处理方式,设置post
请求的大小
server:
tomcat:
max-http-post-size: -1
2
3
2、代码设置处理方式,设置post
请求的大小
@Bean
public TomcatServletWebServerFactory embeddedServletContainerFactory() {
TomcatServletWebServerFactory tomcatEmbeddedServletContainerFactory = new TomcatServletWebServerFactory ();
tomcatEmbeddedServletContainerFactory.addConnectorCustomizers(connector ->{
// 解决post表单请求数据的参数过多,数据被截断的问题
connector.setMaxParameterCount(Integer.MAX_VALUE);
connector.setMaxPostSize(200);
});
return tomcatEmbeddedServletContainerFactory;
}
2
3
4
5
6
7
8
9
10
/**
* 解决映射类超过256条数据的问题,仅对当前类
*/
@Override
@InitBinder //类初始化是调用的方法注解
public void initBinder(WebDataBinder binder) {
binder.setAutoGrowNestedPaths(true);
// 给这个controller配置接收list的长度100000,仅在这个controller有效
binder.setAutoGrowCollectionLimit(10000);
}
2
3
4
5
6
7
8
9
10
如果涉及时间类型,增加注解
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") @DateTimeFormat(pattern="yyyy-MM-dd HH:mm:ss")
# 文件上传大小限制配置
Spring Boot
默认最大request size
为10MB(10485760 bytes)
。
如果超出默认大小会提示异常信息。
Maximum upload size exceeded; nested exception is java.lang.IllegalStateException: org.apache.tomcat.util.http.fileupload.FileUploadBase$FileSizeLimitExceededException: The field avatarfile exceeds its maximum permitted size of 10485760 bytes.
可以通过配置文件增加以下内容解决:
spring:
servlet:
multipart:
# 文件最大
max-file-size: 20MB
# 设置总上传数据总大小
max-request-size: 20MB
2
3
4
5
6
7
# 如何去除文件上传按钮
在一些特定情况不需要文件上传的小图标操作,可以把他们去掉。
这里layoutTemplates
为上传控件模板,可以在这里重写上传控件中的元素样式
$("#xxxx").fileinput({
....
layoutTemplates: {
actionDelete: '',
actionUpload: '',
actionZoom: '',
indicator: ''
.....
},
....
2
3
4
5
6
7
8
9
10
# 如何去除数据监控广告
服务监控中使用的Driud
,默认底部有阿里的广告。如果是一个商业项目这个是很不雅也是不允许的
- 找到本地maven库中的对应的druid-1.1.xx.jar文件,用压缩包软件打开
- 找到support/http/resource/js/common.js, 打开找到 buildFooter 方法
this.buildFooter();
buildFooter : function() {
var html ='此处省略一些相关JS代码';
$(document.body).append(html);
},
2
3
4
5
- 删除此函数和及初始方法后覆盖文件
- 重启项目后,广告就会消失了
# 如何支持多类型数据库
对于某些特殊需要支持不同数据库,参考以下支持oracle
mysql
配置
<!--oracle驱动-->
<dependency>
<groupId>com.oracle</groupId>
<artifactId>ojdbc6</artifactId>
<version>11.2.0.3</version>
</dependency>
2
3
4
5
6
# 数据源配置
spring:
datasource:
type: com.alibaba.druid.pool.DruidDataSource
druid:
# 主库数据源
master:
url: jdbc:mysql://127.0.0.1:3306/ry?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8
username: root
password: password
validationQuery: select 1
# 从库数据源
slave:
# 从数据源开关/默认关闭
enabled: true
url: jdbc:oracle:thin:@127.0.0.1:1521:oracle
username: root
password: password
validationQuery: select 1 from dual
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
注意
对于不同数据源造成的驱动问题,可以删除
driverClassName,默认会自动识别驱动
如需要对不同数据源分页需要操作
application.yml中的
pagehelper配置中删除
helperDialect: mysql会自动识别数据源,新增
autoRuntimeDialect=true表示运行时获取数据源
# 如何实现翻页保留选择
- 配置
checkbox
选项field
属性为state
{
field: 'state',
checkbox: true
},
2
3
4
- 表格选项
options
添加rememberSelected
rememberSelected: true,
# 如何实现跳转至指定页
- 表格选项
options
添加showPageGo
showPageGo: true,
# 如何给字典自定义样式
默认提供了default
、primary
、success
、info
、warning
、danger
这几种样式选择,但是有时候并不满足自己的样式需求,那么就可以自定义,参考如下示例流程。
1、我们先在ruoyi-ui.css
自定义一个粉色样式
.badge.custom-pink {
background-color: #ffeded;
color: #ff66cc;
}
2
3
4
2、找到对应的数据字典,进入字典数据,新增时填写样式属性为badge custom-pink
。
3、在对应的表格页面去实现字典,会根据值匹配加上badge custom-pink
样式生效。
# 如何自定义查询条件参数
1、在options
中添加queryParams
参数
var options = {
url: prefix + "/list",
queryParams: queryParams,
columns: [{
field: 'id',
title: '主键'
},
{
field: 'name',
title: '名称'
}]
};
$.table.init(options);
2
3
4
5
6
7
8
9
10
11
12
13
2、在当前页添加queryParams
方法设置自定义查询条件如userName
function queryParams(params) {
var search = $.table.queryParams(params);
search.userName = $("#userName").val();
return search;
}
2
3
4
5
请求后台参数为:pageSize、pageNum、searchValue、orderByColumn、isAsc、userName
3、如果是表格树,添加参数ajaxParams
参数
var options = {
code: "deptId",
parentCode: "parentId",
uniqueId: "deptId",
url: prefix + "/list",
ajaxParams: {
"userId": "1",
"userName": "ruoyi"
},
columns: [{
field: 'id',
title: '主键'
},
{
field: 'name',
title: '名称'
}]
};
$.treeTable.init(options);
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# 如何让表格某些列隐藏掉
1、在options
中columns
设置visible
visible: false, // 隐藏某列(列选项可见)
ignore: true, // 列选项不可见
2
对于需要列选项不可见状态可以设置ignore
# 如何给默认的表格加边框
table-striped
换成 table-bordered
<div class="col-sm-12 select-table table-bordered">
<table id="bootstrap-table"></table>
</div>
2
3
提示
如果是表格树加边框还需要在options
添加bordered:true
# 普通用户创建文件无权限
常见的有几种错误,对应去调整即可。
1、修改logback.xml
,value路径
<!-- 日志存放路径 -->
<property name="log.path" value="/home/ruoyi/logs" />
2
2、修改ehcache-shiro.xml
,path路径
<diskStore path="java.io.tmpdir"/>
3、修改tomcat
临时的日志目录
server:
tomcat:
basedir: /home/ruoyi/temp
2
3
# Druid关闭掉ping的方式
在使用了新版的druid
以后,日志中会一直在报warn
,内容是discard long time none received connection. , jdbcUrl : ...
,程序运行并没有受到影响,只是会不定时有警告,解决方案是可以回退到1.1.22
可以消除警告,也可以通过关闭mysql
的usePingMethod
关闭ping
检查。
启动类的main方法关闭usePingMethod
public class RuoYiApplication
{
public static void main(String[] args)
{
// System.setProperty("spring.devtools.restart.enabled", "false");
System.setProperty("druid.mysql.usePingMethod", "false");
SpringApplication.run(RuoYiApplication.class, args);
System.out.println("(♥◠‿◠)ノ゙ 若依启动成功 ლ(´ڡ`ლ)゙ \n" +
" .-------. ____ __ \n" +
" | _ _ \\ \\ \\ / / \n" +
" | ( ' ) | \\ _. / ' \n" +
" |(_ o _) / _( )_ .' \n" +
" | (_,_).' __ ___(_ o _)' \n" +
" | |\\ \\ | || |(_,_)' \n" +
" | | \\ `' /| `-' / \n" +
" | | \\ / \\ / \n" +
" ''-' `'-' `-..-' ");
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# 如何降低mysql驱动版本
1、在pom.xml
中properties
新增节点如:
<mysql.version>6.0.6</mysql.version>
2、单应用可以不添加,多模块需要在dependencyManagement
声明依赖
<!-- Mysql驱动包 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql.version}</version>
</dependency>
2
3
4
5
6
注意:如果是6以下的版本需要修改application-druid.yml
中driverClassName
com.mysql.jdbc.Driver 是 mysql-connector-java 5中的
com.mysql.cj.jdbc.Driver 是 mysql-connector-java 6中的
# 如何配置tomcat访问日志
1、修改application.yml
中的server
开发环境配置
# 开发环境配置
server:
# 服务器的HTTP端口,默认为80
port: 80
servlet:
# 应用的访问路径
context-path: /
tomcat:
# 存放Tomcat的日志目录
basedir: D:/tomcat
accesslog:
# 开启日志记录
enabled: true
# 访问日志存放目录
directory: logs
# tomcat的URI编码
uri-encoding: UTF-8
# tomcat最大线程数,默认为200
max-threads: 800
# Tomcat启动初始化的线程数,默认值25
min-spare-threads: 30
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
2、重启项目后,在D:/tomcat/logs
目录就可以看到服务器访问日志了
# 如何配置包路径日志级别
例如echache
日志级别默认是info
,我们需要修改ehcache
的日志级别warn
可通过application.yml
中的logging.level
控制。设置对应包路径日志级别。
# 日志配置
logging:
level:
com.ruoyi: debug
org.springframework: warn
org.apache.shiro.cache.ehcache: warn
2
3
4
5
6
同时在logback.xml
文件新增logger
节点
<logger name="org.apache.shiro.cache.ehcache" level="warn" />
# 如何配置项目访问根路径
目前项目后台访问默认路径是:http://localhost:80
,如果需要自定义项目名
或端口
可以修改配置文件src/main/resources/application.yml
server:
port: 8080
servlet:
context-path: /ruoyi
2
3
4
此配置后访问的默认路径是:http://localhost:8080/ruoyi
# Swagger的启用和禁用
可通过application.yml
中的swagger.enable
控制。为true
时表示启用,为false
时表示禁用。
记得关闭
为了系统安全,通常生产环境不建议开启swagger。
# 如何汉化系统接口Swagger
想必很多小伙伴都曾经使用过Swagger
,但是打开UI界面是纯英文的界面并不太友好,作为国人还是习惯中文界面。
- 找到m2/repository/io/springfox/springfox-swagger-ui/x.x.x/springfox-swagger-ui-x.x.x.jar
- 修改对应springfox-swagger-ui-x.x.x.jar包内
resources
目录下swagger-ui.html
,添加如下JS代码
<!-- 选择中文版 -->
<script src='webjars/springfox-swagger-ui/lang/translator.js' type='text/javascript'></script>
<script src='webjars/springfox-swagger-ui/lang/zh-cn.js' type='text/javascript'></script>
2
3
- 本地修改结束后,在覆盖压缩包文件重启就实现汉化了
# Swagger接口出现转换错误
{
"msg": "Failed to convert property value of type 'java.lang.String' to required type 'java.util.Map' for property 'params'; nested exception is java.lang.IllegalStateException: Cannot convert value of type 'java.lang.String' to required type 'java.util.Map' for property 'params': no matching editors or conversion strategy found",
"code": 500
}
2
3
4
一般在Swagger
页面执行查询接口时出现。params
是BaseEntity.java
的属性,请求的时候把默认的值{}
置空就行了。
如果不想看到他,BaseEntity.java
中的params
参数加上@ApiModelProperty(hidden = true)
/** 请求参数 */
@ApiModelProperty(hidden = true)
private Map<String, Object> params;
2
3
# 如何在html页面格式化日期
Thymeleaf主要使用org.thymeleaf.expression.Dates
这个类来处理日期,在模板中使用"#dates"来表示这个对象。
1、格式化日期
[[${#dates.format(date)}]]
或 th:text="${#dates.format(date)}
[[${#dates.formatISO(date)}]]
或 th:text="${#dates.formatISO(date)}
[[${#dates.format(date, 'yyyy-MM-dd HH:mm:ss')}]]
或 th:text="${#dates.format(date, 'yyyy-MM-dd HH:mm:ss')}
2、获取日期字段
获取当前的年份:[[${#dates.year(date)}]]
获取当前的月份:[[${#dates.month(date)}]]
获取当月的天数:[[${#dates.day(date)}]]
获取当前的小时:[[${#dates.hour(date)}]]
获取当前的分钟:[[${#dates.minute(date)}]]
获取当前的秒数:[[${#dates.second(date)}]]
获取当前的毫秒:[[${#dates.millisecond(date)}]]
获取当前的月份名称:[[${#dates.monthName(date)}]]
获取当前是星期几:[[${#dates.dayOfWeek(date)-1}]]
# 如何在表格中实现图片预览
对于某些图片需要在表格中显示,可以使用imageView
方法
// 在columns中格式化对应相关的列属性
{
field: 'avatar',
title: '用户头像',
formatter: function(value, row, index) {
return $.table.imageView(value, '/profile/avatar');
}
},
2
3
4
5
6
7
8
多图片预览可以自己实现,示例。
// 传入value 图片地址数组
previewImg: function(value) {
var data = [];
for (var key of value) {
var json = {};
json.src = key;
data.push(json);
}
layer.photos({
photos: {
"data": data
},
anim: 6 // 0-6的选择,指定弹出图片动画类型,默认随机
});
},
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 如何去掉页脚及左侧菜单栏
1、去除页脚修改style.css
,同时删除index.html
元素
#content-main {
height: calc(100% - 91px);
overflow: hidden;
}
2
3
4
<div class="footer">
<div class="pull-right">© [[${copyrightYear}]] RuoYi Copyright </div>
</div>
2
3
2、去左侧菜单栏(收起时隐藏左侧菜单)修改style.css
body.fixed-sidebar.mini-navbar #page-wrapper {
margin: 0 0 0 0px;
}
body.body-small.fixed-sidebar.mini-navbar #page-wrapper {
margin: 0 0 0 0px;
}
2
3
4
5
6
7
3、去左侧菜单栏(收起时隐藏左侧菜单)修改index.js
function() {
if ($(this).width() < 769) {
$('body').addClass('mini-navbar');
$('.navbar-static-side').fadeIn(); // 换成 $('.navbar-static-side').hide();
$(".sidebar-collapse .logo").addClass("hide");
}
});
function SmoothlyMenu() {
if (!$('body').hasClass('mini-navbar')) {
$(".navbar-static-side").show(); // 添加显示这一行
$('#side-menu').hide();
$(".sidebar-collapse .logo").removeClass("hide");
setTimeout(function() {
$('#side-menu').fadeIn(500);
},
100);
} else if ($('body').hasClass('fixed-sidebar')) {
$(".navbar-static-side").hide(); // 添加隐藏这一行
$('#side-menu').hide();
$(".sidebar-collapse .logo").addClass("hide");
setTimeout(function() {
$('#side-menu').fadeIn(500);
},
300);
} else {
$('#side-menu').removeAttr('style');
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
4、隐藏左侧菜单,需要添加.canvas-menu
到body元素
<body class = "canvas-menu">
# 左侧菜单如何点击不自动展开
在index.js
把里面的$('#side-menu').metisMenu();
修改成如下方式,其中toggle
属性表示是否自动展开
$('#side-menu').metisMenu({
toggle: false,
});
2
3
# 登录页如何开启注册用户功能
在菜单参数设置
修改参数键名sys.account.registerUser
设置true
即可。默认为false
关闭。
# 如何限制账户只能一个人登录
在application.yml
设置maxSession
为1
即可。
# Shiro
shiro:
session:
# 同一个用户最大会话数,比如2的意思是同一个账号允许最多同时两个人登录(默认-1不限制)
maxSession: 1
# 踢出之前登录的/之后登录的用户,默认踢出之前登录的用户
kickoutAfter: false
2
3
4
5
6
7
# 登录页面如何不显示验证码
在application.yml
设置captchaEnabled
为false
即可
# Shiro
shiro:
user:
# 验证码开关
captchaEnabled: false
2
3
4
5
# 如何导出数据列表PDF格式
目前项目的数据列表导出为Excel
格式,但是有些特定的情况想导出为PDF
格式。
1、ruoyi-common\pom.xml
中添加itextpdf
依赖
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.2.0</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itext-asian</artifactId>
<version>5.2.0</version>
</dependency>
2
3
4
5
6
7
8
9
10
2、新增PDF导出类ExcelPDFUtil.java
package com.ruoyi.common.utils.poi;
import java.io.File;
import java.io.OutputStream;
import java.lang.reflect.Field;
import java.math.BigDecimal;
import java.net.URLEncoder;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.Date;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.poi.util.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.itextpdf.text.BaseColor;
import com.itextpdf.text.Document;
import com.itextpdf.text.Element;
import com.itextpdf.text.Font;
import com.itextpdf.text.PageSize;
import com.itextpdf.text.Paragraph;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.BaseFont;
import com.itextpdf.text.pdf.PdfPCell;
import com.itextpdf.text.pdf.PdfPTable;
import com.itextpdf.text.pdf.PdfWriter;
import com.ruoyi.common.annotation.Excel;
import com.ruoyi.common.annotation.Excel.Type;
import com.ruoyi.common.annotation.Excels;
import com.ruoyi.common.config.RuoYiConfig;
import com.ruoyi.common.core.domain.AjaxResult;
import com.ruoyi.common.core.text.Convert;
import com.ruoyi.common.exception.UtilException;
import com.ruoyi.common.utils.DateUtils;
import com.ruoyi.common.utils.DictUtils;
import com.ruoyi.common.utils.StringUtils;
/**
* Excel转PDF相关处理
*
* @author ruoyi
*/
public class ExcelPDFUtil<T>
{
private static final Logger log = LoggerFactory.getLogger(ExcelPDFUtil.class);
/**
* 文件名称
*/
private String pdfName;
/**
* 导出类型(EXPORT:导出数据;IMPORT:导入模板)
*/
private Type type;
/**
* 文档信息
*/
private Document doc;
/**
* 导入导出数据列表
*/
private List<T> list;
/**
* 注解列表
*/
private List<Object[]> fields;
/**
* 用于dictType属性数据存储,避免重复查缓存
*/
public Map<String, String> sysDictMap = new ConcurrentHashMap<String, String>();
/**
* 标题
*/
private String title;
/**
* 需要排除列属性
*/
public String[] excludeFields;
/**
* 实体对象
*/
public Class<T> clazz;
public ExcelPDFUtil(Class<T> clazz)
{
this.clazz = clazz;
}
/**
* 隐藏Excel中列属性
*
* @param fields 列属性名 示例[单个"name"/多个"id","name"]
*/
public void hideColumn(String... fields)
{
this.excludeFields = fields;
}
public void init(List<T> list, String pdfName, String title, Type type)
{
if (list == null)
{
list = new ArrayList<T>();
}
this.list = list;
this.pdfName = pdfName;
this.type = type;
this.title = title;
createExcelField();
}
/**
* 创建excel第一行标题
*/
public Paragraph createTitle() throws Exception
{
if (StringUtils.isNotEmpty(title))
{
BaseFont bfChinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
Font firsetTitleFont = new Font(bfChinese, 22, Font.BOLD);
Paragraph paragraph = new Paragraph(title, firsetTitleFont);
/**
* 设置行间距,俩个参数:float fixedLeading, float multipliedLeading, 总的行间距可以用getTotalLeading()来查看,
* 可以理解是两行文字,顶部到顶部的距离,如果是0,两行文字是重叠的。
* 计算方式:fixedLeading+multipliedLeading*fontSize。在pdfDocument中是当前字体;在ColumnText,是指最大字体。
* 其中fixedLeading是固定参数,默认值是:1.5倍的fontsize multipliedLeading是可变参数,默认值是0.
*/
// 设置行间距
paragraph.setLeading(0.0F);
// 设置对齐方式:居中
paragraph.setAlignment(Element.ALIGN_CENTER);
// 设置首行缩进
paragraph.setFirstLineIndent(0.0F);
//// 设置段落之间的空白
paragraph.setExtraParagraphSpace(200f);
// 设置段落整体的左缩进
paragraph.setIndentationLeft(0.0f);
// 设置段落整体的右缩进
paragraph.setIndentationRight(0.0f);
// 设置段落下方的空白距离
paragraph.setSpacingAfter(10f);
// 设置段落上方的留白,,如果的本页的第一个段落,该设置无效
paragraph.setSpacingBefore(0.0f);
return paragraph;
}
else
{
return null;
}
}
/**
* 对list数据源将其里面的数据导入到excel表单
*
* @param dataList 导出数据集合
* @param sheetName 工作表的名称
* @return 结果
*/
public AjaxResult exportPDF(List<T> dataList, String sheetName)
{
return exportPDF(dataList, sheetName, StringUtils.EMPTY);
}
/**
* 对list数据源将其里面的数据导入到excel表单
*
* @param dataList 导出数据集合
* @param pdfName 工作表的名称
* @param title 标题
* @return 结果
*/
public AjaxResult exportPDF(List<T> dataList, String pdfName, String title)
{
this.init(dataList, pdfName, title, Type.EXPORT);
return writePdf(null);
}
/**
* 对list数据源将其里面的数据导入到excel表单
*
* @param response 返回数据
* @param dataList 导出数据集合
* @param pdfName 文件名称
* @return 结果
*/
public AjaxResult exportPDF(HttpServletResponse response, List<T> dataList, String pdfName)
{
try
{
return exportPDF(response, dataList, pdfName, StringUtils.EMPTY);
}
catch (Exception e)
{
log.error("导出Excel异常{}", e.getMessage());
throw new UtilException("导出PDF失败,请联系网站管理员!");
}
}
/**
* 对list数据源将其里面的数据导入到excel表单
*
* @param response 返回数据
* @param dataList 导出数据集合
* @param pdfName 文件的名称
* @param title 标题
* @return 结果
*/
public AjaxResult exportPDF(HttpServletResponse response, List<T> dataList, String pdfName, String title)
throws Exception
{
response.setContentType("application/pdf");
response.setHeader("Expires", "0");
response.setHeader("Cache-Control", "must-revalidate, post-check=0, pre-check=0");
response.setHeader("Pragma", "public");
response.setHeader("Content-disposition", "attachment; filename=".concat(String.valueOf(URLEncoder.encode(pdfName + ".pdf", "UTF-8"))));
response.setCharacterEncoding("utf-8");
this.init(dataList, pdfName, title, Type.EXPORT);
return this.writePdf(response);
}
/**
* 创建写入数据到Sheet
*/
public AjaxResult writePdf(HttpServletResponse response)
{
OutputStream out = null;
String filename = "";
try
{
if (response == null)
{
filename = encodingFilename(pdfName);
out = Files.newOutputStream(Paths.get(getAbsoluteFile(filename)));
}
else
{
out = response.getOutputStream();
}
// 创建文档
Rectangle rect = new Rectangle(PageSize.A2);
doc = new Document(rect);
PdfWriter.getInstance(doc, out);
doc.open();
doc.newPage();
// 将标题添加到文档上
if (createTitle() != null)
{
doc.add(createTitle());
}
// 添加表格,fields.size() 列
PdfPTable table = new PdfPTable(fields.size());
// 设置列宽
int numberOfColumns = fields.size();
int[] widths = new int[numberOfColumns];
// 分配总宽度100,平均到每一列
Arrays.fill(widths, 100 / numberOfColumns);
table.setWidths(widths);
// 设置表格宽度比例为%100
table.setWidthPercentage(100);
// 设置表格上面空白宽度
table.setSpacingBefore(0f);
// 设置表格下面空白宽度
table.setSpacingAfter(0f);
// 设置表格默认为无边框
table.getDefaultCell().setBorder(0);
// 字体
BaseFont bfChinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
for (int i = 0; i < list.size(); i++)
{
if (i == 0)
{
for (Object[] os : fields)
{
// 设置标题
Excel excel = (Excel) os[1];
Font textFont = new Font(bfChinese, 12, Font.NORMAL, BaseColor.WHITE); // 正常
PdfPCell cell1 = new PdfPCell(new Paragraph(excel.name(), textFont));
// 设置背景颜色
cell1.setBackgroundColor(new BaseColor(128, 128, 128));
// 边框颜色
cell1.setBorderColor(BaseColor.BLACK);
// 设置内容水平居中显示
cell1.setHorizontalAlignment(Element.ALIGN_CENTER);
// 设置垂直居中
cell1.setVerticalAlignment(Element.ALIGN_MIDDLE);
table.addCell(cell1);
}
}
T vo = list.get(i);
for (Object[] os : fields)
{
Field field = (Field) os[0];
Excel excel = (Excel) os[1];
// 用于读取对象中的属性
Object value = getTargetValue(vo, field, excel);
Font textFont = new Font(bfChinese, 11, Font.NORMAL, BaseColor.BLACK);
String dateFormat = excel.dateFormat();
String readConverterExp = excel.readConverterExp();
String separator = excel.separator();
String dictType = excel.dictType();
if (StringUtils.isNotEmpty(dateFormat) && StringUtils.isNotNull(value))
{
value = parseDateToStr(dateFormat, value);
}
else if (StringUtils.isNotEmpty(readConverterExp) && StringUtils.isNotNull(value))
{
value = convertByExp(Convert.toStr(value), readConverterExp, separator);
}
else if (StringUtils.isNotEmpty(dictType) && StringUtils.isNotNull(value))
{
if (!sysDictMap.containsKey(dictType + value))
{
String lable = convertDictByExp(Convert.toStr(value), dictType, separator);
sysDictMap.put(dictType + value, lable);
}
value = sysDictMap.get(dictType + value);
}
else if (value instanceof BigDecimal && -1 != excel.scale())
{
value = (((BigDecimal) value).setScale(excel.scale(), excel.roundingMode())).doubleValue();
}
else
{
String cellValue = Convert.toStr(value);
value = StringUtils.isNull(value) ? excel.defaultValue() : cellValue + excel.suffix();
}
PdfPCell cell11 = new PdfPCell(new Paragraph(value.toString(), textFont));
table.addCell(cell11);
}
}
doc.add(table);
return AjaxResult.success(filename);
}
catch (Exception e)
{
log.error("导出Excel异常{}", e.getMessage());
throw new UtilException("导出PDF失败,请联系网站管理员!");
}
finally
{
doc.close();
IOUtils.closeQuietly(out);
}
}
/**
* 解析字典值
*
* @param dictValue 字典值
* @param dictType 字典类型
* @param separator 分隔符
* @return 字典标签
*/
public static String convertDictByExp(String dictValue, String dictType, String separator)
{
return DictUtils.getDictLabel(dictType, dictValue, separator);
}
/**
* 解析导出值 0=男,1=女,2=未知
*
* @param propertyValue 参数值
* @param converterExp 翻译注解
* @param separator 分隔符
* @return 解析后值
*/
public static String convertByExp(String propertyValue, String converterExp, String separator)
{
StringBuilder propertyString = new StringBuilder();
String[] convertSource = converterExp.split(",");
for (String item : convertSource)
{
String[] itemArray = item.split("=");
if (StringUtils.containsAny(propertyValue, separator))
{
for (String value : propertyValue.split(separator))
{
if (itemArray[0].equals(value))
{
propertyString.append(itemArray[1] + separator);
break;
}
}
}
else
{
if (itemArray[0].equals(propertyValue))
{
return itemArray[1];
}
}
}
return StringUtils.stripEnd(propertyString.toString(), separator);
}
/**
* 格式化不同类型的日期对象
*
* @param dateFormat 日期格式
* @param val 被格式化的日期对象
* @return 格式化后的日期字符
*/
public String parseDateToStr(String dateFormat, Object val)
{
if (val == null)
{
return "";
}
String str;
if (val instanceof Date)
{
str = DateUtils.parseDateToStr(dateFormat, (Date) val);
}
else if (val instanceof LocalDateTime)
{
str = DateUtils.parseDateToStr(dateFormat, DateUtils.toDate((LocalDateTime) val));
}
else if (val instanceof LocalDate)
{
str = DateUtils.parseDateToStr(dateFormat, DateUtils.toDate((LocalDate) val));
}
else
{
str = val.toString();
}
return str;
}
/**
* 编码文件名
*/
public String encodingFilename(String filename)
{
filename = UUID.randomUUID() + "_" + filename + ".pdf";
return filename;
}
/**
* 获取下载路径
*
* @param filename 文件名称
*/
public String getAbsoluteFile(String filename)
{
String downloadPath = RuoYiConfig.getDownloadPath() + filename;
File desc = new File(downloadPath);
if (!desc.getParentFile().exists())
{
desc.getParentFile().mkdirs();
}
return downloadPath;
}
/**
* 获取bean中的属性值
*
* @param vo 实体对象
* @param field 字段
* @param excel 注解
* @return 最终的属性值
* @throws Exception
*/
private Object getTargetValue(T vo, Field field, Excel excel) throws Exception
{
Object o = field.get(vo);
if (StringUtils.isNotEmpty(excel.targetAttr()))
{
String target = excel.targetAttr();
if (target.contains("."))
{
String[] targets = target.split("[.]");
for (String name : targets)
{
o = getValue(o, name);
}
}
else
{
o = getValue(o, target);
}
}
return o;
}
/**
* 以类的属性的get方法方法形式获取值
*
* @param o
* @param name
* @return value
* @throws Exception
*/
private Object getValue(Object o, String name) throws Exception
{
if (StringUtils.isNotNull(o) && StringUtils.isNotEmpty(name))
{
Class<?> clazz = o.getClass();
Field field = clazz.getDeclaredField(name);
field.setAccessible(true);
o = field.get(o);
}
return o;
}
/**
* 得到所有定义字段
*/
private void createExcelField()
{
this.fields = getFields();
this.fields = this.fields.stream().sorted(Comparator.comparing(objects -> ((Excel) objects[1]).sort())).collect(Collectors.toList());
}
/**
* 获取字段注解信息
*/
public List<Object[]> getFields()
{
List<Object[]> fields = new ArrayList<Object[]>();
List<Field> tempFields = new ArrayList<>();
tempFields.addAll(Arrays.asList(clazz.getSuperclass().getDeclaredFields()));
tempFields.addAll(Arrays.asList(clazz.getDeclaredFields()));
for (Field field : tempFields)
{
if (!ArrayUtils.contains(this.excludeFields, field.getName()))
{
// 单注解
if (field.isAnnotationPresent(Excel.class))
{
Excel attr = field.getAnnotation(Excel.class);
if (attr != null && (attr.type() == Type.ALL || attr.type() == type))
{
field.setAccessible(true);
fields.add(new Object[] { field, attr });
}
}
// 多注解
if (field.isAnnotationPresent(Excels.class))
{
Excels attrs = field.getAnnotation(Excels.class);
Excel[] excels = attrs.value();
for (Excel attr : excels)
{
if (!ArrayUtils.contains(this.excludeFields, field.getName() + "." + attr.targetAttr())
&& (attr != null && (attr.type() == Type.ALL || attr.type() == type)))
{
field.setAccessible(true);
fields.add(new Object[] { field, attr });
}
}
}
}
}
return fields;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
3、修改需要导出为PDF的方法。
// 通过AjaxResult返回文件名称到前端方式(RuoYi不分离版本)
@PostMapping("/export")
public AjaxResult export(Xxxx xxxx)
{
List<Xxxx> list = xxxxService.selectXxxxList(xxxx);
ExcelPDFUtil<Xxxx> util = new ExcelPDFUtil<Xxxx>(Xxxx.class);
return util.exportPDF(list, "文件名称");
}
// 通过response返回流到前端方式(RuoYi-Vue分离版本)
@PostMapping("/export")
public void export(HttpServletResponse response, Xxxx xxxx)
{
List<Xxxx> list = xxxxService.selectXxxxList(xxxx);
ExcelPDFUtil<Xxxx> util = new ExcelPDFUtil<Xxxx>(Xxxx.class);
util.exportPDF(response, list, "文件名称");
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# 如何Excel导出时添加水印
1、ruoyi-common\pom.xml
中添加ooxml-schemas
依赖
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>ooxml-schemas</artifactId>
<version>1.4</version>
</dependency>
2
3
4
5
2、新增水印类ExcelWaterMark.java
package com.ruoyi.common.utils.poi;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.Transparency;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import javax.imageio.ImageIO;
import org.apache.poi.openxml4j.opc.PackagePartName;
import org.apache.poi.openxml4j.opc.PackageRelationship;
import org.apache.poi.openxml4j.opc.TargetMode;
import org.apache.poi.xssf.usermodel.XSSFPictureData;
import org.apache.poi.xssf.usermodel.XSSFRelation;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Excel 生成水印
*
* @author ruoyi
*/
public class ExcelWaterMark
{
private static final Logger log = LoggerFactory.getLogger(ExcelWaterMark.class);
/**
* 给 Excel 添加水印
*
* @param workbook XSSFWorkbook
* @param waterMarkText 水印文字内容
*/
public static void insertWaterMarkText(XSSFWorkbook workbook, String waterMarkText)
{
BufferedImage image = createWatermarkImage(waterMarkText);
// 导出到字节流B
ByteArrayOutputStream os = new ByteArrayOutputStream();
try
{
ImageIO.write(image, "png", os);
}
catch (IOException e)
{
log.error("添加水印失败");
}
int pictureIdx = workbook.addPicture(os.toByteArray(), XSSFWorkbook.PICTURE_TYPE_PNG);
XSSFPictureData pictureData = workbook.getAllPictures().get(pictureIdx);
for (int i = 0; i < workbook.getNumberOfSheets(); i++)
{
// 获取每个Sheet表
XSSFSheet sheet = workbook.getSheetAt(i);
PackagePartName ppn = pictureData.getPackagePart().getPartName();
String relType = XSSFRelation.IMAGES.getRelation();
PackageRelationship pr = sheet.getPackagePart().addRelationship(ppn, TargetMode.INTERNAL, relType, null);
sheet.getCTWorksheet().addNewPicture().setId(pr.getId());
}
}
/**
* 创建水印图片 excel
*
* @param waterMark 水印内容
*/
public static BufferedImage createWatermarkImage(String waterMark)
{
String[] textArray = waterMark.split("\n");
java.awt.Font font = new java.awt.Font("microsoft-yahei", java.awt.Font.PLAIN, 20);
Integer width = 500;
Integer height = 200;
BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
// 背景透明 开始
Graphics2D g = image.createGraphics();
image = g.getDeviceConfiguration().createCompatibleImage(width, height, Transparency.TRANSLUCENT);
g.dispose();
// 背景透明 结束
g = image.createGraphics();
g.setColor(new Color(Integer.parseInt("#C5CBCF".substring(1), 16))); // 设定画笔颜色
g.setFont(font); // 设置画笔字体
g.shear(0.1, -0.26); // 设定倾斜度
// 设置字体平滑
g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
int y = 150;
for (int i = 0; i < textArray.length; i++)
{
g.drawString(textArray[i], 0, y); // 画出字符串
y = y + font.getSize();
}
g.dispose(); // 释放画笔
return image;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
3、修改ExcelUtil.java
方法。
将
createWorkbook
方法this.wb = new SXSSFWorkbook(500);
替换成this.wb = new XSSFWorkbook();
将
annotationDataStyles
方法style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
替换成style.setFillPattern(FillPatternType.NO_FILL);
在
writeSheet
方法最后一行新增水印调用方法。
ExcelWaterMark.insertWaterMarkText((XSSFWorkbook) this.wb, "若依管理系统");
# 如何Excel导出子对象多个字段
// 单个字段导出
@Excel(name = "部门名称", targetAttr = "deptName", type = Type.EXPORT)
private Dept dept;
// 多个字段导出
@Excels({
@Excel(name = "部门名称", targetAttr = "deptName", type = Type.EXPORT),
@Excel(name = "部门负责人", targetAttr = "leader", type = Type.EXPORT)
})
private Dept dept;
2
3
4
5
6
7
8
9
10
# 更多操作字符串参数读取问题
事件中需要传递字符串参数,可以参考resetPwd
传递方式。
onclick=resetPwd(" + row.userId + ',' + "'" + row.userName + "'" + ")
完整代码
formatter: function(value, row, index) {
var actions = [];
actions.push('<a class="btn btn-success btn-xs ' + editFlag + '" href="javascript:void(0)" onclick="$.operate.editTab(\'' + row.userId + '\')"><i class="fa fa-edit"></i>编辑</a> ');
actions.push('<a class="btn btn-danger btn-xs ' + removeFlag + '" href="javascript:void(0)" onclick="$.operate.remove(\'' + row.userId + '\')"><i class="fa fa-remove"></i>删除</a> ');
var more = [];
more.push("<a class='btn btn-default btn-xs " + resetPwdFlag + "' href='javascript:void(0)' onclick=resetPwd(" + row.userId + ',' + "'" + row.userName + "'" + ")><i class='fa fa-key'></i>重置密码</a> ");
more.push("<a class='btn btn-default btn-xs " + editFlag + "' href='javascript:void(0)' onclick='authRole(" + row.userId + ")'><i class='fa fa-check-square-o'></i>分配角色</a>");
actions.push('<a tabindex="0" class="btn btn-info btn-xs" role="button" data-container="body" data-placement="left" data-toggle="popover" data-html="true" data-trigger="hover" data-content="' + more.join('') + '"><i class="fa fa-chevron-circle-right"></i>更多操作</a>');
return actions.join('');
}
2
3
4
5
6
7
8
9
10
# 单元格内容过长显示处理方法
1、使用系统自带的方法格式化处理
{
field: 'remark',
title: '备注',
align: 'center',
formatter: function(value, row, index) {
return $.table.tooltip(value);
}
},
2
3
4
5
6
7
8
2、添加css控制
.select-table table {
table-layout:fixed;
}
.select-table .table td {
/* 超出部分隐藏 */
overflow:hidden;
/* 超出部分显示省略号 */
text-overflow:ellipsis;
/*规定段落中的文本不进行换行 */
white-space:nowrap;
/* 配合宽度来使用 */
height:40px;
}
2
3
4
5
6
7
8
9
10
11
12
13
14
# 表格禁用某列复选框选择方法
条件成立禁用checkbox返回(disabled : true)即可。
{
checkbox: true,
formatter: function (value, row, index) {
if($.common.equals("ry", row.loginName)){
return { disabled : true}
} else {
return { disabled : false}
}
}
},
2
3
4
5
6
7
8
9
10
# 表格默认勾选某列复选框方法
条件成立禁用checkbox返回(disabled : true)即可。
{
checkbox: true,
formatter: function (value, row, index) {
if($.common.equals("ry", row.loginName)){
return { checked : true}
} else {
return { checked : false}
}
}
},
2
3
4
5
6
7
8
9
10
提示
如果默认勾选,并且配置了 rememberSelected: true,
需要特殊处理下。参考demo
<!DOCTYPE html>
<html lang="zh" xmlns:th="http://www.thymeleaf.org" xmlns:shiro="http://www.pollix.at/thymeleaf/shiro">
<head>
<th:block th:include="include :: header('翻页记住选择')" />
</head>
<body class="gray-bg">
<div class="container-div">
<div class="btn-group-sm" id="toolbar" role="group">
<a class="btn btn-success" onclick="checkItem()">
<i class="fa fa-check"></i> 选中项
</a>
</div>
<div class="row">
<div class="col-sm-12 select-table table-striped">
<table id="bootstrap-table"></table>
</div>
</div>
</div>
<div th:include="include :: footer"></div>
<script th:inline="javascript">
var prefix = ctx + "demo/table";
var datas = [[${@dict.getType('sys_normal_disable')}]];
$(function() {
var options = {
uniqueId: "userCode",
url: prefix + "/list",
rememberSelected: true,
columns: [{
field: 'state',
checkbox: true,
formatter: function(value, row, index) {
if($.inArray(row.userCode, table.rememberSelectedIds[table.options.id]) !== -1 || row.userCode == 1000001 || row.userCode == 1000002){
if($.inArray(row.userCode, uncheckUserCode) !== -1)
{
return { checked : false };
}
var selectedRows = table.rememberSelecteds[table.options.id];
func = $.inArray('check', ['check', 'check-all']) > -1 ? 'union' : 'difference';
if($.common.isNotEmpty(selectedRows)) {
table.rememberSelecteds[table.options.id] = _[func](selectedRows, row);
} else {
table.rememberSelecteds[table.options.id] = _[func]([], row);
}
return { checked : true };
}
return { checked : false };
}
},
{
field : 'userId',
title : '用户ID'
},
{
field : 'userCode',
title : '用户编号'
},
{
field : 'userName',
title : '用户姓名'
},
{
field : 'userPhone',
title : '用户手机'
},
{
field : 'userEmail',
title : '用户邮箱'
},
{
field : 'userBalance',
title : '用户余额'
},
{
field: 'status',
title: '用户状态',
align: 'center',
formatter: function(value, row, index) {
return $.table.selectDictLabel(datas, value);
}
},
{
title: '操作',
align: 'center',
formatter: function(value, row, index) {
var actions = [];
actions.push('<a class="btn btn-success btn-xs" href="#"><i class="fa fa-edit"></i>编辑</a> ');
actions.push('<a class="btn btn-danger btn-xs" href="#"><i class="fa fa-remove"></i>删除</a>');
return actions.join('');
}
}]
};
$.table.init(options);
});
var uncheckUserCode = [];
$("#bootstrap-table").on("uncheck.bs.table uncheck-all.bs.table", function (e, rows) {
if(rows.length > 0) {
for (var index in rows) {
uncheckUserCode.unshift(rows[index].userCode);
}
} else {
uncheckUserCode.unshift(rows.userCode);
}
});
$("#bootstrap-table").on("check.bs.table check-all.bs.table", function (e, rows) {
if(rows.length > 0) {
for (var index in rows) {
deleteItem(rows[index].userCode);
}
} else {
deleteItem(rows.userCode);
}
});
function deleteItem(item) {
for (var key in uncheckUserCode) {
if (uncheckUserCode[key] === item) {
uncheckUserCode.splice(key, 1)
}
}
}
// 选中数据
function checkItem(){
// var arrays = $.table.selectColumns("userId");
var arrays = $.table.selectColumns("userCode");
alert(arrays);
}
</script>
</body>
</html>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
# 页面如何一次初始化多个表格
在options
中添加id
参数,如果有按钮组也需要添加toolbar
。
// 表格1
var options = {
id: "bootstrap-table1",
toolbar: "toolbar1",
// 省略 ....
};
$.table.init(options);
// 表格2
var options = {
id: "bootstrap-table2",
toolbar: "toolbar2",
// 省略 ....
};
$.table.init(options);
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 表格底部合计列拖动显示问题
在options
中添加onLoadSuccess
参数。
onLoadSuccess: onLoadSuccess,
2
// 监听表体fixed-table-body滚动事件,赋值给表尾fixed-table-footer
function onLoadSuccess() {
$(".fixed-table-body").on("scroll",function(){
var sl=this.scrollLeft;
$(this).next()[0].scrollLeft = sl;
})
}
2
3
4
5
6
7
# 表格操作列传递行数据的对象
传递JSON字符串
actions.push('<a class="btn btn-success btn-xs href="javascript:void(0)" onclick="edit(' + JSON.stringify(row).replace(/"/g, '"') + ')"><i class="fa fa-edit"></i>编辑</a> ');
传递JSON对象
actions.push("<a class='btn btn-success btn-xs href='javascript:void(0)' onclick='edit(" + JSON.stringify(row) + ")'><i class='fa fa-edit'></i>编辑</a> ");
获取xxxx字段
function edit(row) {
alert(row.xxxx);
}
2
3
# 日期控件初始化时间并格式化
使用thymeleaf在页面直接获取当前时间并格式化输出
<input type="text" th:value="${#dates.format(new java.util.Date(), 'yyyy-MM-dd')}" />
<a th:text="${#dates.format(new java.util.Date().getTime(), 'yyyy-MM-dd HH:mm:ss')}">time</a>
2
# 如何调整首页左侧菜单栏宽度
调整style.css
对应样式宽度,例如宽度200修改成250
body.fixed-sidebar .navbar-static-side, body.canvas-menu .navbar-static-side {
width: 250px;
}
2
3
nav .logo {
width: 250px;
}
2
3
#page-wrapper {
margin: 0 0 0 250px;
}
2
3
# 如何默认显示表格卡片视图
在options
中添加 mobileResponsive
cardView
参数
mobileResponsive: false,
cardView: true,
2
# 编辑和删除操作按钮不可用
这种情况一般是因为第一列不是唯一键或formatter
序号造成的。解决方案如下,指定唯一列属性 配合删除/修改使用 未指定则使用表格行首列
在options
中添加 uniqueId
参数,userId
修改成你表的唯一列字段。
uniqueId: 'userId',
# Tomcat部署多个War包项目异常
default-domain
的值不一样就可以了
在application.yml
里面配置上
spring:
jmx:
default-domain: applicationname
2
3
# 部署多个项目Ehcache缓存异常
同一服务器部署多个项目,有可能会导致shiro-activeSessionCache
缓存冲突。
net.sf.ehcache.ObjectExistsException: Cache shiro-activeSessionCache already exists
可以在ShiroConfig
设置不同名称区分一下就可以了。
public OnlineSessionDAO sessionDAO()
{
OnlineSessionDAO sessionDAO = new OnlineSessionDAO();
sessionDAO.setActiveSessionsCacheName("缓存名字");
return sessionDAO;
}
2
3
4
5
6
# Tomcat临时目录tmp抛错误异常
首先,我们应该知道,对于http POST请求来说,它需要使用这个临时目录来存储post数据。
其次,因为该目录是挂在到/temp目录下的临时文件,那么对于一些OS系统,像centOS将经常删除这个临时目录,所有导致该目录不存在了
解决方案
1.在application.yml
文件中设置multipart
location
,并重启项目
spring:
http:
multipart:
location: /data/upload_tmp
2
3
4
2.在application.yml
文件中设置
server
tomcat:
basedir: /tmp/tomcat
2
3
3.在配置文件添加bean
@Bean
public MultipartConfigElement multipartConfigElement() {
MultipartConfigFactory factory = new MultipartConfigFactory();
factory.setLocation("/tmp/tomcat");
return factory.createMultipartConfig();
}
2
3
4
5
6
4.添加启动参数-java.tmp.dir=/path/to/application/temp/
,并重启。
# 如何部署配置支持https访问
Nginx
配置为例,完整流程如下
申请下载ssl证书 证书有很多种,申请成功后会得到一个压缩包,里面有2个证书
1、安装OpenSSL
yum -y install openssl openssl-devel
2、运行添加ssl
模块
./configure --prefix=/usr/local/nginx --with-http_ssl_module
3、配置完成后,运行命令
make
4、然后将刚刚编译好的nginx
覆盖掉原有的nginx
(这个时候nginx要停止状态)
cp objs/nginx /usr/local/nginx/sbin/
5、复制crt
证书文件和key
私钥文件到Nginx
服务器/usr/local/nginx/conf
目录(此处为 Nginx
默认安装目录,请根据实际情况操作)下。
6、编辑 Nginx
根目录下的 conf/nginx.conf
文件。添加内容如下:
# https 服务配置
server {
# 侦听80端口
listen 443 default ssl;
ssl on;
#证书文件名称
ssl_certificate 1_ruoyi.vip_bundle.crt;
#私钥文件名称
ssl_certificate_key 2_ruoyi.vip.key;
# 定义访问域名
server_name ruoyi.vip;
location / {
# 存放了静态页面的根目录
root /home/ruoyi/projects/static-web;
# 默认主页
index index.html;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
7、重启Nginx
通过https
访问 https://ruoyi.vip
8、如需把http
的域名请求转成https
,添加rewrite
rewrite ^(.*) https://$server_name$1 permanent;
9、解决重定向后https变成了http 的问题
proxy_redirect http:// https://;
# 特殊字符串被过滤的解决办法
默认所有的都会过滤脚本,可以在application.yml
配置xss.excludes
属性排除URL
# 防止XSS攻击
xss:
# 过滤开关
enabled: true
# 排除链接(多个用逗号分隔)
excludes: /system/notice/*
# 匹配链接
urlPatterns: /system/*,/monitor/*,/tool/*
2
3
4
5
6
7
8
# 进入首页如何自动展开某菜单
例如,进入自动打开用户管理,调用applyPath
,填入你请求菜单对应的url地址。
applyPath("/system/role")
例如,进入系统自动跳转当前用户所拥有的第一菜单。
var menus= [[${menus}]];
applyPath(menus[0].children[0].url);
2
# 进入首页如何默认记忆控制台
例如用户退出后,下次登陆系统,能默认打开之前工作路径。
可以在index.html
,index-topnav.html
,去掉window.performance.navigation.type == 1
if($.common.equals("history", mode) && window.performance.navigation.type == 1)
换成
if($.common.equals("history", mode))
# 首页侧边栏如何不自动收缩
1、index.js
里面的$('#side-menu').metisMenu();
,替换成如下:
$('#side-menu').metisMenu({
// 是否自动展开
toggle: false,
});
2
3
4
2、去掉菜单联动index.html
,index-topnav.html
。
// 是否页签与菜单联动
var isLinkage = false;
2
3、index.js
里面的menuItem
,syncMenuTab
方法去掉下面三行代码。
// $('.nav ul').removeClass("in");
// $dataObj.parents("ul").addClass("in")
// $dataObj.parents("li").addClass("active").siblings().removeClass("active").find('li').removeClass("active");
2
3
# 页面输出对象日期不准确问题
@JsonFormat
设置默认时区timezone
是GMT
,如果实体类中日期字段属性使用了注解,例如:
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private Date createTime;
2
thymeleaf
使用[[${Object}]]
输出对象会调用注解@JsonFormat
进行日期格式化,如果@JsonFormat
没有声明使用的时区,格式化日期时将使用默认的时区GMT
,会导致输出日期与实际日期少8
小时。
因此,在使用thymeleaf
的[[${Object}]]
输出对象时,Java
实体类需要加上时区声明,例如:
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone ="GMT+8")
private Date createTime;
2
避免出现日期输出不准确问题。
# 打包如何分离jar包和资源文件
特殊情况需要分离lib
和resouce
可以修改ruoyi-admin
参考如下
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>ruoyi</artifactId>
<groupId>com.ruoyi</groupId>
<version>4.4.0</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<packaging>jar</packaging>
<artifactId>ruoyi-admin</artifactId>
<description>
web服务入口
</description>
<dependencies>
<!-- SpringBoot集成thymeleaf模板 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<!-- spring-boot-devtools -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
<optional>true</optional> <!-- 表示依赖不会传递 -->
</dependency>
<!-- swagger2-->
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger2</artifactId>
</dependency>
<!--防止进入swagger页面报类型转换错误,排除2.9.2中的引用,手动增加1.5.21版本-->
<dependency>
<groupId>io.swagger</groupId>
<artifactId>swagger-annotations</artifactId>
<version>1.5.21</version>
</dependency>
<dependency>
<groupId>io.swagger</groupId>
<artifactId>swagger-models</artifactId>
<version>1.5.21</version>
</dependency>
<!-- swagger2-UI-->
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger-ui</artifactId>
</dependency>
<!-- Mysql驱动包 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
<!-- 核心模块-->
<dependency>
<groupId>com.ruoyi</groupId>
<artifactId>ruoyi-framework</artifactId>
</dependency>
<!-- 定时任务-->
<dependency>
<groupId>com.ruoyi</groupId>
<artifactId>ruoyi-quartz</artifactId>
</dependency>
<!-- 代码生成-->
<dependency>
<groupId>com.ruoyi</groupId>
<artifactId>ruoyi-generator</artifactId>
</dependency>
</dependencies>
<build>
<!-- jar包名 -->
<finalName>${project.artifactId}</finalName>
<plugins>
<!-- 分离lib -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-dependency-plugin</artifactId>
<executions>
<execution>
<id>copy-dependencies</id>
<phase>package</phase>
<goals>
<goal>copy-dependencies</goal>
</goals>
<configuration>
<!-- 依赖包输出目录,将来不打进jar包里 -->
<outputDirectory>${project.build.directory}/lib</outputDirectory>
<excludeTransitive>false</excludeTransitive>
<stripVersion>false</stripVersion>
<includeScope>runtime</includeScope>
</configuration>
</execution>
</executions>
</plugin>
<!-- copy资源文件 -->
<plugin>
<artifactId>maven-resources-plugin</artifactId>
<executions>
<execution>
<id>copy-resources</id>
<phase>package</phase>
<goals>
<goal>copy-resources</goal>
</goals>
<configuration>
<resources>
<resource>
<directory>src/main/resources</directory>
</resource>
</resources>
<outputDirectory>${project.build.directory}/resources</outputDirectory>
</configuration>
</execution>
</executions>
</plugin>
<!-- 打jar包时忽略配置文件 -->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<configuration>
<excludes>
<exclude>**/*.yml</exclude>
<exclude>**/*.xml</exclude>
</excludes>
</configuration>
</plugin>
<!-- spring boot repackage -->
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<layout>ZIP</layout>
<includes>
<include>
<groupId>non-exists</groupId>
<artifactId>non-exists</artifactId>
</include>
</includes>
</configuration>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
提示
启动命令java -jar -Dloader.path=resources,lib ruoyi-admin.jar
# Linux系统验证码乱码解决方法
在云服务器(少许),或者干净的服务器上,服务器没有安装字体。
1、上传本地的 Arial.ttf (opens new window) 字体
2、此时执行以下三个命令:(建立字体索引信息,更新字体缓存)
mkfontscale
mkfontdir
fc-cache -fv
【1】上传字体到服务器/usr/share/font/myfonts
【2】使mkfontscale
和mkfontdir
命令正常运行,安装
yum install mkfontscale
【3】使fc-cache
命令正常运行。如果提示fc-cache: command not found
yum install fontconfig
【4】此时执行以下三个命令:(建立字体索引信息,更新字体缓存)
mkfontscale mkfontdir fc-cache -fv
【5】刷新页面,重启服务器的软件(不重启不生效)
3、重新刷新你的页面
Docker环境安装字体包
# Linux环境下安装字体
RUN set -xe && apk --no-cache add ttf-dejavu fontconfig
2
# Mac环境下安装字体
RUN apt-get update && apt-get install -y fonts-dejavu fontconfig
2
# 公共数据库定时任务没有被执行
经常会有小伙伴遇到定时任务没有被执行,或者执行了但是报错找不到对应的方法。
定时任务是分布式
的,如果多个机器链接同一个数据库,定时任务会随机在某个机器上跑,所以有时候不是没有被执行,而是被其他机器上执行了。如果你的方法只在本机有,所以会提示找不到对应定时任务的方法。
这种情况或只有一台机器的话可以注释掉ScheduleConfig.java
,这样的话就只会走本机(quartz
相关定时任务的表也不需要),它也不会在去读quartz
表走集群操作。
# 如何设置用户登录会话超时时间
找到ruoyi-admin\src\main\resources
下面的application.yml
配置文件
# Shiro
shiro:
session:
# Session超时时间,-1代表永不过期(默认30分钟)
expireTime: 30
2
3
4
5
# 如何实现用户免密登录配置方法
免密使用的场景,例如短信验证码,第三方应用登录等。下面列出一个简单的实现方法,当然还有更多实现方式可以自己尝试。
1、新增一个登录类型枚举类LoginType
package com.ruoyi.framework.shiro.token;
/**
* 登录类型枚举类
*
* @author ruoyi
*/
public enum LoginType
{
/**
* 密码登录
*/
PASSWORD("password"),
/**
* 免密码登录
*/
NOPASSWD("nopasswd");
private String desc;
LoginType(String desc)
{
this.desc = desc;
}
public String getDesc()
{
return desc;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
2、自定义登录Token
package com.ruoyi.framework.shiro.token;
import org.apache.shiro.authc.UsernamePasswordToken;
/**
* 自定义登录Token
*
* @author ruoyi
*/
public class UserToken extends UsernamePasswordToken
{
private static final long serialVersionUID = 1L;
private LoginType type;
public UserToken()
{
}
public UserToken(String username, String password, LoginType type, boolean rememberMe)
{
super(username, password, rememberMe);
this.type = type;
}
public UserToken(String username, LoginType type)
{
super(username, "", false, null);
this.type = type;
}
public UserToken(String username, String password, LoginType type)
{
super(username, password, false, null);
this.type = type;
}
public LoginType getType()
{
return type;
}
public void setType(LoginType type)
{
this.type = type;
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
3、对应Realm
中添加登录类型判断,例如UserRealm
(这里演示公用一个realm
,如单独有免密realm
不需要)
/**
* 登录认证
*/
@Override
protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken token) throws AuthenticationException
{
UserToken upToken = (UserToken) token;
LoginType type = upToken.getType();
String username = upToken.getUsername();
String password = "";
if (upToken.getPassword() != null)
{
password = new String(upToken.getPassword());
}
User user = null;
try
{
if (LoginType.PASSWORD.equals(type))
{
user = loginService.login(username, password);
}
else if (LoginType.NOPASSWD.equals(type))
{
user = loginService.login(username);
}
}
catch (CaptchaException e)
{
throw new AuthenticationException(e.getMessage(), e);
}
catch (UserNotExistsException e)
{
throw new UnknownAccountException(e.getMessage(), e);
}
catch (UserPasswordNotMatchException e)
{
throw new IncorrectCredentialsException(e.getMessage(), e);
}
catch (UserPasswordRetryLimitExceedException e)
{
throw new ExcessiveAttemptsException(e.getMessage(), e);
}
catch (UserBlockedException e)
{
throw new LockedAccountException(e.getMessage(), e);
}
catch (RoleBlockedException e)
{
throw new LockedAccountException(e.getMessage(), e);
}
catch (Exception e)
{
log.info("对用户[" + username + "]进行登录验证..验证未通过{}", e.getMessage());
throw new AuthenticationException(e.getMessage(), e);
}
SimpleAuthenticationInfo info = new SimpleAuthenticationInfo(user, password, getName());
return info;
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
4、LoginService
添加login
方法,去掉密码验证。
/**
* 登录
*/
public User login(String username)
{
// 验证码校验
if (!StringUtils.isEmpty(ServletUtils.getRequest().getAttribute(ShiroConstants.CURRENT_CAPTCHA)))
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.jcaptcha.error")));
throw new CaptchaException();
}
// 用户名或密码为空 错误
if (StringUtils.isEmpty(username))
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("not.null")));
throw new UserNotExistsException();
}
// 用户名不在指定范围内 错误
if (username.length() < UserConstants.USERNAME_MIN_LENGTH
|| username.length() > UserConstants.USERNAME_MAX_LENGTH)
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.password.not.match")));
throw new UserPasswordNotMatchException();
}
// 查询用户信息
User user = userService.selectUserByLoginName(username);
if (user == null)
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.not.exists")));
throw new UserNotExistsException();
}
if (UserStatus.DELETED.getCode().equals(user.getDelFlag()))
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.password.delete")));
throw new UserDeleteException();
}
if (UserStatus.DISABLE.getCode().equals(user.getStatus()))
{
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_FAIL, MessageUtils.message("user.blocked", user.getRemark())));
throw new UserBlockedException();
}
AsyncManager.me().execute(AsyncFactory.recordLogininfor(username, Constants.LOGIN_SUCCESS, MessageUtils.message("user.login.success")));
recordLoginInfo(user);
return user;
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
5、在对应的登录方法中传入LoginType.NOPASSWD
调用
UserToken token = new UserToken(username, LoginType.NOPASSWD);
Subject subject = SecurityUtils.getSubject();
subject.login(token);
2
3
# 如何处理Long类型精度丢失问题
当字段实体类为Long
类型且值超过前端js
显示的长度范围时会导致前端回显错误,解决方案如下
1、使用JsonSerialize
注解序列化的时候把Long自动转为String
(针对单个属性)
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
@JsonSerialize(using = ToStringSerializer.class)
private Long xxx;
2
3
4
5
2、添加JacksonConfig
配置全局序列化(针对所有属性)
package com.ruoyi.framework.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.databind.MapperFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
/**
* Jackson配置
*
* @author ruoyi
*
*/
@Configuration
public class JacksonConfig
{
@Bean
public MappingJackson2HttpMessageConverter jackson2HttpMessageConverter()
{
final Jackson2ObjectMapperBuilder builder = new Jackson2ObjectMapperBuilder();
builder.serializationInclusion(JsonInclude.Include.NON_NULL);
final ObjectMapper objectMapper = builder.build();
SimpleModule simpleModule = new SimpleModule();
// Long 转为 String 防止 js 丢失精度
simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
objectMapper.registerModule(simpleModule);
// 忽略 transient 关键词属性
objectMapper.configure(MapperFeature.PROPAGATE_TRANSIENT_MARKER, true);
// 时区设置
objectMapper.setTimeZone(TimeZone.getTimeZone("GMT+8"));
objectMapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
return new MappingJackson2HttpMessageConverter(objectMapper);
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
# 如何修改超级管理员登录密码
1、如果是自己知道超级管理员的密码且需要修改的情况。
默认口令 admin/admin123
,可以登录后在首页个人中心修改密码。
2、如果自己忘记了超级管理员的密码可以重新生成秘钥替换数据库密码。
public static void main(String[] args)
{
// 第一个参数为账户名 第二个参数为密码 第三个参数为盐对应用户表salt(如果没有可以不用填)
System.out.println(new PasswordService().encryptPassword("admin", "admin123", "111111"));
}
2
3
4
5
# 如何修改数据监控登录账户密码
控制台管理用户名和密码默认为ruoyi/123456
找到ruoyi-admin\src\main\resources
下面的application-druid.yml
配置文件
找到如下节点配置,设置控制台账号密码
# 控制台管理用户名和密码
login-username: 你的监控台账号
login-password: 你的监控台密码
2
3
# 分页插件如何手写count查询支持
增加countSuffix
count 查询后缀配置参数,该参数是针对PageInterceptor
配置的,默认值为_COUNT
。
分页插件会优先通过当前查询的msId + countSuffix
查找手写的分页查询。
如果存在就使用手写的count
查询,如果不存在,仍然使用之前的方式自动创建count
查询。
例如,如果存在下面两个查询:
<select id="selectLeftjoin" resultType="com.github.pagehelper.model.User">
select a.id,b.name,a.py from user a
left join user b on a.id = b.id
order by a.id
</select>
<select id="selectLeftjoin_COUNT" resultType="Long">
select count(distinct a.id) from user a
left join user b on a.id = b.id
</select>
2
3
4
5
6
7
8
9
10
上面的countSuffix
使用的默认值_COUNT
,分页插件会自动获取到selectLeftjoin_COUNT
查询,这个查询需要自己保证结果数正确。
返回值的类型必须是resultType="Long"
,入参使用的和selectLeftjoin
查询相同的参数,所以在SQL
中要按照selectLeftjoin
的入参来使用。
因为selectLeftjoin_COUNT
方法是自动调用的,所以不需要在接口提供相应的方法,如果需要单独调用,也可以提供。
# 如何修改成自定义的Cookie名称
在ShiroConfig
中sessionManager
方法添加,其中ruoyi
为设置的Cookie
名称
// 自定义 Cookie
manager.setSessionIdCookie(new SimpleCookie("ruoyi"));
2
# 如何格式化前端日期时间戳内容
对应一些时间格式需要在前端进行格式化操作情况,解决方案如下
1、后端使用JsonFormat
注解格式化日期,时间戳yyyy-MM-dd HH:mm:ss
/** 创建时间 */
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private Date time;
2
3
2、前端使用dateFormat
方法格式化日期,时间戳yyyy-MM-dd HH:mm:ss
{
field: 'createTime',
title: '创建时间',
formatter: function(value, row, index) {
return $.common.dateFormat(value, "yyyy-MM-dd HH-mm-ss");
}
},
2
3
4
5
6
7
# 使用https上传文件返回https路径
通过https
请求进行文件上传却返回http
的文件链接地址,主要原因是通过request.getRequestURL
获取导致的。
我们可以在nginx
配置location
处加上
proxy_set_header X-Forwarded-Scheme $scheme;
然后代码通过request.getHeader("X-Forwarded-Scheme")
获取真实的scheme
String scheme = request.getHeader("X-Forwarded-Scheme");
String serverName = request.getServerName();
int port = request.getServerPort();
String path = request.getContextPath();
String basePath = scheme + "://" + serverName + path;
2
3
4
5
# 使用Velocity模板引擎兼容$符号
当我们服务器端使用velocity
模板来渲染前端页面的时候,而前端使用jquery,vue,angular等等也使用$运算符渲染变量,那么就会产生冲突,
对于这种特殊情况需要加入新的指令#[[您前端不需要让velocity处理的内容]]#
,可以完美解决这个问题。
示例:
// 无法解析
let list = (params) => vm.$u.get("/${moduleName}/${businessName}/list", params);
// 正常解析
let list = (params) => vm.#[[$u]]#.get("/${moduleName}/${businessName}/list", params);
2
3
4
5
# 如何解决多数据源事务的一致性
# 如何优雅的关闭后台系统服务
# 如何使用Redis实现集群会话管理
# 如何使用Jwt实现登录授权访问控制
# 如何解决导出使用下载插件出现异常
导出文件的逻辑是先创建一个临时文件,等待前端请求下载结束后马上删除这个临时文件。但是有些下载插件,例如迅雷(他们是二次下载),这个时候文件已经删除,会导致异常,找不到文件。
解决方案:如果有硬性要求话,可以把所有的导出都改成流的形式返回给前端,不采用临时文件的方法。
// 默认的post请求 生成临时文件
@PostMapping("/export")
@ResponseBody
public AjaxResult export(Xxxx xxxx)
{
List<Xxxx> list = xxxxService.selectXxxxList(xxxx);
ExcelUtil<Xxxx> util = new ExcelUtil<Xxxx>(Xxxx.class);
return util.exportExcel(list, "xxxx");
}
2
3
4
5
6
7
8
9
修改相关导出文件java
代码和ry-ui.js
通用导出方法
// 通过流的形式返回给前端
@GetMapping("/export")
@ResponseBody
public void export(HttpServletResponse response, Xxxx xxxx) throws IOException
{
List<Xxxx> list = xxxxService.selectXxxxList(xxxx);
ExcelUtil<Xxxx> util = new ExcelUtil<Xxxx>(Xxxx.class);
util.exportExcel(response, list, "xxxx");
}
2
3
4
5
6
7
8
9
// ry-ui.js导出数据修改成get请求方式
exportExcel: function(formId) {
table.set();
$.modal.confirm("确定导出所有" + table.options.modalName + "吗?", function() {
var currentId = $.common.isEmpty(formId) ? $('form').attr('id') : formId;
var params = $("#" + table.options.id).bootstrapTable('getOptions');
var dataParam = $("#" + currentId).serializeArray();
dataParam.push({ "name": "orderByColumn", "value": params.sortName });
dataParam.push({ "name": "isAsc", "value": params.sortOrder });
window.location.href = table.options.exportUrl + "?" + $.param(dataParam);
});
},
2
3
4
5
6
7
8
9
10
11
12
# 如何解决请求地址存在中文出现异常
shrio1.7.0
版本才会出现,对于请求地址需要中文的情况下可以做以下处理。
1、自定义CustomShiroFilterFactoryBean.java
,设置setBlockNonAscii
属性为false
package com.ruoyi.framework.config;
import org.apache.shiro.spring.web.ShiroFilterFactoryBean;
import org.apache.shiro.web.filter.InvalidRequestFilter;
import org.apache.shiro.web.filter.mgt.DefaultFilter;
import org.apache.shiro.web.filter.mgt.FilterChainManager;
import org.apache.shiro.web.filter.mgt.FilterChainResolver;
import org.apache.shiro.web.filter.mgt.PathMatchingFilterChainResolver;
import org.apache.shiro.web.mgt.WebSecurityManager;
import org.apache.shiro.web.servlet.AbstractShiroFilter;
import org.apache.shiro.mgt.SecurityManager;
import org.springframework.beans.factory.BeanInitializationException;
import javax.servlet.Filter;
import java.util.Map;
/**
* 自定义ShiroFilterFactoryBean解决资源中文路径问题
*
* @author ruoyi
*/
public class CustomShiroFilterFactoryBean extends ShiroFilterFactoryBean
{
@Override
public Class<MySpringShiroFilter> getObjectType()
{
return MySpringShiroFilter.class;
}
@Override
protected AbstractShiroFilter createInstance() throws Exception
{
SecurityManager securityManager = getSecurityManager();
if (securityManager == null)
{
String msg = "SecurityManager property must be set.";
throw new BeanInitializationException(msg);
}
if (!(securityManager instanceof WebSecurityManager))
{
String msg = "The security manager does not implement the WebSecurityManager interface.";
throw new BeanInitializationException(msg);
}
FilterChainManager manager = createFilterChainManager();
// Expose the constructed FilterChainManager by first wrapping it in a
// FilterChainResolver implementation. The AbstractShiroFilter implementations
// do not know about FilterChainManagers - only resolvers:
PathMatchingFilterChainResolver chainResolver = new PathMatchingFilterChainResolver();
chainResolver.setFilterChainManager(manager);
Map<String, Filter> filterMap = manager.getFilters();
Filter invalidRequestFilter = filterMap.get(DefaultFilter.invalidRequest.name());
if (invalidRequestFilter instanceof InvalidRequestFilter)
{
// 此处是关键,设置false跳过URL携带中文400,servletPath中文校验bug
((InvalidRequestFilter) invalidRequestFilter).setBlockNonAscii(false);
}
// Now create a concrete ShiroFilter instance and apply the acquired SecurityManager and built
// FilterChainResolver. It doesn't matter that the instance is an anonymous inner class
// here - we're just using it because it is a concrete AbstractShiroFilter instance that accepts
// injection of the SecurityManager and FilterChainResolver:
return new MySpringShiroFilter((WebSecurityManager) securityManager, chainResolver);
}
private static final class MySpringShiroFilter extends AbstractShiroFilter
{
protected MySpringShiroFilter(WebSecurityManager webSecurityManager, FilterChainResolver resolver)
{
if (webSecurityManager == null)
{
throw new IllegalArgumentException("WebSecurityManager property cannot be null.");
}
else
{
this.setSecurityManager(webSecurityManager);
if (resolver != null)
{
this.setFilterChainResolver(resolver);
}
}
}
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
2、替换ShiroConfig.java
中的过滤器配置为自定义
// 默认的
ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
ShiroFilterFactoryBean更换为CustomShiroFilterFactoryBean
// 自定义的
CustomShiroFilterFactoryBean shiroFilterFactoryBean = new CustomShiroFilterFactoryBean();
2
3
4
5
6
7
# 如何解决用户登录记住我出现的异常
当用户有多个角色的时候(超过5个),用户信息序列化后长度会比较长,Base64
编码后长度会超过4096
,如果此时启用rememberMe
功能,系统就要把这个信息存入cookie
中,可是浏览器一般限制cookie
长度在4096
长度之内,这样机就会导致存入cookie
失败,从而导致rememberMe
功能在这种情况下失效报错。
如果遇到角色过多,导致的cookie
请求头数据太大导致报错,可以添加配置文件。
server:
max-http-header-size: 65546
2
如果遇到nginx
部署,登录时出现502
,也是请求头过大导致,可以配置如下属性。
# 设置代理服务器(nginx)保存用户头信息的缓冲区大小
proxy_buffer_size 8k;
2