`
ahuaxuan
  • 浏览: 641665 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

模板:velocity和freemarker的比较

阅读更多
/**
*作者:张荣华(ahuaxuan)
*2007-04-16
*转载请注明出处及作者
*/

模板技术在现代的软件开发中有着重要的地位,而目前最流行的两种模板技术恐怕要算freemarker和velocity了,webwork2.2对两者都有不错的支持,也就是说在webwork2中你可以随意选择使用freemarker或velocity作为view,模板技术作为view的好处是很多,尤其和jsp比较起来优点更大,众所周知jsp需要在第一次被执行的时候编译成servlet,那么这个过程是很慢的,当然很多应用服务器都提供预编译的功能,但是在开发的时候仍然给我们程序员带来了很多痛苦,每次修改都要多几秒钟,那在一天的开发中就有很多时间浪费在jsp的编译上了。用webwork in action的作者的话来说:“每次修改之后重新运行都要等等几秒是令人失望的,而频繁地修改jsp更是会令你的失望情绪变本加厉“。我们把模板技术引入到view中去可以带来更好的开发效率,而且模板的速度要比jsp快(虽然编译过后的jsp在速度上已经满足我的需求了,呵呵)。 当然模板技术可以用在很多领域,可不只在view那里。我们可以通过模板技术来生成xml,生成jsp,生成java文件等等,说到这里,大家通常会使用模板技术用在公司的框架里,这样就可以很快速的生成添删改查的代码,需要的只是模板,其他比如还有邮件模板等等。

以上是模板的作用,当然模板还有其他领域的应用,希望能和大家多讨论,提高我们的生产效率。

那么现在开源的模板技术有好几种,多了之后就有一个选择的问题了,如何选择一个满足自己需要的模板的呢,我大概了看了一下两种模板技术,写了一个例子,我使用了几种设计模式来完成了这个例子,这个例子中,我同时使用了freemarker和velocity,这样同学们可以通过代码很直观的比较两种模板技术,通过这个例子,我认识到freemarker在功能上要比velocity强大
1在view层的时候,它提供了format日期和数字的功能,我想大家都有在页面上format日期或数字的经验,用jsp的同学可能对jstl的fmt标签很有感情,使用了freemarker之后也可以使用freemarker提供的功能来formmat日期和数据,这个功能我想是很贴心的

2通过我的使用我发现freemaker的eclipseplugin要比velocity的eclipseplugin好,如果你是用idea那很遗憾,我没有找到类似的插件。好在很多地方呢,我看到的是freemarker的插件除了支持freemarker语法也支持html语句,而velocity的插件貌似只支持velocity的语法,html就只是用普通的文本来显示了,在这一点上freemarker占上风了(不要和我说高手都是用windows记事本之类的话,这本来就违背了模板技术的初衷)

3freemarker对jsptag的支持很好,算了,不到迫不得已还是不要这样做吧。

还有就是两者的语法格式,这一点上不同的人有不同倾向

下面就介绍一下这个例子吧

/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class TemplateTest {

	/**
	 * @param args
	 */
	public static void main(String[] args) throws Exception{
		/* 准备数据 */
        Map latest = new HashMap();
        latest.put("url", "products/greenmouse.html");
        latest.put("name", "green mouse");
        
        Map root = new HashMap();
        root.put("user", "Big Joe");
        root.put("latestProduct", latest);
        root.put("number", new Long(2222));
        root.put("date",new Date());
        
        List listTest = new ArrayList();
        listTest.add("1");
        listTest.add("2");
        
        root.put("list",listTest);
        
        TemplateEngine freemarkerEngine = (TemplateEngine)TemplateFactory.getInstance().getBean("freemarker");
        freemarkerEngine.run(root);//使用freemarker模板技术
        
        TemplateEngine velocityEngine = (TemplateEngine)TemplateFactory.getInstance().getBean("velocity");
        velocityEngine.run(root);//使用velocity模板技术
	}

}


工厂类,用来得到模板引擎
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class TemplateFactory {
	private static TemplateFactory instance;
	private Map objectMap;
	
	static{
		instance = new TemplateFactory();
	}
	
	public TemplateFactory() {
		super();
		this.objectMap = new HashMap();
		synchronized (this) {
			objectMap.put("freemarker", new FreemarkerTemplateEngine(){
				public String getTemplatePath() {
					return "template";
				}
			});
			
			objectMap.put("velocity", new VelocityTemplateEngine());
		}
	}

	public static TemplateFactory getInstance(){
		return instance;
	}
	
	/**
	 * 模仿spring的工厂
	 * @param beanName
	 * @return
	 */
	public Object getBean(String beanName){
		return objectMap.get(beanName);
	}

}
引擎接口
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public interface TemplateEngine {
	
	void run(Map context)throws Exception;

}
模板引擎的实现使用method template模式,因为有两个实现,这两个实现又存在公共的逻辑,所以选择了这个模式

/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public abstract class AbstractTemplateEngine implements TemplateEngine{

	public abstract String getTemplatePath();
	
	public abstract String getTemplate();
	
	public abstract String getEngineType();
	
	public void run(Map context)throws Exception{
		if(Constants.ENGINE_TYPE_FREEMARKER.equals(getEngineType()))
			executeFreemarker(context);
		else
			executeVelocity(context);
	}
	
	private void executeFreemarker(Map context)throws Exception{
		Configuration cfg = new Configuration();
	    cfg.setDirectoryForTemplateLoading(
	            new File(getTemplatePath()));
	    cfg.setObjectWrapper(new DefaultObjectWrapper());
	    
	    cfg.setCacheStorage(new freemarker.cache.MruCacheStorage(20, 250));
	            
	    Template temp = cfg.getTemplate(getTemplate());

	    Writer out = new OutputStreamWriter(System.out);
	    temp.process(context, out);
	    out.flush();
	}
	
	private void executeVelocity(Map root)throws Exception{
		
		Velocity.init();
		VelocityContext context = new VelocityContext(root);
		org.apache.velocity.Template template = null;
		
	    template = Velocity.getTemplate(getTemplatePath()+getTemplate());
	    
		StringWriter sw = new StringWriter();
		template.merge( context, sw );
		System.out.print(sw.toString());

	}

}



这个是freemarker实现
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class FreemarkerTemplateEngine extends AbstractTemplateEngine{
	private static final String DEFAULT_TEMPLATE = "FreemarkerExample.ftl";
	
	/**
	 * 这个方法应该实现的是读取配置文件
	 */
	public String getTemplatePath() {
		return null;
	}
	
	public void run(Map root) throws Exception{
		super.run(root);
	}

	public String getTemplate() {
		// TODO Auto-generated method stub
		return DEFAULT_TEMPLATE;
	}

	public String getEngineType() {
		return Constants.ENGINE_TYPE_FREEMARKER;
	}
}
这个是velocity实现
/**
 * 
 * @author 张荣华
 * 转载请注明出处
 */
public class VelocityTemplateEngine extends AbstractTemplateEngine{

private static final String DEFAULT_TEMPLATE = "VelocityExample.vm";

	public String getTemplatePath() {
		return "/template/";
	}
	
	public void run(Map map) throws Exception{
		super.run(map);
	}

	public String getTemplate() {
		// TODO Auto-generated method stub
		return DEFAULT_TEMPLATE;
	}

	public String getEngineType() {
		// TODO Auto-generated method stub
		return Constants.ENGINE_TYPE_VELOCITY;
	}
}


以下是模板
1,freemarker模板
freemarker template test:
string test-----------${user}-----------${number}-----------${latestProduct.url}-----------${latestProduct.name}
condition test-----------
<#if user == "Big Joe">
list iterator-----------
<#list list as aa>
${aa}
</#list> 
</#if>
date test-----------${date?string("MMM/dd/yyyy")}


2,velocity模板

******************************************************************************************************************
velocity template test:
string test-----------${user}-----------${number}-----------${latestProduct.url}-----------${latestProduct.name}
condition test-----------
#if ($user == "Big Joe")
list iterator-----------
#foreach( $aa in $list )
$aa
#end
#end
date test-----------${date}


至此整个例子就结束了,以上只是最简单的介绍,当然这两种技术还有待我们的深入研究。这个例子只不过是比较直观的表现两种技术的使用而已
而且如果想学习方法模板模式和工厂模式的同学可以下载代码看看

作者:张荣华,未经作者同意不得随意转载!
  • template.rar (2.1 MB)
  • 描述: 可运行代码
  • 下载次数: 3275
分享到:
评论
41 楼 weiqingfei 2007-05-18  
ray_linn 写道
weiqingfei 写道

jsp只编译一次,以后就都是servlet的事情了。

你的意思是说,模板比直接的servlet输出还要快?????


FreeMarker作者宣称Freemarker是达到html级别。


怎么可能,原理上说不通嘛。
我虽然没有用过freemarker,但是我也知道它是servlet结合模板来生成html的,
这个过程应该是这样的
1.servlet加载模板(也许是常驻内存)
2.动态解析模板,并替换插入数据
3.写入输出流

而jsp编译成的servlet,代码就内置了所有的html字符串,只是把数据都顺序的写入输出流。

至于业务上的时间花费,双方都一样。

死活都想不通,多了一个动态解析的模板,怎么可能比直接使用servlet要快。
40 楼 Qieqie 2007-05-18  
快成混水了

这里的快慢不管谁优秀一点在整个系统中只是个零头
我们还是从软件工程表达能力等角度上来比较这些技术
让我们更好认识 对我们、对产品可能更有意义
39 楼 ray_linn 2007-05-18  
weiqingfei 写道
ray_linn 写道
weiqingfei 写道
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?


如果你带领一个团队,你如何保证每个人一定不写业务逻辑在jsp pages里?


详细试样书里不让写,程序员就不能写。
在这方面,菜鸟比高手听话多了。



为此付出的规范,检查,如果你认为值得的话。
38 楼 ray_linn 2007-05-18  
weiqingfei 写道

jsp只编译一次,以后就都是servlet的事情了。

你的意思是说,模板比直接的servlet输出还要快?????


FreeMarker作者宣称Freemarker是达到html级别。
37 楼 weiqingfei 2007-05-18  
ray_linn 写道
weiqingfei 写道
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?


如果你带领一个团队,你如何保证每个人一定不写业务逻辑在jsp pages里?


详细试样书里不让写,程序员就不能写。
在这方面,菜鸟比高手听话多了。
36 楼 weiqingfei 2007-05-18  
ahuaxuan 写道
weiqingfei 写道
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?

和jsp相比,velocity这样的模板语言有着类似的功能,脚本能力则有所欠缺-对于大多数开发者来说,也许脚本能力的欠缺反而是好事。如果选择jsp,则应该多考虑jstl和jsp表达式,避免使用杂乱的jsp脚本。

模板语言尽管它们是解释执行的,但常常比jsp更快。所以jsp并没有特别的性能优势。(实际上情况很可能正相反:模板技术的性能可能更高)

使用jstl和jsp表达式对大多数开发者来说应该是家常便饭,但是同样的功能,比如说c:foreach标签就比#foreach in来得更繁琐,而且每次修改jsp都需要编译,大得页面一编译就要好几秒钟,一个项目累计下来花在编译上就有很多时间了


jsp只编译一次,以后就都是servlet的事情了。

你的意思是说,模板比直接的servlet输出还要快?????
35 楼 ray_linn 2007-05-18  
sg552 写道

——“JSP最大毛病就是能把表现逻辑和商业逻辑混在一起”。我想说,高手写的JSP,跟菜鸟写的JSP是不一样的。高手用JSP写的东西,比菜鸟用STRUTS写的都M-V-C分明。JIVE论坛,就是JSP写的。你能说它把表现逻辑和商业逻辑都混在一起吗?

—— “这种糊涂蛋跑来说,表现逻辑商业逻辑混在一起是长处。” 谁是糊涂蛋?那句话说“表现逻辑商业逻辑混在一起是长处”?  我真的没找到。这句话是你自己说的吧?


满大街是高手多,还是菜鸟多?高手和菜鸟一起工作的时候,你如何去统一这个团队?
34 楼 ray_linn 2007-05-18  
weiqingfei 写道
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?


如果你带领一个团队,你如何保证每个人一定不写业务逻辑在jsp pages里?
33 楼 ahuaxuan 2007-05-18  
weiqingfei 写道
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?

和jsp相比,velocity这样的模板语言有着类似的功能,脚本能力则有所欠缺-对于大多数开发者来说,也许脚本能力的欠缺反而是好事。如果选择jsp,则应该多考虑jstl和jsp表达式,避免使用杂乱的jsp脚本。

模板语言尽管它们是解释执行的,但常常比jsp更快。所以jsp并没有特别的性能优势。(实际上情况很可能正相反:模板技术的性能可能更高)

使用jstl和jsp表达式对大多数开发者来说应该是家常便饭,但是同样的功能,比如说c:foreach标签就比#foreach in来得更繁琐,而且每次修改jsp都需要编译,大得页面一编译就要好几秒钟,一个项目累计下来花在编译上就有很多时间了
32 楼 ddandyy 2007-05-18  
-_-

连JSP都成为模板了
31 楼 myy 2007-05-18  
支持 sg552 ,要用模板,jsp 就是最好模板。
30 楼 weiqingfei 2007-05-18  
1.没有人规定一定要在jsp里面写业务逻辑吧
2.刚才是谁说的,模板比jsp运行的快?
29 楼 sg552 2007-05-18  
你发帖子很有意思。

面每句话都是你的观点,却都说是对方的。

要说不会看字,那是你。
——“商业逻辑”是ahuaxuan说的。
——“你要是连这都看不懂,谈什么写程序。”这句话才是你说的。  如果你们俩是一个,那可以。

——“你的逻辑却认为C++因为可以手工申请和释放内存,所以比java强。” 我上面的发言,哪里有这样的逻辑了?发言里我说的是:请拿数据说话,并且举了一个JAVA 与C 语言的数据例子,结果是JAVA 某些运算比C/C++ 还要快。

——“JSP最大毛病就是能把表现逻辑和商业逻辑混在一起”。我想说,高手写的JSP,跟菜鸟写的JSP是不一样的。高手用JSP写的东西,比菜鸟用STRUTS写的都M-V-C分明。JIVE论坛,就是JSP写的。你能说它把表现逻辑和商业逻辑都混在一起吗?

—— “这种糊涂蛋跑来说,表现逻辑商业逻辑混在一起是长处。” 谁是糊涂蛋?那句话说“表现逻辑商业逻辑混在一起是长处”?  我真的没找到。这句话是你自己说的吧?
28 楼 ray_linn 2007-05-18  
sg552 写道
根据楼上某楼说,在JSP中不写代码,不写逻辑,谁能做的到?
中国这么多项目,谁可以拿出视图层中没有逻辑的项目???

如果模板要求使用标准的MVC,那么VELOCITY和FREEMAKER中的MACRO是做什么的???

如果事情是想象的那么简单,只是模板功能,那么JSP早就被入门程序员都会的System.out.println 取代了,还用的着模板出马?

楼上的比喻
"用C++我们可以代码写得这么复杂,我们可以申请内存,释放内存,用java你可以么~" 似乎不太恰当。 JAVA是离不开指针和内存的。一个 String str="123" 就涉及到上句内容的全部了。


你连字儿不会看。我的文字里清楚写着“商业逻辑”,你要是连这都看不懂,谈什么写程序。

C++必须手工申请和释放内存,而java为了改进这个,才使用了GC,你的逻辑却认为C++因为可以手工申请和释放内存,所以比java强。


同样道理,JSP最大毛病就是能把表现逻辑和商业逻辑混在一起,导致代码难以修改,难以分工,Template就是为了彻底剥离二者而设计的。却有这种糊涂蛋跑来说,表现逻辑商业逻辑混在一起是长处。

是本质性的错误。
27 楼 sg552 2007-05-18  
如果把逻辑划分为业务逻辑和表现逻辑,那么我觉得是美工的问题。  MACRO写起来跟一个方法一样,有参量有对象有返回值,能不用的时候就不用。

总之,我的观点是,模板就用在模板的地方,想取代JSP,太难。
26 楼 jarit 2007-05-18  
macro里放css/js之类公用的头,不知怎么有什么IDE能自动包含进来。。。不然太不方便了
25 楼 ahuaxuan 2007-05-18  
sg552 写道
根据楼上某楼说,在JSP中不写代码,不写逻辑,谁能做的到?
中国这么多项目,谁可以拿出视图层中没有逻辑的项目???

如果模板要求使用标准的MVC,那么VELOCITY和FREEMAKER中的MACRO是做什么的???

如果事情是想象的那么简单,只是模板功能,那么JSP早就被入门程序员都会的System.out.println 取代了,还用的着模板出马?

楼上的比喻
"用C++我们可以代码写得这么复杂,我们可以申请内存,释放内存,用java你可以么~" 似乎不太恰当。 JAVA是离不开指针和内存的。一个 String str="123" 就涉及到上句内容的全部了。

我们说的逻辑是指业务逻辑,不是表现逻辑,页面没有业务逻辑的系统多得是,举个例子,银行利息计算就属于业务逻辑,而计算得结果要在页面展现出来就不是业务逻辑。
macro得功能和tag得功能是一样得,只不过是用来组织数据得展现
24 楼 ddandyy 2007-05-18  
上一个帖子还在说随意的JAVA代码  现在就开始模糊到逻辑了
如果是逻辑  那JSP 还有其他模板  都有自己的if之类的写法的  这些和JAVA代码无关吧
不写JAVA代码太容易做到了
23 楼 sg552 2007-05-18  
根据楼上某楼说,在JSP中不写代码,不写逻辑,谁能做的到?
中国这么多项目,谁可以拿出视图层中没有逻辑的项目???

如果模板要求使用标准的MVC,那么VELOCITY和FREEMAKER中的MACRO是做什么的???

如果事情是想象的那么简单,只是模板功能,那么JSP早就被入门程序员都会的System.out.println 取代了,还用的着模板出马?

楼上的比喻
"用C++我们可以代码写得这么复杂,我们可以申请内存,释放内存,用java你可以么~" 似乎不太恰当。 JAVA是离不开指针和内存的。一个 String str="123" 就涉及到上句内容的全部了。
22 楼 ray_linn 2007-05-18  
模板的目的是要弱化jsp asp的功能,要剥离可以写入商业逻辑的机会...不是强化...晕倒.

这发言就好像说"用C++我们可以代码写得这么复杂,我们可以申请内存,释放内存,用java你可以么~"

相关推荐

    velocity和freemarker的比较

    以上是对Velocity和FreeMarker的基本介绍和比较,它们都是Java Web开发中的重要工具,理解它们的特性和应用场景有助于选择最适合项目的技术栈。对于标签“源码”和“工具”,我们可以进一步研究这两个模板引擎的源...

    Velocity 和 FreeMarker区别

    在众多模板引擎中,Velocity和FreeMarker是两种非常受欢迎的选择。下面我们将从多个方面对这两种模板引擎进行详细的比较。 #### 一、简介 - **Velocity**:是一个基于Java的模板引擎,它允许开发者使用简单的模板...

    jsp、freemarker、velocity简介和对比

    本文将重点介绍三种常用的Java Web模板引擎:JSP(Java Server Pages)、Freemarker以及Velocity,并对它们进行深入比较。 #### JSP (Java Server Pages) JSP是一种基于Java技术的服务器端动态网页技术,通过在...

    当前流行的模板引擎效率分析(velocity,freeMarker,Smarty4j,httl)

    本篇文章将对四个流行的Java模板引擎——Velocity、FreeMarker、Smarty4j以及HTTL进行效率分析,旨在探讨它们在处理业务逻辑编译和性能方面的优劣。 1. Velocity: Velocity是Apache软件基金会的一个开源项目,以其...

    freemarker&velocity的使用

    Freemarker和Velocity是两种广泛使用的模板引擎,它们在Java Web开发中扮演着重要的角色,主要用于生成动态HTML或其他格式的文本。这两者都是基于MVC(Model-View-Controller)设计模式,允许开发者将业务逻辑与展示...

    velocity document

    虽然Velocity和Freemarker都是视图模板引擎,但它们有各自的特性和优缺点: 1. **语法简洁性**:Freemarker的语法可能更为简洁,如其使用`&lt;#if&gt;`、`&lt;#foreach&gt;`等,而Velocity则使用`#if`、`#foreach`。 2. **灵活...

    Velocity介紹

    Velocity、Freemarker和Thymeleaf都是Java领域的模板引擎,它们各有特点。Freemarker语法更为复杂,但功能更强大;Thymeleaf强调的是XML友好的语法,更适合静态页面的生成。Velocity则以其简洁和易用性受到许多...

    velocity模板引擎

    - **include与parse**:为了实现页面布局的模块化,Velocity提供了`#include`和`#parse`两种方式来引入其他模板文件。`#include`主要用于简单地插入另一个文件的内容而不执行其中的Velocity语法;`#parse`则不仅会...

    Freemarker和Velocity的eclipse插件

    例如,Freemarker插件可以帮助快速定位模板中的语法错误,而Velocity插件则可以提供模板变量和方法的智能提示,减少出错的可能性。 总的来说,Freemarker和Velocity的Eclipse插件对于Java Web开发者来说是必不可少...

    大型商城网站springmvc+freemarker+velocity+ibatis

    标题 "大型商城网站springmvc+freemarker+velocity+ibatis" 暗示了这是一个基于SpringMVC、Freemarker、Velocity和iBatis框架构建的电子商务平台。这个项目可能是一个B2C(Business-to-Consumer)类型的商城,允许...

    velocity freemarke 模版 静态化 实现

    Velocity和FreeMarker是两种常用的Java模板引擎,它们可以方便地结合JSP(JavaServer Pages)进行动态网页生成,并实现页面静态化。本文将详细介绍如何使用Velocity和FreeMarker模板实现页面静态化,并提供具体的...

    Velocity用户手册(中文)

    4. **Freemarker**: 虽然Velocity和Freemarker都是模板引擎,但它们各有特点,可以根据项目需求选择合适的工具。 总之,Velocity作为一款强大的模板引擎,对于Web开发中的前后端分离有着重要作用,它的简单性和灵活...

    基于Freemarker模板的代码生成器后台代码(controller,service,dao)模板文件

    这个压缩包提供的就是一套基于Freemarker模板的代码生成器模板文件,分别对应控制器(Controller)、实体类(Bean)、服务接口(Service)和服务实现(ServiceImpl)、数据访问对象(Dao)。 1. **Controller.ftl**...

    SpringMVC模板 

    在这个话题中,我们将深入探讨SpringMVC与两种常用的模板引擎——Velocity和FreeMarker的集成与应用。 首先,让我们了解SpringMVC的基本架构。在SpringMVC中,DispatcherServlet是入口点,它负责接收HTTP请求并根据...

    velocity的所有jar包

    Velocity是一个开源的Java模板引擎,它允许开发者将HTML与Java代码分离,使Web开发者...对于新项目,考虑使用更新的版本(如Velocity 2.x)或者其他的模板引擎,如FreeMarker或Thymeleaf,可能会带来更多的优势和功能。

    Velocity--java的模板引擎

    Velocity与JSP、FreeMarker等其他模板引擎相比,有其独特优势。Velocity的语法简洁,更接近自然语言,且由于它不支持脚本,避免了在模板中引入过多的业务逻辑,使得模板更加纯粹。此外,Velocity的性能也相对较高,...

    FreeMarker概述2008

    综上所述,FreeMarker作为一个强大的模板引擎,提供了丰富的模板语言和灵活的数据模型,是Web开发中的有力工具。尽管Velocity等其他工具也有其优点,但FreeMarker的广泛特性和强大功能使其在许多场景下成为更优选择...

    Freemarker和Velocity介绍最新版中文2.29MB最新版本

    FreeMarker是一个用Java语言编写的模板引擎,它基于模板来生成文本输出,仅是利用模板加上数据生成文本页面,即是由Java程序准备要显示的数据,由FreeMarker生成页面,通过模板显示准备的数据,并可以在模板中使用...

Global site tag (gtag.js) - Google Analytics