SpringMvc + Freemarker + JSON + FileUpload + Log4j + Hibernate + c3p0 + oscache
2012-09-03 15:00
507 查看
项目所需jar
第一步,首先是web.xml的配置
其次:applicationContex.xml 配置
第三 springmvc.xml配置
oscache配置 oscache.properties
第四 log4j 配置 log4j.properties
Hibernate 相关
UUID生成方式
二级缓存
第一步 : 执行hql "from Student" 将学生全部查询出来
第二步: 执行 hql "from Student where id = 1" (这样是不会从二级缓存里面去找的)
但是改成 session.get(Student.class , 1); 这样就可以从二级缓存里面找到
JSON相关
首先要现在springmvc.xml中配置好json转换器 ,然后就可以使用json相关的注解和一些方法
其中DateJsonSerelized类的定义如下:
之后在SpringMvc Controller方法中,将该对象作为json返回时,就会根据上面的方式操作该对象
Freemarker相关
ftl文件中访问静态属性和方法
SpringMVC相关
监听器
类型转换器
在需要转换类型的controller类中加入如下方法
接收Map类型参数
文件上传(需要commons-io-2.0.1.jar commons-fileupload-1.2.2.jar 包的支持)
先在spring.mvc配置
AOP
如果系统有地方需要用到AOP ,那么可以自定义注解,以及注解处理类,如下:
先定义注解类
然后定义AOP处理类
antlr-2.7.6.jar aopalliance-1.0.jar asm-3.1.jar aspectjweaver.jar c3p0-0.9.1.jar cglib-2.2.jar commons-collections-3.1.jar commons-fileupload-1.2.2.jar commons-io-2.0.1.jar commons-logging-1.1.1.jar dom4j-1.6.1.jar fastjson-1.1.22.jar freemarker.jar hibernate-jpa-2.0-api-1.0.1.Final.jar hibernate3.jar jackson-annotations-2.0.4.jar jackson-core-2.0.4.jar jackson-core-asl-1.4.2.jar jackson-databind-2.0.4.jar jackson-mapper-asl-1.4.2.jar javassist-3.12.0.GA.jar jstl.jar jta-1.1.jar mail.jar mysql-connector-java-5.1.10-bin.jar org.springframework.aop-3.1.1.RELEASE.jar org.springframework.asm-3.1.1.RELEASE.jar org.springframework.aspects-3.1.1.RELEASE.jar org.springframework.beans-3.1.1.RELEASE.jar org.springframework.context-3.1.1.RELEASE.jar org.springframework.context.support-3.1.1.RELEASE.jar org.springframework.core-3.1.1.RELEASE.jar org.springframework.expression-3.1.1.RELEASE.jar org.springframework.jdbc-3.1.1.RELEASE.jar org.springframework.orm-3.1.1.RELEASE.jar org.springframework.transaction-3.1.1.RELEASE.jar org.springframework.web-3.1.2.RELEASE.jar org.springframework.web.servlet-3.1.1.RELEASE.jar oscache-2.1.jar slf4j-api-1.6.1.jar standard.jar
第一步,首先是web.xml的配置
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5"> <display-name>ajaxchart</display-name> <welcome-file-list> <welcome-file>index.html</welcome-file> <welcome-file>index.htm</welcome-file> <welcome-file>index.jsp</welcome-file> <welcome-file>default.html</welcome-file> <welcome-file>default.htm</welcome-file> <welcome-file>default.jsp</welcome-file> </welcome-file-list> <!-- 配置配置文件路径 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value> classpath*:applicationContext.xml, classpath*:springcxf.xml </param-value> </context-param> <!-- Spring监听器 --> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!-- SpringMvc配置 --> <servlet> <servlet-name>springmvc</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:springmvc.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> <!-- 如果不指定该参数,tomcat系统时不会初始化所有的action,等到第一次访问的时候才初始化 --> </servlet> <servlet-mapping> <servlet-name>springmvc</servlet-name> <url-pattern>*.htm</url-pattern> </servlet-mapping> <!-- 配置全局的错误页面 --> <error-page> <error-code>404</error-code> <location>/404.jsp</location> </error-page> <error-page> <error-code>500</error-code> <location>/404.jsp</location> </error-page> </web-app>
其次:applicationContex.xml 配置
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd"> <context:component-scan base-package="com.zf" /> <context:annotation-config /> <tx:annotation-driven transaction-manager="transactionManager" /> <!-- 注解方式支持事务 (如果要让注解方式也支持事务,就要配置该项)--> <aop:aspectj-autoproxy proxy-target-class="true" /> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close"> <!-- 指定连接数据库的驱动 --> <property name="driverClass" value="com.mysql.jdbc.Driver"/> <!-- 指定连接数据库的URL --> <property name="jdbcUrl" value="jdbc:mysql://192.168.1.140:3306/ajax"/> <!-- 指定连接数据库的用户名 --> <property name="user" value="root"/> <!-- 指定连接数据库的密码 --> <property name="password" value="root"/> <!-- 指定连接数据库连接池的最大连接数 --> <property name="maxPoolSize" value="20"/> <!-- 指定连接数据库连接池的最小连接数 --> <property name="minPoolSize" value="1"/> <!-- 指定连接数据库连接池的初始化连接数 --> <property name="initialPoolSize" value="1"/> <!-- 指定连接数据库连接池的连接的最大空闲时间 --> <property name="maxIdleTime" value="20"/> </bean> <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean" > <property name="dataSource" ref="dataSource"></property> <property name="hibernateProperties"> <props> <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop> <prop key="hibernate.show_sql">true</prop> <prop key="hibernate.query.substitutions">true 1,false 0</prop> <prop key="hibernate.hbm2ddl.auto">update</prop> <prop key="hibernate.cache.use_second_level_cache">true</prop> <prop key="hibernate.cache.provider_class">org.hibernate.cache.OSCacheProvider</prop> <prop key="hibernate.cache.use_query_cache">true</prop> </props> </property> <property name="packagesToScan"> <list> <value>com.zf.pojo</value> </list> </property> </bean> <!-- 事务管理器 --> <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory"></property> </bean> <!-- XML方式配置事务 --> <tx:advice transaction-manager="transactionManager" id="transactionAdvice"> <tx:attributes> <tx:method name="*" propagation="REQUIRED" /> </tx:attributes> </tx:advice> <!-- AOP配置事务 --> <aop:config> <aop:pointcut expression="execution(* com.zf.service.*.*(..))" id="servicePoint"/> <aop:advisor advice-ref="transactionAdvice" pointcut-ref="servicePoint"/> </aop:config> </beans>
第三 springmvc.xml配置
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd"> <context:annotation-config /> <context:component-scan base-package="com.zf.control" /> <aop:aspectj-autoproxy proxy-target-class="true" /> <!-- Freemarker配置 --> <bean class="org.springframework.web.servlet.view.freemarker.FreeMarkerConfigurer"> <property name="templateLoaderPath" value="/" /> <property name="freemarkerSettings"> <props> <prop key="template_update_delay">0</prop> <!-- 测试时设为0,一般设为3600 --> <prop key="defaultEncoding">utf-8</prop> </props> </property> </bean> <!-- Freemarker视图解析器 --> <bean class="org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver"> <property name="viewClass" value="org.springframework.web.servlet.view.freemarker.FreeMarkerView" /> <property name="suffix" value=".ftl" /> <property name="contentType" value="text/html; charset=UTF-8" /> <property name="allowSessionOverride" value="true" /> </bean> <!-- View Resolver (如果用Freemarker 就可以将该视图解析器去掉) --> <!-- <bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"> --> <!-- <property name="viewClass" value="org.springframework.web.servlet.view.JstlView" /> --> <!-- <property name="prefix" value="/" /> 指定Control返回的View所在的路径 --> <!-- <property name="suffix" value=".jsp" /> 指定Control返回的ViewName默认文件类型 --> <!-- </bean> --> <!-- 将OpenSessionInView 打开 --> <bean id="openSessionInView" class="org.springframework.orm.hibernate3.support.OpenSessionInViewInterceptor"> <property name="sessionFactory" ref="sessionFactory"></property> </bean> <!-- 处理在类级别上的@RequestMapping注解--> <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"> <property name="interceptors"> <!-- 配置过滤器 --> <list> <ref bean="openSessionInView" /> </list> </property> </bean> <!-- 配置文件上传 --> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!-- one of the properties available; the maximum file size in bytes --> <property name="maxUploadSize" value="100000" /> </bean> <!-- JSON配置--> <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter" > <property name="messageConverters"> <list> <!-- 该类只有org.springframework.web-3.1.2.RELEASE.jar及以上版本才有 使用该配置后,才可以使用JSON相关的一些注解--> <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"> <property name="objectMapper"> <bean class="com.fasterxml.jackson.databind.ObjectMapper" /> </property> </bean> </list> </property> </bean> </beans>
oscache配置 oscache.properties
cache.memory=true cache.key=__oscache_cache cache.capacity=10000 cache.unlimited.disk=true
第四 log4j 配置 log4j.properties
#将等级为INFO的信息输出到 stuout 和 R两个地方 log4j.rootCategory=INFO, stdout , R #定义输出端的类型可以为如下的值: #org.apache.log4j.ConsoleAppender 控制台 #org.apache.log4j.FileAppender 文件 #org.apache.log4j.DailyRollingFileAppender 每天产生一个文件 #org.apache.log4j.RollingFileAppender 文件大小到达一定尺寸后才生新的文件 #org.apache.log4j.WriterAppender 将日志信息以流的方式发送到任意指定的地方 log4j.appender.stdout=org.apache.log4j.ConsoleAppender #定义stdout输出样式的类型 可以为的值如下: #org.apache.log4j.HTMLLayout html形式 #org.apache.log4j.PatternLayout #org.apache.log4j.SimpleLayout 基本信息(日志信息的级别和信息字符串) #org.apache.log4j.TTCCLayout 包含日志产生的时间、线程、类别等信息 log4j.appender.stdout.layout=org.apache.log4j.PatternLayout #指定布局的样式(该属性只有当上面的layout属性为 PatternLayout 时能用) #[QC] 这里随便写 # %p 级别信息 [DEBUG,INFO,WARN,ERROR,FATAL] 这些 #%C.%M(%L) 类名.方法名(行号) #%t 线程名 #%n 换行 #%d 输出当前时间 #%m 代码中的要输出的消息 log4j.appender.stdout.layout.ConversionPattern = [ETDS] %p %d{yyyy-MM-dd HH:mm:ss } %C.%M(%L) %m%n #定义输出端类型为 每天产生一个文件 log4j.appender.R=org.apache.log4j.DailyRollingFileAppender #定义要输出到的文件 log4j.appender.R.File=C:/etds/etds.log #定义布局类型 log4j.appender.R.layout=org.apache.log4j.PatternLayout #自定义布局样式 log4j.appender.R.layout.ConversionPattern=%p %d{yyyy-MM-dd HH:mm:ss } %C.%M(%L) %m%n #将com.zf.log包下面所有类的等级设置为DEBUG #格式为 log4j.logger.yourpackagename=级别 #下面全都是包的等级了 log4j.logger.com.zf.log=DEBUG log4j.logger.com.opensymphony.oscache=ERROR log4j.logger.net.sf.navigator=ERROR log4j.logger.org.apache.commons=ERROR log4j.logger.org.apache.struts=WARN log4j.logger.org.displaytag=ERROR log4j.logger.org.springframework=DEBUG log4j.logger.com.ibatis.db=WARN log4j.logger.org.apache.velocity=FATAL log4j.logger.com.canoo.webtest=WARN log4j.logger.org.hibernate.ps.PreparedStatementCache=WARN log4j.logger.org.hibernate=DEBUG log4j.logger.org.logicalcobwebs=WARN
Hibernate 相关
UUID生成方式
@GenericGenerator(name = "uuid" , strategy="org.hibernate.id.UUIDGenerator") @Id @GeneratedValue(generator = "uuid") @Column(name="id") private String id ;
二级缓存
@Table(name = "plane") @Entity @Cache(usage = CacheConcurrencyStrategy.READ_WRITE)Hibernate 使用二级缓存时要注意的是,如果使用 hql 查询,他会根据hql 以及参数 去做相应的缓存,比如:
第一步 : 执行hql "from Student" 将学生全部查询出来
第二步: 执行 hql "from Student where id = 1" (这样是不会从二级缓存里面去找的)
但是改成 session.get(Student.class , 1); 这样就可以从二级缓存里面找到
JSON相关
首先要现在springmvc.xml中配置好json转换器 ,然后就可以使用json相关的注解和一些方法
@JsonSerialize(using = DateJsonSerelized.class) //表示该字段转换为json时,用DateJsonSerelized类进行转换格式 @Temporal(TemporalType.DATE) @Column(name = "date") private Date date ; @JsonIgnore //表示转换成JSON对象时忽略该字段 @OneToMany(mappedBy = "media" , fetch = FetchType.LAZY ) private List<Plane> planes ;
其中DateJsonSerelized类的定义如下:
package com.zf.common; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; import com.fasterxml.jackson.core.JsonGenerator; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.JsonSerializer; import com.fasterxml.jackson.databind.SerializerProvider; public class DateJsonSerelized extends JsonSerializer<Date>{ private SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd"); @Override public void serialize(Date value, JsonGenerator jgen, SerializerProvider sp) throws IOException, JsonProcessingException { jgen.writeString(sdf.format(value)); } }
之后在SpringMvc Controller方法中,将该对象作为json返回时,就会根据上面的方式操作该对象
Freemarker相关
ftl文件中访问静态属性和方法
/** * Access static Method FTL访问静态方法 和属性(可以将该方法提取出来,让所有的Controller继承,避免每个类中都要写一个该方法) * @param packname * @return * @throws TemplateModelException */ private final static BeansWrapper wrapper = BeansWrapper.getDefaultInstance(); private final static TemplateHashModel staticModels = wrapper.getStaticModels(); protected TemplateHashModel useStaticPacker(Class<?> clazz) { try { return (TemplateHashModel) staticModels.get(clazz.getName()); } catch (TemplateModelException e) { throw new RuntimeException(e); } };然后在需要的controler中调用该方法就可以了
@RequestMapping("freemarker_{userid}") public ModelAndView freemarker(@PathVariable("userid") String userid ){ System.out.println(userid); ModelAndView mav = new ModelAndView("index"); mav.getModelMap().put("username", "zhoufeng"); mav.getModelMap().put("MediaService", useStaticPacker(MediaService.class)); //允许Freemarker访问MediaService类的静态方法 return mav; }
SpringMVC相关
监听器
package com.zf.common; import javax.annotation.Resource; import org.springframework.context.ApplicationEvent; import org.springframework.context.ApplicationListener; import org.springframework.stereotype.Component; import com.zf.service.MediaService; /** * SpringMVC 监听器 在启动容器的时候会随着启动 * @author zhoufeng * */ @Component public class StartUpHandler implements ApplicationListener<ApplicationEvent>{ @Resource(name = "MediaService") private MediaService mediaService ; @Override public void onApplicationEvent(ApplicationEvent event) { //将数据全部查询出来,放入缓存 mediaService.queryAll(); } }
类型转换器
在需要转换类型的controller类中加入如下方法
@Resource(name = "DateEdite") private DateEdite dateEdite ;
/** * 类型转换器 * @param binder */ @InitBinder public void initBinder(WebDataBinder binder){ binder.registerCustomEditor(Date.class, dateEdite); }
package com.zf.common; import java.beans.PropertyEditorSupport; import java.text.ParseException; import java.text.SimpleDateFormat; import org.springframework.stereotype.Component; /** * 自定义Date类型的类型转换器 * @author zhoufeng * */ @Component("DateEdite") public class DateEdite extends PropertyEditorSupport{ private SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); @Override public String getAsText() { return getValue() == null ? "" : sdf.format(getValue()); } @Override public void setAsText(String text) throws IllegalArgumentException { if(text == null || !text.matches("^\\d{4}-\\d{2}-\\d{2}$")) setValue(null); else try { setValue(sdf.parse(text)); } catch (ParseException e) { setValue(null); } } }
接收Map类型参数
@RequestMapping("mapparams") @ResponseBody public String mapParams(MapParam maps){ Set<Map.Entry<Object, Object>> mapsSet = maps.getMaps().entrySet(); for (Entry<Object, Object> entry : mapsSet) { log.info(entry.getKey() + " : " + entry.getValue()); } return "xx"; }
class MapParam{ private Map<Object , Object> maps ; public Map<Object, Object> getMaps() { return maps; } public void setMaps(Map<Object, Object> maps) { this.maps = maps; } }
文件上传(需要commons-io-2.0.1.jar commons-fileupload-1.2.2.jar 包的支持)
先在spring.mvc配置
<!-- 配置文件上传 --> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!-- one of the properties available; the maximum file size in bytes --> <property name="maxUploadSize" value="100000" /> </bean>然后写controller方法
/** * 单文件上传 * @param name * @param file * @return * @throws IOException */ @RequestMapping("fileupload") @ResponseBody public String fildUpload(@RequestParam("name") String name , @RequestParam("file") MultipartFile file) throws IOException{ InputStream is = file.getInputStream(); FileUtils.copyInputStreamToFile(is, new File("c://testupload/" + name)); return "upload success!" ; } /** * 多文件上传 * @param request * @param name * @return * @throws IOException */ @RequestMapping("fileuploads") @ResponseBody public String fileUploads(MultipartHttpServletRequest request , @RequestParam("name") String[] name ) throws IOException{ List<MultipartFile> files = request.getFiles("file"); for (int i = 0; i < files.size(); i++) { FileUtils.copyInputStreamToFile( files.get(i).getInputStream(), new File("c://testupload/" + name[i])); } log.info("upload success!"); return "upload success!" ; }
AOP
如果系统有地方需要用到AOP ,那么可以自定义注解,以及注解处理类,如下:
先定义注解类
/** * Action权限注解 * @author lcy */ @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.METHOD}) public @interface PowerCode { //boolean fordis(); String[] codes(); }
然后定义AOP处理类
@Around("@annotation(com.evertrip.etdspro.misc.annotations.ETAdminPower)") public Object etadmin(ProceedingJoinPoint point) throws Throwable { final BaseController controller = (BaseController) point.getTarget(); if(controller.isAopBreakked) return point.proceed(); final Etuser etuser = controller.obtainCurrentETUser(); if (etuser != null && etuser.getIsadmin()) return point.proceed(); // 放过 controller.setSessionmessageValue("权限不足"); return new ModelAndView("common/alert"); }
相关文章推荐
- freemarker+springMVC+ajaxfileupload实现异步图片上传(单张)
- freemarker+springMVC+ajaxfileupload实现异步图片上传(多张)
- spring mvc ajaxfileupload文件上传返回json下载问题
- Spring mvc+hibernate+freemarker(开源项目)
- springmvc+ajaxfileupload实现头像上传并预览
- Spring mvc+hibernate+freemarker(实战)
- Spring mvc+hibernate+freemarker(开源项目)
- spring mvc upload file exmaple
- 异常-----springmvc + ajaxfileupload解决ajax不能异步上传图片的问题。java.lang.ClassCastException: org.apache.catalina.connector.RequestFacade cannot be cast to org.springframework.web.multipart.
- Spring mvc+hibernate+freemarker(实战)
- 使用springMVC + ajaxfileupload上传文件
- Spring mvc+hibernate+freemarker(实战)
- ajaxfileupload.js+SpringMVC实现文件上传
- springMVC + ajaxfileupload异步上传图片预览,裁剪并保存图片
- Spring mvc + hibernate +freemarker(开源项目)
- Spring mvc+hibernate+freemarker(开源项目)
- Spring MVC+hibernate 使用@ResponseBody返回json报500 (Internal Server Error)
- Spring MVC+Jackson+Hibernate将一个List转化为JSON
- 一个完整的springmvc + ajaxfileupload实现图片异步上传的案例
- Spring MVC Multiple File Upload example