- 浏览: 53973 次
最新评论
目标:面向多区域用户
添加更新记录:用户输入本区域时间, 数据库存储转换后的标准时间
查看记录: 数据库时间相同,view层根据用户区域显示相应的时间
说明本项目要求日期格式固定,不按照用户local显示. 如果要求按照用户local显示日期格式则需要使用如下的format, 可参考liferay源码
分析
liferay 提供了用户自定义zone (控制面板->setting->zone, 详见"liferay区域设置")。
页面显示时可以根据用户zone格式化取得的时间。
向后台发送数据时根据客户zone取得客户输入时间,转换为tomcat时间后再传入数据库
原则(强烈建议
对于日期类型从后台向前台读取的过程中, 在business层不要做类型转换(例如试图转为long或String ), 只有在页面显示的时候才将其转为合适的日期格式和区域。
对于其他类型做同样的建议,如double, longtext,例如在business层中不要试图截取小数点位数和文本长度,在页面根据需要截取。如果业务上有特殊要求除外(如财务运算)
timezone 涉及的 web app 层次
1.view (jsp)
2.business(portletaction + service + finder/persistence)
//此处因为portlect负责读取输入的日期数据, finder负责写入日期数据,为方便说明timezone的传递,将其归入business层次, 勿纠结于层次划分
3.DB (mysql)
------------- util code -----------
DateUtil类提供公共方法获取包含用户时区和local的format
html/init.jsp提供公共format,包含用户zone和local 也可以直接使用DateUtil类提供的方法
html/init.jsp
------------- zone设置 -----------
用户A,leferay中设置 zone = GMT+7
TOMCAT zone = GMT+8
mysql zone = system = GMT+9
设定
用户试图输入日期 GMT+8 2013-01-01 12:00:00 保存至数据库
执行操作时web server 系统时间 GMT+8 2013-01-01 12:10:10
// web server 系统时区可任意设置,不影响以下结果
------------- 添加更新记录 -----------
VIEW 层
用户A输入日期 " 2013-01-01 11:00:00",提交 (GMT+7)
BUSINESS 层
获取页面输入日期,更新数据库
说明:
(1) dateFormatDateTime 的默认时区为用户时区GMT+7
(2) dateRead 获取时间为用户输入时间, 按照用户时区GMT+7转换为日期后再转换为tomcat时区GMT+8。
(3) new Date() 获取时间为 web server 系统时间, 时区为tomcat设置的时区GMT+8。
注意:
ParamUtil.getDate(actionRequest, "date", dateFormatDateTime)方法实际隐含了两次转换
(1) 用户输入时间 2013-01-01 12:00:00,按用户时区(dateFormatDateTime) 转换为日期类型 GMT+7 2013-01-01 11:00:00
(2) GMT+7 2013-01-01 11:00:00 转换为tomcat时区 GMT+8 2013-01-01 12:00:00
DB层
存储日期
数据库表中存储的dateInput和dateNew分别为
dateInput = 2013-01-01 12:00:00
dateNew = 2013-01-01 04:10:10
注意:
数据库本身有时区定义,参见mysql时区定义。
mysql的时区定义会对now()获得字符串产生影响。
可以认为数据库存储的时间类型字段不包含时区信息,insert/update字符串即是所见字符串
例如:
传入参数:
dateInput = GMT+8 2013-01-01 12:00:00
dateNew = GMT+8 2013-01-01 12:10:10
mysql-〉 select dateInput, dateNew
结果: 2013-01-01 12:00:00 2013-01-01 12:10:10
传入参数:
dateInput = GMT+9 2013-01-01 12:00:00
dateNew = GMT+9 2013-01-01 12:10:10
mysql-〉 select dateInput, dateNew
结果不变: 2013-01-01 12:00:00 2013-01-01 12:10:10
------------- 读取数据到页面 -----------
DB层
存储日期
数据库表中存储的dateInput和dateNew分别为
dateInput = 2013-01-01 12:00:00
dateNew = 2013-01-01 12:10:10
BUSINESS 层
获取数据库存储日期,返回到页面
结果:
dateInput = GMT+8 2013-01-01 12:00:00
dateNew = GMT+8 2013-01-01 12:10:10
VIEW 层
显示日期
<input value=<%=DateUtil.safeDateTime(map.get("dateInput"), locale, timeZone)%> />;
<input value=<%=dateFormatDateTime.format(dateNew)%> />;
<input value=<%=dateFormatDateTime.format(newDate())%> />;
页面显示结果
dateInput : 2013-01-01 11:00:00
dateNew : 2013-01-01 12:10:10
dateNew : 2013-01-01 12:10:10
------------- 总结 -----------
1. web app server 时区与 web server时区无关。
2. db server时区对程序无影响(不要使用now()函数更新数据库)
3. new date(), new dateformat() 生成对象时区为web app server 时区
4. 如果不考虑用户时区,则只需要设定tomcat时区即可保持系统一致。代码中适用new dateformat()格式化日期,部另外设置时区。
------------- [color=blue] END 2013-01-07 -----------
添加更新记录:用户输入本区域时间, 数据库存储转换后的标准时间
查看记录: 数据库时间相同,view层根据用户区域显示相应的时间
说明本项目要求日期格式固定,不按照用户local显示. 如果要求按照用户local显示日期格式则需要使用如下的format, 可参考liferay源码
Format dateFormat = FastDateFormatFactoryUtil.getSimpleDateFormat("MMMM, yyyy", locale); dateFormat .setTimeZone(timeZone); Format timeFormat = FastDateFormatFactoryUtil.getSimpleDateFormat("h:mma", locale); timeFormat .setTimeZone(timeZone);
分析
liferay 提供了用户自定义zone (控制面板->setting->zone, 详见"liferay区域设置")。
页面显示时可以根据用户zone格式化取得的时间。
向后台发送数据时根据客户zone取得客户输入时间,转换为tomcat时间后再传入数据库
原则(强烈建议
对于日期类型从后台向前台读取的过程中, 在business层不要做类型转换(例如试图转为long或String ), 只有在页面显示的时候才将其转为合适的日期格式和区域。
对于其他类型做同样的建议,如double, longtext,例如在business层中不要试图截取小数点位数和文本长度,在页面根据需要截取。如果业务上有特殊要求除外(如财务运算)
timezone 涉及的 web app 层次
1.view (jsp)
2.business(portletaction + service + finder/persistence)
//此处因为portlect负责读取输入的日期数据, finder负责写入日期数据,为方便说明timezone的传递,将其归入business层次, 勿纠结于层次划分
3.DB (mysql)
------------- util code -----------
DateUtil类提供公共方法获取包含用户时区和local的format
public class DateUtil { public static SimpleDateFormat getDateFormatDate(Locale locale, TimeZone timeZone) { SimpleDateFormat dateFormatDate = new SimpleDateFormat("yyyy-MM-dd", locale); dateFormatDate.setTimeZone(timeZone); return dateFormatDate; } public static SimpleDateFormat getDateFormatDateTime(Locale locale, TimeZone timeZone) { SimpleDateFormat dateFormatDate = new SimpleDateFormat( "yyyy-MM-dd HH:mm:ss", locale); dateFormatDate.setTimeZone(timeZone); return dateFormatDate; } public static String safeDate(Object obj, Locale locale, TimeZone timeZone) { SimpleDateFormat dfCustomer = getDateFormatDate(locale, timeZone); SimpleDateFormat dfServer = new SimpleDateFormat("yyyy-MM-dd"); String str = ""; try { if (Validator.isNotNull(obj)) { if (obj instanceof Date) { str = dfCustomer.format(obj); //important } else { str = dfCustomer.format(dfServer.parse(obj.toString())); //important } } } catch (Exception e) { str = ""; } return str; } public static String safeDateTime(Object obj, Locale locale, TimeZone timeZone) { SimpleDateFormat dfCustomer = getDateFormatDateTime(locale, timeZone); SimpleDateFormat dfServer = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String str = ""; try { if (Validator.isNotNull(obj)) { if (obj instanceof Date) { str = dfCustomer.format(obj); //important } else { str = dfCustomer.format(dfServer.parse(obj.toString())); //important } } } catch (Exception e) { str = ""; } return str; } }
html/init.jsp提供公共format,包含用户zone和local 也可以直接使用DateUtil类提供的方法
html/init.jsp
SimpleDateFormat dateFormatDate = DateUtil new SimpleDateFormat("yyyy-MM-dd", locale); dateFormatDate.setTimeZone(timeZone); SimpleDateFormat dateFormatDateTime = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", locale); dateFormatDateTime.setTimeZone(timeZone);
------------- zone设置 -----------
用户A,leferay中设置 zone = GMT+7
TOMCAT zone = GMT+8
mysql zone = system = GMT+9
设定
用户试图输入日期 GMT+8 2013-01-01 12:00:00 保存至数据库
执行操作时web server 系统时间 GMT+8 2013-01-01 12:10:10
// web server 系统时区可任意设置,不影响以下结果
------------- 添加更新记录 -----------
VIEW 层
用户A输入日期 " 2013-01-01 11:00:00",提交 (GMT+7)
BUSINESS 层
获取页面输入日期,更新数据库
//portlet: ThemeDisplay themeDisplay = (ThemeDisplay)actionRequest.getAttribute( WebKeys.THEME_DISPLAY); SimpleDateFormat dateFormatDateTime = DateUtil.getDateFormatDateTime(themeDisplay.getLocale(), themeDisplay.getTimeZone()); Date dateRead = ParamUtil.getDate(actionRequest, "date", dateFormatDateTime); Date dateNew = new Date(); XXXLocalServiceUtil.XXX(xx, dateInput, dateNew); /* 上面代码得到的日期为: dateInput = GMT+8 2013-01-01 12:00:00 dateNew = GMT+8 2013-01-01 12:10:10 */ //service: ModelXXX modelxxx = ModelXXXPersistence.create(Id); modelxxx.setdateInput(dateInput); modelxxx.setdateNew(dateNew); ModelXXXPersistence.update(modelxxx, false);
说明:
(1) dateFormatDateTime 的默认时区为用户时区GMT+7
(2) dateRead 获取时间为用户输入时间, 按照用户时区GMT+7转换为日期后再转换为tomcat时区GMT+8。
(3) new Date() 获取时间为 web server 系统时间, 时区为tomcat设置的时区GMT+8。
注意:
ParamUtil.getDate(actionRequest, "date", dateFormatDateTime)方法实际隐含了两次转换
(1) 用户输入时间 2013-01-01 12:00:00,按用户时区(dateFormatDateTime) 转换为日期类型 GMT+7 2013-01-01 11:00:00
(2) GMT+7 2013-01-01 11:00:00 转换为tomcat时区 GMT+8 2013-01-01 12:00:00
DB层
存储日期
数据库表中存储的dateInput和dateNew分别为
dateInput = 2013-01-01 12:00:00
dateNew = 2013-01-01 04:10:10
注意:
数据库本身有时区定义,参见mysql时区定义。
mysql的时区定义会对now()获得字符串产生影响。
可以认为数据库存储的时间类型字段不包含时区信息,insert/update字符串即是所见字符串
例如:
传入参数:
dateInput = GMT+8 2013-01-01 12:00:00
dateNew = GMT+8 2013-01-01 12:10:10
mysql-〉 select dateInput, dateNew
结果: 2013-01-01 12:00:00 2013-01-01 12:10:10
传入参数:
dateInput = GMT+9 2013-01-01 12:00:00
dateNew = GMT+9 2013-01-01 12:10:10
mysql-〉 select dateInput, dateNew
结果不变: 2013-01-01 12:00:00 2013-01-01 12:10:10
------------- 读取数据到页面 -----------
DB层
存储日期
数据库表中存储的dateInput和dateNew分别为
dateInput = 2013-01-01 12:00:00
dateNew = 2013-01-01 12:10:10
BUSINESS 层
获取数据库存储日期,返回到页面
//finder: map.put("dateInput", record[0] == null? "" : record[0]); map.put("dateNew", record[1] == null? "" : record[1]); /* 查询结果集 record[0] = Timestamp GMT+8 2013-01-01 12:00:00 record[1] = Timestamp GMT+8 2013-01-01 12:10:10 */ //service: Date dateInput = (Date)(map.get("dateInput")); Date dateNew = (Date)(map.get("dateNew")); /* 下面一种写法结果与上面代码相同, 但不提倡,finder和service中将Date和String来回转换存在风险SimpleDateFormat dateFormatDateTime = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); Date dateInput = dateFormatDateTime.parse(map.get("dateInput").toString()); Date dateNew = dateFormatDateTime.parse(map.get("dateNew").toString()); */
结果:
dateInput = GMT+8 2013-01-01 12:00:00
dateNew = GMT+8 2013-01-01 12:10:10
VIEW 层
显示日期
<input value=<%=DateUtil.safeDateTime(map.get("dateInput"), locale, timeZone)%> />;
<input value=<%=dateFormatDateTime.format(dateNew)%> />;
<input value=<%=dateFormatDateTime.format(newDate())%> />;
页面显示结果
dateInput : 2013-01-01 11:00:00
dateNew : 2013-01-01 12:10:10
dateNew : 2013-01-01 12:10:10
------------- 总结 -----------
1. web app server 时区与 web server时区无关。
2. db server时区对程序无影响(不要使用now()函数更新数据库)
3. new date(), new dateformat() 生成对象时区为web app server 时区
4. 如果不考虑用户时区,则只需要设定tomcat时区即可保持系统一致。代码中适用new dateformat()格式化日期,部另外设置时区。
------------- [color=blue] END 2013-01-07 -----------
发表评论
-
AOP
2017-02-28 13:14 482http://blog.csdn.net/moreev ... -
liferay 6.1 CE + LDAP 导入自定义字段
2013-12-21 16:06 898导入自定义字段时无论如何尝试都不成功,但是自定义字段的值导 ... -
liferay 6.1 CE LDAP
2013-12-14 14:20 2324goole了许多网页,讲的是五花八门,最后想起来去port ... -
research
2013-12-10 10:42 0<aui:select label="typ ... -
fetch 与 find 的区别
2013-11-11 09:55 907liferay中 findxxxx 与fetchxxx 方 ... -
liferay 一对多 多对多 关系 (model)
2013-11-07 11:24 708通过model层实现 例如 people 与 ad ... -
liferay 一对多 多对多 关系 (build-service)
2013-11-07 11:04 690通过配置service.xml 实现 说明:一对多 ... -
liferay Logical Architecture
2013-09-11 15:20 1184Logical Architecture 详 ... -
liferay 文件下载失败
2013-07-01 14:48 588现象: http://issues.liferay.com/ ... -
403
2013-06-05 16:41 0portal.properties auth.forward ... -
[转] Liferay多数据源配置及开发
2013-06-05 16:16 762http://www.chinasb.org/archives ... -
aui-autocomplete [ 三 ] 源码解析
2013-01-09 09:53 1584懒了, 只翻译几个目前用到的属性,其实英文写的很明白的,需要花 ... -
aui-autocomplete [ 二 ] 验证未通过保留输入值
2013-01-09 09:25 854<% long myUserId = ParamU ... -
aui-autocomplete [ 一 ] override
2013-01-08 13:58 859liferay aui-autocomplete 如: 数据 ... -
Liferay AutoFields 添加事件
2013-01-08 09:30 0new Liferay.AutoFields( ... -
liferay auto_filed
2012-12-03 14:51 807new Liferay.AutoFields( ... -
liferay at java.util.regex.Pattern$Curly.match0
2012-11-21 13:17 923build-service 出现liferay at java ... -
actionurl 参数顺序
2012-10-19 14:16 1507PortletURL editURL = renderResp ... -
java.lang.ClassCastException: java.math.BigInteger cannot be cast to java.lang.
2012-10-19 14:06 4053现象: liferay中查询数据,用到如下语句 select ... -
Unable to find required classes (javax.activation.DataHandler and javax.mail.int
2012-09-04 13:41 10962.1.8 错误原因: 需要mail.jar和activat ...
相关推荐
**AppWeb服务器详解** AppWeb是一款轻量级、高性能的嵌入式Web服务器,它以其高效、可扩展和安全的特性在IT行业中受到广泛关注。"appweb-5.1.0-src.tgz" 是AppWeb的5.1.0版本源代码包,这代表着开发者可以获取到...
在uni-app框架中,开发移动应用时经常需要集成H5页面,通过`<web-view>`组件展示。然而,原生的H5图片长按下载功能在uni-app内可能无法正常工作,因为平台的限制或者API的不兼容。在这种情况下,我们需要自定义处理...
《AppWeb源码分析与应用》 AppWeb是一款开源的嵌入式Web服务器,它以其轻量级、高效能的特点,被广泛应用于各种嵌入式设备和物联网系统中。本篇将围绕“appweb-src-3.2.3-3.tar.gz”这个压缩包中的源代码,深入探讨...
angulr-bootstrap-admin-web-app-with-angularjs 2.0版本来拉。由于最近比较忙所有没有持续更新这个版本。在此特地发一下2.0.1版本的。如果有需要我之前版本的请查看我的资源,移驾之前的资源。绝对官网下载的完整包...
`appweb-src-2.4.0-0.tar.gz`是一个强大的嵌入式Web服务器解决方案,它为资源有限的设备带来了丰富的Web服务功能,包括动态脚本支持和高度的可定制性。开发者可以通过其源代码包进行编译、安装,并根据需求进行配置...
《AppWeb 3.2.2-1 源码解析与应用开发指南》 AppWeb,一个轻量级、高性能的嵌入式Web服务器,以其开源、小巧且功能强大的特性,深受开发者们的喜爱。本篇文章将围绕"appweb-src-3.2.2-1.tar.gz"这一版本的源码进行...
"web-app-manifest-loader" 是一个专门针对Web应用程序清单(Web App Manifest)处理的前端开源库。本文将深入探讨这个库的核心功能、用途以及它如何融入Web开发流程。 Web应用程序清单(Web App Manifest)是一种...
Appweb是一個嵌入式HTTP Web服務器,主要的設計思路是安全。這是直接集成到客戶的應用和設備,便於開發和部署基於Web的應用程序和設備。它正迅速( 3500多要求每秒)和緊湊(從11萬) ,其中包括支持動態網頁製作,...
webpack-web-app-manifest-plugin webpack-web-app-manifest-plugin是一个Webpack插件,可生成PWA清单并与生成的资产JSON集成。如何安装npm install --save-dev webpack-web-app-manifest-plugin 或者yarn add ...
Web-App-Developer-2-by-InfoQ
**AppWeb 2.4.2-2M-src 源码详解** AppWeb 是一款轻量级、高效的嵌入式 Web 服务器,其源码包 "appweb-2.4.2-2M-src" 提供了在各种平台,包括 ARM 和 PC 上搭建和移植的完整解决方案。这款 Web 服务器以其小巧的...
标题“appweb-3.3.2-0-src”揭示了我们要讨论的核心——appweb服务器的一个特定版本,即3.3.2。这是一个开源的、轻量级的Web服务器,专为嵌入式系统设计,适用于各种资源有限的环境。"src"后缀表明我们获得的是源...
<title>uni-app-web-view <!-- 引入uni-app SDK --> <script type="text/javascript" src="https://js.cdn.aliyun.dcloud.net.cn/dev/uni-app/uni.webview.0.1.52.js"> document.addEventListener('...
**AppWeb 6.2.1 源代码详解** AppWeb 是一款开源的嵌入式 Web 服务器,它设计轻量级、高效且可高度定制。标题中的 "appweb-6.2.1-src.tgz" 表明这是一个 AppWeb 的 6.2.1 版本的源代码压缩包,格式为 tar.gz,通常...
AppWeb是完全遵循GPL软件许可协议(GNU General Public License)的开源Web服务器。 AppWeb Web服务器软件在功能上比GoAhead更加丰富和实用, 除了GoAhead已有的功能之外,还有如下特点: 支持服务器端的...
2. **主循环**:在`src/appweb.c`中,你可以找到服务器的主要事件循环,这是AppWeb运行的核心部分。 3. **配置解析**:AppWeb使用`conf.c`和`conf.h`中的函数来解析配置文件。了解这些函数的工作原理,可以方便地...
Add-WindowsFeature Web-Server,Web-WebServer,Web-Common-Http,Web-Static-Content,Web-App-Dev,Web-Asp-Net,Web-Net-Ext,Web-ISAPI-Ext,Web-ISAPI-Filter,Web-Includes,Web-Security,Web-Windows-Auth,Web-...
Add-WindowsFeature Web-Server, Web-WebServer, Web-Common-Http, Web-Static-Content, Web-App-Dev, Web-Asp-Net, Web-Net-Ext, Web-ISAPI-Ext, Web-ISAPI-Filter, Web-Includes, Web-Security, Web-Windows-Auth...
WhatsApp业务API设置脚本(英语) 该存储库包含用于使用docker-compose设置WhatsApp Business API的所有配置脚本。 有关与WhatsApp Business API有关的更多详细信息,请访问以下位置的文档: : 执照WhatsApp业务API...
mall-app-web是一个电商系统的移动端项目,基于uni-app实现。主要包括首页门户、商品推荐、商品搜索、商品展示、购物车、订单流程、会员中心、客户服务、帮助中心等功能。.zip 适合学习/练手、毕业设计、课程设计、...