찾다
웹 프론트엔드HTML 튜토리얼前端js和css的压缩合并之wro4j_html/css_WEB-ITnose

me:注:

找了一圈,在java社区和node社区,就没有找到中意的解决方案,总是带有各种各样的毛病。

看来ror社区还是拥有最强悍的设计和构架能力的,也许得益于ruby的精神。

最牛牛的人有时候只要一两个就够了。

下面的文还是不错的,可惜wro4j的愚笨复杂设计啊。

from: 使用wro4j和maven在编译期间压缩js和css文件

最近在对一个web系统做性能优化. 

而对用到的静态资源文件的压缩整合则是前端性能优化中很重要的一环. 

好处不仅在于能够减小请求的文件体积,而且能够减少浏览器的http请求数. 

因为是基于java的web系统,并且使用的是nginx+tomcat做为服务器. 

最后考虑用wro4j和maven plugin在编译期间压缩静态资源. 

优化前: 

基本上所有的jsp都引用了这一大坨静态文件: 

<link rel="stylesheet" type="text/css" href="${ctxPath}/css/skin.css"/>  <link rel="stylesheet" type="text/css" href="${ctxPath}/css/jquery-ui-1.8.23.custom.css"/>  <link rel="stylesheet" type="text/css" href="${ctxPath}/css/validationEngine.jquery.css"/>    <script type="text/javascript">var GV = {ctxPath: '${ctxPath}',imgPath: '${ctxPath}/css'};</script>  <script type="text/javascript" src="${ctxPath}/js/jquery-1.7.2.min.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery-ui-1.8.23.custom.min.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.validationEngine.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.validationEngine-zh_CN.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.fixedtableheader.min.js"></script>  <script type="text/javascript" src="${ctxPath}/js/roll.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.pagination.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.rooFixed.js"></script>  <script type="text/javascript" src="${ctxPath}/js/jquery.ui.datepicker-zh-CN.js"></script>  <script type="text/javascript" src="${ctxPath}/js/json2.js"></script>  <script type="text/javascript" src="${ctxPath}/js/common.js"></script>  

引用的文件很多,并且文件体积没有压缩,导致页面请求的时间非常长. 

另外还有一个问题,就是为了能够充分利用浏览器的缓存,静态资源的文件名称最好能够做到版本化控制. 

这样前端web服务器就可以放心大胆的开启缓存功能而不用担心缓存过期问题,因为如果一旦静态资源文件有修改的话, 

会重新生成一个文件名称. 

下面我根据自己项目的经验,来介绍下如何较好的解决这两个问题. 

分两步进行. 

第一步:引入wro4j,在编译时期将上述分散的多个文件整合成少数几个文件,并且将文件最小化. 

第二步:在生成的静态资源文件的文件名称上加入时间信息 

这是两步优化之后的引用情况: 

${platform:cssFile("/wro/basic") }  <script type="text/javascript">var GV = {ctxPath: '${ctxPath}',imgPath: '${ctxPath}/css'};</script>  ${platform:jsFile("/wro/basic") }  ${platform:jsFile("/wro/custom") }  

只引用了1个css文件,2个js文件.http请求从10几个减少到3个,并且整体文件体积缩小了近一半. 

下面介绍优化流程. 

第一步:合并并且最小化文件. 

1.添加wro4j的maven依赖 

<wro4j.version>1.6.2</wro4j.version>       ...     <dependency>    <groupId>ro.isdc.wro4j</groupId>    <artifactId>wro4j-core</artifactId>    <version>${wro4j.version}</version>    <exclusions>     <exclusion>       <!-- 因为项目中的其他jar包已经引入了不同版本的slf4j,所以这里避免jar重叠所以不引入 -->      <groupId>org.slf4j</groupId>      <artifactId>slf4j-api</artifactId>     </exclusion>    </exclusions>   </dependency>  

2.添加wro4j maven plugin 

   <plugin>      <groupId>ro.isdc.wro4j</groupId>      <artifactId>wro4j-maven-plugin</artifactId>      <version>${wro4j.version}</version>      <executions>       <execution>        <phase>compile</phase>        <goals>         <goal>run</goal>        </goals>       </execution>      </executions>      <configuration>       <targetGroups>basic,custom</targetGroups>        <!-- 这个配置是告诉wro4j在打包静态资源的时候是否需要最小化文件,开发的时候可以设成false,方便调试 -->       <minimize>true</minimize>       <destinationFolder>${basedir}/src/main/webapp/wro/</destinationFolder>       <contextFolder>${basedir}/src/main/webapp/</contextFolder>    <!-- 这个配置是第二步优化需要用到的,暂时忽略 -->       <wroManagerFactory>com.rootrip.platform.common.web.wro.CustomWroManagerFactory</wroManagerFactory>      </configuration>           </plugin>  

如果开发环境是eclipse的话,可以下载m2e-wro4j这个插件. 

下载地址:http://download.jboss.org/jbosstools/updates/m2e-wro4j/ 

这个插件的主要功能是能够帮助我们在开发环境下修改对应的静态文件,或者pom.xml文件的时候能够自动生成打包好的js和css文件. 

对开发来说就会方便很多.只要修改源文件就能看见修改后的结果. 

3.在WEB-INF目录下添加wro.xml文件,这个文件的作用就是告诉wro4j需要以怎样的策略打包jss和css文件. 

<?xml version="1.0" encoding="UTF-8"?>  <groups xmlns="http://www.isdc.ro/wro" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   xsi:schemaLocation="http://www.isdc.ro/wro wro.xsd">     <group name="basic">    <css>/css/basic.css</css>    <css>/css/skin.css</css>    <css>/css/jquery-ui-1.8.23.custom.css</css>    <css>/css/validationEngine.jquery.css</css>        <js>/js/jquery-1.7.2.min.js</js>    <js>/js/jquery-ui-1.8.23.custom.min.js</js>    <js>/js/jquery.validationEngine.js</js>    <js>/js/jquery.fixedtableheader.min.js</js>    <js>/js/roll.js</js>    <js>/js/jquery.pagination.js</js>    <js>/js/jquery.rooFixed.js</js>    <js>/js/jquery.ui.datepicker-zh-CN.js</js>    <js>/js/json2.js</js>   </group>      <group name="custom">    <js>/js/jquery.validationEngine-zh_CN.js</js>    <js>/js/common.js</js>   </group>    </groups>  

官方文档:http://code.google.com/p/wro4j/wiki/WroFileFormat 

其实这个配置文件很好理解,如果不愿看官方文档的朋友我在这简单介绍下. 

上面这样配置的目的就是告诉wro4j要将 

/css/basic.css 

/css/skin.css 

/css/jquery-ui-1.8.23.custom.css 

/css/validationEngine.jquery.css 

这四个文件整合到一起,生成一个叫basic.css的文件到指定目录(wro4j-maven-plugin里配置的),将 

/js/jquery-1.7.2.min.js 

/js/jquery-ui-1.8.23.custom.min.js 

/js/jquery.validationEngine.js 

/js/jquery.fixedtableheader.min.js 

/js/roll.js 

/js/jquery.pagination.js 

/js/jquery.rooFixed.js 

/js/jquery.ui.datepicker-zh-CN.js 

/js/json2.js 

这几个文件整合到一起,生成一个叫basic.js的文件到指定目录. 

最后将 

/js/jquery.validationEngine-zh_CN.js 

/js/common.js 

这两个文件整合到一起,,生成一个叫custom.js的文件到指定目录. 

第一步搞定,这时候如果你的开发环境是eclipse并且安装了插件的话,应该就能在你工程的%your webapp%/wor/目录下看见生成好的 

basic.css,basic.js和custom.js这三个文件了. 

然后你再将你的静态资源引用路径改成 

<link rel="stylesheet" type="text/css" href="${ctxPath}/wro/basic.css"/>  <script type="text/javascript" src="${ctxPath}/wro/basic.js"></script>  <script type="text/javascript" src="${ctxPath}/wro/custom.js"></script>  

就ok了.每次修改被引用到的css或js文件的时候,这些文件都将重新生成. 

如果开发环境是eclipse但是没有安装m2e-wro4j插件的话,pom.xml可能需要额外配置. 

请参考: https://community.jboss.org/en/tools/blog/2012/01/17/css-and-js-minification-using-eclipse-maven-and-wro4j  

第二步:给生成的文件名称中加入时间信息并通过el自定义函数引用脚本文件. 

1. 创建DailyNamingStrategy类 

public class DailyNamingStrategy extends TimestampNamingStrategy {      protected final Logger log = LoggerFactory.getLogger(DailyNamingStrategy.class);     @Override   protected long getTimestamp() {    String dateStr = DateUtil.formatDate(new Date(), "yyyyMMddHH");    return Long.valueOf(dateStr);   }         }  

2.创建CustomWroManagerFactory类 

//这个类就是在wro4j-maven-plugin里配置的wroManagerFactory参数  public class CustomWroManagerFactory extends    DefaultStandaloneContextAwareManagerFactory {   public CustomWroManagerFactory() {    setNamingStrategy(new DailyNamingStrategy());   }  }  

上面这两个类的作用是使用wro4j提供的文件命名策略,这样生成的文件名就会带上时间信息了. 

例如:basic-2013020217.js 

但是现在又会发现一个问题:如果静态资源文件名称不固定的话,那怎么样引用呢? 

这时候就需要通过动态生成<script>与<link>来解决了. </script>

因为项目使用的是jsp页面,所以通过el自定义函数来实现标签生成. 

3.创建PlatformFunction类 

public class PlatformFunction {      private static Logger log = LoggerFactory.getLogger(PlatformFunction.class);         private static ConcurrentMap<String, String> staticFileCache = new ConcurrentHashMap<>();      private static AtomicBoolean initialized = new AtomicBoolean(false);      private static final String WRO_Path = "/wro/";      private static final String JS_SCRIPT = "<script type=\"text/javascript\" src=\"%s\"></script>";   private static final String CSS_SCRIPT = "<link rel=\"stylesheet\" type=\"text/css\" href=\"%s\">";      private static String contextPath = null;       /**   * 该方法根据给出的路径,生成js脚本加载标签   * 例如传入参数/wro/custom,该方法会寻找webapp路径下/wro目录中以custom开头,以js后缀结尾的文件名称名称.   * 然后拼成<script type="text/javascript" src="${ctxPath}/wro/custom-20130201.js"></script>返回   * 如果查找到多个文件,返回根据文件名排序最大的文件   * @param str   * @return   */   public static String jsFile(String filePath) {    String jsFile = staticFileCache.get(buildCacheKey(filePath, "js"));    if(jsFile == null) {     log.error("加载js文件失败,缓存中找不到对应的文件[{}]", filePath);    }    return String.format(JS_SCRIPT, jsFile);   }      /**   * 该方法根据给出的路径,生成css脚本加载标签   * 例如传入参数/wro/custom,该方法会寻找webapp路径下/wro目录中以custom开头,以css后缀结尾的文件名称名称.   * 然后拼成<link rel="stylesheet" type="text/css" href="${ctxPath}/wro/basic-20130201.css">返回   * 如果查找到多个文件,返回根据文件名排序最大的文件   * @param str   * @return   */   public static String cssFile(String filePath) {    String cssFile = staticFileCache.get(buildCacheKey(filePath, "css"));    if(cssFile == null) {     log.error("加载css文件失败,缓存中找不到对应的文件[{}]", filePath);    }    return String.format(CSS_SCRIPT, cssFile);   }      public static void init() throws IOException {    if(initialized.compareAndSet(false, true)) {     ServletContext sc = Platform.getInstance().getServletContext();     if(sc == null) {      throw new PlatformException("查找静态资源的时候的时候发现servlet context 为null");     }     contextPath = Platform.getInstance().getContextPath();     File wroDirectory = new ServletContextResource(sc, WRO_Path).getFile();     if(!wroDirectory.exists() || !wroDirectory.isDirectory()) {      throw new PlatformException("查找静态资源的时候发现对应目录不存在[" + wroDirectory.getAbsolutePath() + "]");     }     //将wro目录下已有文件加入缓存     for(File file : wroDirectory.listFiles()) {      handleNewFile(file);     }     //监控wro目录,如果有文件生成,则判断是否是较新的文件,是的话则把文件名加入缓存     new Thread(new WroFileWatcher(wroDirectory.getAbsolutePath())).start();    }   }     private static void handleNewFile(File file) {    String fileName = file.getName();    Pattern p = Pattern.compile("^(\\w+)\\-\\d+\\.(js|css)$");    Matcher m = p.matcher(fileName);    if(!m.find() || m.groupCount() < 2) return;    String fakeName = m.group(1);    String fileType = m.group(2);    //暂时限定只能匹配/wro/目录下的文件    String key = buildCacheKey(WRO_Path + fakeName, fileType);    if(staticFileCache.putIfAbsent(key, fileName) != null) {     synchronized(staticFileCache) {      String cachedFileName = staticFileCache.get(key);      if(fileName.compareTo(cachedFileName) > 0) {       staticFileCache.put(key, contextPath + WRO_Path + fileName);      }     }    }   }      private static String buildCacheKey(String fakeName, String fileType) {    return fakeName + "-" + fileType;   }      static class WroFileWatcher implements Runnable {        private static Logger log = LoggerFactory.getLogger(WroFileWatcher.class);        private String wroAbsolutePathStr;        public WroFileWatcher(String wroPathStr) {     this.wroAbsolutePathStr = wroPathStr;    }      @Override    public void run() {     Path path = Paths.get(wroAbsolutePathStr);     File wroDirectory = path.toFile();     if(!wroDirectory.exists() || !wroDirectory.isDirectory()) {      String message = "监控wro目录的时候发现对应目录不存在[" + wroAbsolutePathStr + "]";      log.error(message);      throw new PlatformException(message);     }     log.warn("开始监控wro目录[{}]", wroAbsolutePathStr);     try {      WatchService watcher = FileSystems.getDefault().newWatchService();      path.register(watcher, StandardWatchEventKinds.ENTRY_CREATE);            while (true) {       WatchKey key = null;       try {        key = watcher.take();       } catch (InterruptedException e) {        log.error("", e);        continue;       }       for (WatchEvent<?> event : key.pollEvents()) {        if (event.kind() == StandardWatchEventKinds.OVERFLOW) {         continue;        }        WatchEvent<Path> e = (WatchEvent<Path>) event;        Path filePath = e.context();        handleNewFile(filePath.toFile());       }       if (!key.reset()) {        break;       }      }     } catch (IOException e) {      log.error("监控wro目录发生错误", e);     }     log.warn("停止监控wro目录[{}]", wroAbsolutePathStr);    }   }  }  

对应的tld文件就不给出了,根据方法签名编写就行了. 

其中的cssFile和jsFile方法分别实现引用css和js文件. 

在页面使用的时候类似这样: 

${platform:cssFile("/wro/basic") } 

${platform:jsFile("/wro/custom") } 

这个类的主要功能就是使用jdk7的WatchService监控wro目录的新增文件事件, 

一旦有新的文件加到目录里,判断这个文件是不是最新的,如果是的话则使用这个文件名称引用. 

这样一旦有新加的资源文件放到wro目录里,则能够自动被引用,不需要做任何代码上的修改,并且基本不影响性能. 

到此为止功能已经实现. 

但是我考虑到还有两个问题有待完善: 

1.因为生成的文件名称精确到小时,如果这个小时之内有多次代码修改,生成的文件名都完全一样. 

这样就算线上的代码有修改,对于已经有该文本缓存的浏览器来说,不会重新请求文件,也就看不到文件变化. 

不过一般来说线上代码不会如此频繁改动,对于大多数应用来说影响不大. 

2.在开发环境开发一段时间之后,wro目录下会生成一大堆的文件(因为m2e-wro4j插件在生成新的文件的时候不会删除旧文件,如果文件名相同会覆盖掉以前的文件), 

这时候就需要手动删除时间靠前的旧文件,虽然系统会忽略旧文件,但是我相信大多数程序员和我一样是有些许洁癖的吧. 

解决办法还是不少,比如可以写脚本定期清理掉旧文件. 

时间有限,有些地方考虑的不是很完善,欢迎拍砖. 

参考资料: 

http://meri-stuff.blogspot.sk/2012/08/wro4j-page-load-optimization-and-lessjs.html#Configuration  

https://community.jboss.org/en/tools/blog/2012/01/17/css-and-js-minification-using-eclipse-maven-and-wro4j  

http://code.google.com/p/wro4j/wiki/MavenPlugin  

http://code.google.com/p/wro4j/wiki/WroFileFormat  

http://java.dzone.com/articles/using-java-7s-watchservice

+

+

+

-

+

+

+

성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
HTML의 역할 : 웹 컨텐츠 구조HTML의 역할 : 웹 컨텐츠 구조Apr 11, 2025 am 12:12 AM

HTML의 역할은 태그 및 속성을 통해 웹 페이지의 구조와 내용을 정의하는 것입니다. 1. HTML은 읽기 쉽고 이해하기 쉽게하는 태그를 통해 컨텐츠를 구성합니다. 2. 접근성 및 SEO와 같은 시맨틱 태그 등을 사용하십시오. 3. HTML 코드를 최적화하면 웹 페이지로드 속도 및 사용자 경험이 향상 될 수 있습니다.

HTML 및 코드 : 용어를 자세히 살펴 봅니다HTML 및 코드 : 용어를 자세히 살펴 봅니다Apr 10, 2025 am 09:28 AM

"Code"는 "Code"BroadlyIncludeLugageslikeJavaScriptandPyThonforFunctureS (htMlisAspecificTypeofCodeFocudecturecturingWebContent)

HTML, CSS 및 JavaScript : 웹 개발자를위한 필수 도구HTML, CSS 및 JavaScript : 웹 개발자를위한 필수 도구Apr 09, 2025 am 12:12 AM

HTML, CSS 및 JavaScript는 웹 개발의 세 가지 기둥입니다. 1. HTML은 웹 페이지 구조를 정의하고 등과 같은 태그를 사용합니다. 2. CSS는 색상, 글꼴 크기 등과 같은 선택기 및 속성을 사용하여 웹 페이지 스타일을 제어합니다.

HTML, CSS 및 JavaScript의 역할 : 핵심 책임HTML, CSS 및 JavaScript의 역할 : 핵심 책임Apr 08, 2025 pm 07:05 PM

HTML은 웹 구조를 정의하고 CSS는 스타일과 레이아웃을 담당하며 JavaScript는 동적 상호 작용을 제공합니다. 세 사람은 웹 개발에서 의무를 수행하고 화려한 웹 사이트를 공동으로 구축합니다.

HTML은 초보자를 위해 쉽게 배우나요?HTML은 초보자를 위해 쉽게 배우나요?Apr 07, 2025 am 12:11 AM

HTML은 간단하고 배우기 쉽고 결과를 빠르게 볼 수 있기 때문에 초보자에게 적합합니다. 1) HTML의 학습 곡선은 매끄럽고 시작하기 쉽습니다. 2) 기본 태그를 마스터하여 웹 페이지를 만들기 시작하십시오. 3) 유연성이 높고 CSS 및 JavaScript와 함께 사용할 수 있습니다. 4) 풍부한 학습 리소스와 현대 도구는 학습 과정을 지원합니다.

HTML의 시작 태그의 예는 무엇입니까?HTML의 시작 태그의 예는 무엇입니까?Apr 06, 2025 am 12:04 AM

anexampleStartingtaginhtmlis, whithbeginsaparagraph.startingtagsareessentialinhtmlastheyinitiate rements, definetheirtypes, andarecrucialforstructurituringwebpages 및 smanstlingthedom.

메뉴에서 점선 분할 효과의 중심 정렬을 달성하기 위해 CSS의 Flexbox 레이아웃을 사용하는 방법은 무엇입니까?메뉴에서 점선 분할 효과의 중심 정렬을 달성하기 위해 CSS의 Flexbox 레이아웃을 사용하는 방법은 무엇입니까?Apr 05, 2025 pm 01:24 PM

메뉴에서 점선 분할 효과를 설계하는 방법은 무엇입니까? 메뉴를 설계 할 때는 일반적으로 접시 이름과 가격 사이에 왼쪽과 오른쪽을 정렬하는 것이 어렵지 않지만 점선 또는 중간의 점은 어떻습니까?

온라인 코드 편집기는 코드 입력을 구현하는 데 사용하는 HTML 요소는 무엇입니까?온라인 코드 편집기는 코드 입력을 구현하는 데 사용하는 HTML 요소는 무엇입니까?Apr 05, 2025 pm 01:21 PM

웹 코드 편집기의 HTML 요소 분석 많은 온라인 코드 편집기를 사용하면 사용자가 HTML, CSS 및 JavaScript 코드를 입력 할 수 있습니다. 최근에 누군가가 제안했습니다 ...

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
3 몇 주 전By尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

Eclipse용 SAP NetWeaver 서버 어댑터

Eclipse용 SAP NetWeaver 서버 어댑터

Eclipse를 SAP NetWeaver 애플리케이션 서버와 통합합니다.