`
cuker919
  • 浏览: 97676 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

查看TOMCAT内存使用情况(总结)

 
阅读更多

您可以用把下列代码放在一个JSP文件中,如写入memory.jsp,放到你的TOMCAT下的任何一应用中,就可以看到你的TOMCAT总大可使用多少内存,已经使用了多少.

<%--
Document : memory
Created on : 2009-4-9, 1:35:17
Author : Administrator
--%>

<%@page contentType="text/html" pageEncoding="UTF-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
"http://www.w3.org/TR/html4/loose.dtd">

<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>JVM memory</title>
</head>
<body>
<%
double total = (Runtime.getRuntime().totalMemory()) / (1024.0 * 1024);
double max = (Runtime.getRuntime().maxMemory()) / (1024.0 * 1024);
double free = (Runtime.getRuntime().freeMemory()) / (1024.0 * 1024);
out.println("Java 虚拟机试图使用的最大内存量(当前JVM的最大可用内存)maxMemory(): " + max + "MB<br/>");
out.println("Java 虚拟机中的内存总量(当前JVM占用的内存总数)totalMemory(): " + total + "MB<br/>");
out.println("Java 虚拟机中的空闲内存量(当前JVM空闲内存)freeMemory(): " + free + "MB<br/>");
out.println("因为JVM只有在需要内存时才占用物理内存使用,所以freeMemory()的值一般情况下都很小,<br/>" +
"而JVM实际可用内存并不等于freeMemory(),而应该等于 maxMemory()-totalMemory()+freeMemory()。<br/>");
out.println("JVM实际可用内存: " + (max - total + free) + "MB<br/>");
out.println("jspcn");
%>
</body>

</html>


上面的例子基本可以满足性能调试使用,下面的比较详细,需要开发代码,暂时使用上面简单例子。


---------------------------------------------------------

同样也可以进入tomcat管理页面,查看内存使用情况,但是平常在生产环境都禁掉了管理员登录页面。

http://zhumeng8337797.blog.163.com/blog/static/10076891420129223928915/


1. Tomcat6中没有设置任何默认用户,因而需要手动往Tomcat6conf文件夹下的tomcat-users.xml文件中添加用户。

如:<role rolename="manager"/>
<user username="tomcat" password="tomcat" roles="manager"/>

注:添加完需要重启Tomcat6

2. 访问http://localhost:8080/manager/status,输入上面添加的用户名和密码。

3. 然后在如下面的JVM下可以看到内存的使用情况。
JVM:Free memory: 2.50 MB Total memory: 15.53 MB Max memory: 63.56 MB

Free memory:当前可用的内存;

Total memory:当前已经分配的JVM内存;

Max memory:当前允许分配的最大JVM内存;

上周的题目有些简单,但是tomcat也需要我们研究一下,因为涉及的知识不多,但是将来这些都是部署时候非常有用的技术。



-------------------------------------------------------

java不用jni,也可以获得当前系统性能信息

http://kakaluyi.iteye.com/blog/211492

最近做个项目,就是要取得cpu占有率等等的系统信息,一开始以为要用动态链接库了,但后来发现可以像下面这样做,不去调用jni,这样省去了很多看新技术的时间o(∩_∩)o...

在Java中,可以获得总的物理内存、剩余的物理内存、已使用的物理内存等信息,下面例子可以取得这些信息,并且获得在Windows下的内存使用率。
首先编写一个MonitorInfoBean类,用来装载监控的一些信息,包括物理内存、剩余的物理内存、已使用的物理内存、内存使用率等字段,该类的代码如下:

Java代码 收藏代码
  1. packagecom.amgkaka.performance;
  2. /***//**
  3. *监视信息的JavaBean类.
  4. *@authoramg
  5. *@version1.0
  6. *Creationdate:2008-4-25-上午10:37:00
  7. */
  8. publicclassMonitorInfoBean{
  9. /***//**可使用内存.*/
  10. privatelongtotalMemory;
  11. /***//**剩余内存.*/
  12. privatelongfreeMemory;
  13. /***//**最大可使用内存.*/
  14. privatelongmaxMemory;
  15. /***//**操作系统.*/
  16. privateStringosName;
  17. /***//**总的物理内存.*/
  18. privatelongtotalMemorySize;
  19. /***//**剩余的物理内存.*/
  20. privatelongfreePhysicalMemorySize;
  21. /***//**已使用的物理内存.*/
  22. privatelongusedMemory;
  23. /***//**线程总数.*/
  24. privateinttotalThread;
  25. /***//**cpu使用率.*/
  26. privatedoublecpuRatio;
  27. publiclonggetFreeMemory(){
  28. returnfreeMemory;
  29. }
  30. publicvoidsetFreeMemory(longfreeMemory){
  31. this.freeMemory=freeMemory;
  32. }
  33. publiclonggetFreePhysicalMemorySize(){
  34. returnfreePhysicalMemorySize;
  35. }
  36. publicvoidsetFreePhysicalMemorySize(longfreePhysicalMemorySize){
  37. this.freePhysicalMemorySize=freePhysicalMemorySize;
  38. }
  39. publiclonggetMaxMemory(){
  40. returnmaxMemory;
  41. }
  42. publicvoidsetMaxMemory(longmaxMemory){
  43. this.maxMemory=maxMemory;
  44. }
  45. publicStringgetOsName(){
  46. returnosName;
  47. }
  48. publicvoidsetOsName(StringosName){
  49. this.osName=osName;
  50. }
  51. publiclonggetTotalMemory(){
  52. returntotalMemory;
  53. }
  54. publicvoidsetTotalMemory(longtotalMemory){
  55. this.totalMemory=totalMemory;
  56. }
  57. publiclonggetTotalMemorySize(){
  58. returntotalMemorySize;
  59. }
  60. publicvoidsetTotalMemorySize(longtotalMemorySize){
  61. this.totalMemorySize=totalMemorySize;
  62. }
  63. publicintgetTotalThread(){
  64. returntotalThread;
  65. }
  66. publicvoidsetTotalThread(inttotalThread){
  67. this.totalThread=totalThread;
  68. }
  69. publiclonggetUsedMemory(){
  70. returnusedMemory;
  71. }
  72. publicvoidsetUsedMemory(longusedMemory){
  73. this.usedMemory=usedMemory;
  74. }
  75. publicdoublegetCpuRatio(){
  76. returncpuRatio;
  77. }
  78. publicvoidsetCpuRatio(doublecpuRatio){
  79. this.cpuRatio=cpuRatio;
  80. }
  81. }

接着编写一个获得当前的监控信息的接口,该类的代码如下所示:

Java代码 收藏代码
  1. packagecom.amgkaka.performance;
  2. /***//**
  3. *获取系统信息的业务逻辑类接口.
  4. *@authoramg*@version1.0
  5. *Creationdate:2008-3-11-上午10:06:06
  6. */
  7. publicinterfaceIMonitorService{
  8. /***//**
  9. *获得当前的监控对象.
  10. *@return返回构造好的监控对象
  11. *@throwsException
  12. *@authoramgkaka
  13. *Creationdate:2008-4-25-上午10:45:08
  14. */
  15. publicMonitorInfoBeangetMonitorInfoBean()throwsException;
  16. }

该类的实现类MonitorServiceImpl如下所示:

Java代码 收藏代码
  1. packagecom.amgkaka.performance;
  2. importjava.io.InputStreamReader;
  3. importjava.io.LineNumberReader;
  4. importsun.management.ManagementFactory;
  5. importcom.sun.management.OperatingSystemMXBean;
  6. /***//**
  7. *获取系统信息的业务逻辑实现类.
  8. *@authoramg*@version1.0Creationdate:2008-3-11-上午10:06:06
  9. */
  10. publicclassMonitorServiceImplimplementsIMonitorService{
  11. //可以设置长些,防止读到运行此次系统检查时的cpu占用率,就不准了
  12. privatestaticfinalintCPUTIME=5000;
  13. privatestaticfinalintPERCENT=100;
  14. privatestaticfinalintFAULTLENGTH=10;
  15. /***//**
  16. *获得当前的监控对象.
  17. *@return返回构造好的监控对象
  18. *@throwsException
  19. *@authoramg*Creationdate:2008-4-25-上午10:45:08
  20. */
  21. publicMonitorInfoBeangetMonitorInfoBean()throwsException{
  22. intkb=1024;
  23. //可使用内存
  24. longtotalMemory=Runtime.getRuntime().totalMemory()/kb;
  25. //剩余内存
  26. longfreeMemory=Runtime.getRuntime().freeMemory()/kb;
  27. //最大可使用内存
  28. longmaxMemory=Runtime.getRuntime().maxMemory()/kb;
  29. OperatingSystemMXBeanosmxb=(OperatingSystemMXBean)ManagementFactory
  30. .getOperatingSystemMXBean();
  31. //操作系统
  32. StringosName=System.getProperty("os.name");
  33. //总的物理内存
  34. longtotalMemorySize=osmxb.getTotalPhysicalMemorySize()/kb;
  35. //剩余的物理内存
  36. longfreePhysicalMemorySize=osmxb.getFreePhysicalMemorySize()/kb;
  37. //已使用的物理内存
  38. longusedMemory=(osmxb.getTotalPhysicalMemorySize()-osmxb
  39. .getFreePhysicalMemorySize())
  40. /kb;
  41. //获得线程总数
  42. ThreadGroupparentThread;
  43. for(parentThread=Thread.currentThread().getThreadGroup();parentThread
  44. .getParent()!=null;parentThread=parentThread.getParent())
  45. ;
  46. inttotalThread=parentThread.activeCount();
  47. doublecpuRatio=0;
  48. if(osName.toLowerCase().startsWith("windows")){
  49. cpuRatio=this.getCpuRatioForWindows();
  50. }
  51. //构造返回对象
  52. MonitorInfoBeaninfoBean=newMonitorInfoBean();
  53. infoBean.setFreeMemory(freeMemory);
  54. infoBean.setFreePhysicalMemorySize(freePhysicalMemorySize);
  55. infoBean.setMaxMemory(maxMemory);
  56. infoBean.setOsName(osName);
  57. infoBean.setTotalMemory(totalMemory);
  58. infoBean.setTotalMemorySize(totalMemorySize);
  59. infoBean.setTotalThread(totalThread);
  60. infoBean.setUsedMemory(usedMemory);
  61. infoBean.setCpuRatio(cpuRatio);
  62. returninfoBean;
  63. }
  64. /***//**
  65. *获得CPU使用率.
  66. *@return返回cpu使用率
  67. *@authoramg*Creationdate:2008-4-25-下午06:05:11
  68. */
  69. privatedoublegetCpuRatioForWindows(){
  70. try{
  71. StringprocCmd=System.getenv("windir")
  72. +"\\system32\\wbem\\wmic.exeprocessgetCaption,CommandLine,"
  73. +"KernelModeTime,ReadOperationCount,ThreadCount,UserModeTime,WriteOperationCount";
  74. //取进程信息
  75. long[]c0=readCpu(Runtime.getRuntime().exec(procCmd));
  76. Thread.sleep(CPUTIME);
  77. long[]c1=readCpu(Runtime.getRuntime().exec(procCmd));
  78. if(c0!=null&&c1!=null){
  79. longidletime=c1[0]-c0[0];
  80. longbusytime=c1[1]-c0[1];
  81. returnDouble.valueOf(
  82. PERCENT*(busytime)/(busytime+idletime))
  83. .doubleValue();
  84. }else{
  85. return0.0;
  86. }
  87. }catch(Exceptionex){
  88. ex.printStackTrace();
  89. return0.0;
  90. }
  91. }
  92. /***//**
  93. *读取CPU信息.
  94. *@paramproc
  95. *@return
  96. *@authoramg*Creationdate:2008-4-25-下午06:10:14
  97. */
  98. privatelong[]readCpu(finalProcessproc){
  99. long[]retn=newlong[2];
  100. try{
  101. proc.getOutputStream().close();
  102. InputStreamReaderir=newInputStreamReader(proc.getInputStream());
  103. LineNumberReaderinput=newLineNumberReader(ir);
  104. Stringline=input.readLine();
  105. if(line==null||line.length()<FAULTLENGTH){
  106. returnnull;
  107. }
  108. intcapidx=line.indexOf("Caption");
  109. intcmdidx=line.indexOf("CommandLine");
  110. introcidx=line.indexOf("ReadOperationCount");
  111. intumtidx=line.indexOf("UserModeTime");
  112. intkmtidx=line.indexOf("KernelModeTime");
  113. intwocidx=line.indexOf("WriteOperationCount");
  114. longidletime=0;
  115. longkneltime=0;
  116. longusertime=0;
  117. while((line=input.readLine())!=null){
  118. if(line.length()<wocidx){
  119. continue;
  120. }
  121. //字段出现顺序:Caption,CommandLine,KernelModeTime,ReadOperationCount,
  122. //ThreadCount,UserModeTime,WriteOperation
  123. Stringcaption=Bytes.substring(line,capidx,cmdidx-1)
  124. .trim();
  125. Stringcmd=Bytes.substring(line,cmdidx,kmtidx-1).trim();
  126. if(cmd.indexOf("wmic.exe")>=0){
  127. continue;
  128. }
  129. //log.info("line="+line);
  130. if(caption.equals("SystemIdleProcess")
  131. ||caption.equals("System")){
  132. idletime+=Long.valueOf(
  133. Bytes.substring(line,kmtidx,rocidx-1).trim())
  134. .longValue();
  135. idletime+=Long.valueOf(
  136. Bytes.substring(line,umtidx,wocidx-1).trim())
  137. .longValue();
  138. continue;
  139. }
  140. kneltime+=Long.valueOf(
  141. Bytes.substring(line,kmtidx,rocidx-1).trim())
  142. .longValue();
  143. usertime+=Long.valueOf(
  144. Bytes.substring(line,umtidx,wocidx-1).trim())
  145. .longValue();
  146. }
  147. retn[0]=idletime;
  148. retn[1]=kneltime+usertime;
  149. returnretn;
  150. }catch(Exceptionex){
  151. ex.printStackTrace();
  152. }finally{
  153. try{
  154. proc.getInputStream().close();
  155. }catch(Exceptione){
  156. e.printStackTrace();
  157. }
  158. }
  159. returnnull;
  160. }
  161. /***//**
  162. *测试方法.
  163. *@paramargs
  164. *@throwsException
  165. *@authoramg*Creationdate:2008-4-30-下午04:47:29
  166. */
  167. publicstaticvoidmain(String[]args)throwsException{
  168. IMonitorServiceservice=newMonitorServiceImpl();
  169. MonitorInfoBeanmonitorInfo=service.getMonitorInfoBean();
  170. System.out.println("cpu占有率="+monitorInfo.getCpuRatio());
  171. System.out.println("可使用内存="+monitorInfo.getTotalMemory());
  172. System.out.println("剩余内存="+monitorInfo.getFreeMemory());
  173. System.out.println("最大可使用内存="+monitorInfo.getMaxMemory());
  174. System.out.println("操作系统="+monitorInfo.getOsName());
  175. System.out.println("总的物理内存="+monitorInfo.getTotalMemorySize()+"kb");
  176. System.out.println("剩余的物理内存="+monitorInfo.getFreeMemory()+"kb");
  177. System.out.println("已使用的物理内存="+monitorInfo.getUsedMemory()+"kb");
  178. System.out.println("线程总数="+monitorInfo.getTotalThread()+"kb");
  179. }
  180. }

该实现类中需要用到一个自己编写byte的工具类,该类的代码如下所示:

Java代码 收藏代码
  1. packagecom.amgkaka.performance;
  2. /***//**
  3. *byte操作类.
  4. *@authoramg*@version1.0
  5. *Creationdate:2008-4-30-下午04:57:23
  6. */
  7. publicclassBytes{
  8. /***//**
  9. *由于String.subString对汉字处理存在问题(把一个汉字视为一个字节),因此在
  10. *包含汉字的字符串时存在隐患,现调整如下:
  11. *@paramsrc要截取的字符串
  12. *@paramstart_idx开始坐标(包括该坐标)
  13. *@paramend_idx截止坐标(包括该坐标)
  14. *@return
  15. */
  16. publicstaticStringsubstring(Stringsrc,intstart_idx,intend_idx){
  17. byte[]b=src.getBytes();
  18. Stringtgt="";
  19. for(inti=start_idx;i<=end_idx;i++){
  20. tgt+=(char)b[i];
  21. }
  22. returntgt;
  23. }
  24. }

运行下MonitorBeanImpl类,读者将会看到当前的内存、cpu利用率等信息。


分享到:
评论

相关推荐

    实验室管理系统 微信小程序+SSM毕业设计 源码+数据库+论文+启动教程.zip

    实验室管理系统 微信小程序+SSM毕业设计 源码+数据库+论文+启动教程 项目启动教程:https://www.bilibili.com/video/BV1BfB2YYEnS

    基于java的苹果网吧计费管理系统设计与实现.docx

    基于java的苹果网吧计费管理系统设计与实现.docx

    纸中世界-跳跃游戏.sb3

    纸中世界-跳跃游戏.sb3

    Keysight 网络分析仪新建校准件操作指导

    本操作指导用于在 ENA 系列网络分析仪 E5080B 上自定义校准件。目前 Keysight 网络分析仪的 PNA 系列 N52xxB、P50xx 系列、P937x 系列、PXI 板卡式网分以及 ENA 系列的 E5080B、E5081B 的操作界面均统一到如下界面,操作方式相同。

    调查海域浮游动物各类群栖息密度的空间分布表格.docx

    调查海域浮游动物各类群栖息密度的空间分布表格.docx

    ssm框架Java项目源码-高校毕业生就业管理系统+jsp毕设-大作业.zip

    本项目“高校毕业生就业管理系统”是一套基于SSM框架(Spring+SpringMVC+MyBatis)精心开发的Java Web应用,旨在为高校毕业生、高校就业指导部门以及企业用户提供一个高效、便捷的就业信息管理平台。 系统主要功能包括:学生用户可以查看和发布个人简历,搜索并筛选合适的工作岗位,申请心仪的职位;企业用户可以发布招聘信息,筛选和查看应聘者的简历,进行面试邀请等操作;高校就业指导部门则可以对学生的就业情况进行统计和分析,以更好地提供就业指导服务。 此外,系统采用了B/S架构,用户只需通过浏览器即可访问,无需安装客户端软件,方便快捷。数据库设计合理,数据存储安全,系统性能稳定。 本项目的开发,不仅为计算机相关专业的学生提供了一个实践SSM框架的好机会,帮助他们更好地理解和掌握Java Web开发技术,还能有效提升高校毕业生的就业效率和质量。

    使用 Python 进行视频编辑.zip

    电影剪辑 笔记MoviePy 最近升级到 v2.0,引入了重大的重大变化。有关如何更新 v2.0 代码的更多信息,请参阅本指南。MoviePy(在线文档在此处)是一个用于视频编辑的 Python 库剪切、连接、插入标题、视频合成(又名非线性编辑)、视频处理和创建自定义效果。MoviePy 可以读取和写入所有最常见的音频和视频格式,包括 GIF,并且可以在 Windows/Mac/Linux 上运行,并搭载 Python 3.9+。例子在此示例中,我们打开一个视频文件,选择 10 到 20 秒之间的子剪辑,在屏幕中心添加标题,然后将结果写入新文件# Import everything needed to edit video clipsfrom moviepy import *# Load file example.mp4 and keep only the subclip from 00:00:10 to 00:00:20clip = VideoFileClip("long_examples/example2.mp4").with_subcl

    基于java的视频播放器系统设计与实现.docx

    基于java的视频播放器系统设计与实现.docx

    基于java的车辆出租管理系统设计与实现.docx

    基于java的车辆出租管理系统设计与实现.docx

    mqtt等协议的pcap文件

    mqtt等协议的pcap文件

    小白的Python入门教程部分章节源码.zip

    学习python

    修木工施工规范及流程.docx

    修木工施工规范及流程.docx

    适用于 Windows,Linux 和 Python 3 (3.5,3.6,3.7) 的 Tensorflow Faster R-CNN.zip

    适用于 Windows/Linux 和 Python 3 (3.5/3.6/3.7) 的 Tensorflow Faster R-CNNtf-faster-rcnn使用 Python 3 在 Windows 和 Linux 上使用 Tensorflow Faster R-CNN这是在 Windows 和 Linux 上编译 Faster R-CNN 的分支。它深受这里和这里的出色工作的启发。目前,此存储库支持 Python 3.5、3.6 和 3.7。感谢@morpheusthewhite请注意我没有时间或意图修复此分支的所有问题,因为我不将其用于商业用途。我创建此分支只是为了好玩。如果您想做出任何承诺,我们非常欢迎。Tensorflow 已经发布了一个对象检测 API。请参考它。https: //github.com/tensorflow/models/tree/master/research/object_detection如何使用此分支安装 tensorflow,最好是 GPU 版本。按照说明操作。如果没有安装 GPU 版本,则需要注释掉代码中的所有 GP

    章节2:编程基本概念之python程序的构成

    Python是一种高级、解释型、面向对象的编程语言,以其简洁的语法、强大的功能和广泛的应用领域而著称。它无需事先编译,代码在运行时逐行解释执行,提供了极大的灵活性和快速开发的能力。Python支持多种数据类型,包括整数、浮点数、字符串、布尔值、列表、元组、字典和集合等,以及丰富的操作符和流程控制结构,使得开发者可以编写出复杂且灵活的代码。 Python拥有一个广泛的标准库,涵盖了文件操作、网络通信、文本处理、正则表达式、数学运算等多个领域,为开发者提供了大量的模块和函数。此外,Python还拥有丰富的第三方库,如NumPy、Pandas、Matplotlib等用于数据分析和可视化的库,以及Django、Flask等用于Web开发的框架,这些库和框架进一步扩展了Python的应用领域和功能。 Python在Web开发、数据科学、人工智能、自动化运维和游戏开发等多个领域都有广泛的应用。在Web开发方面,Python提供了Django和Flask等强大的Web框架,使得开发者可以轻松地开发出各种Web应用和网站。在数据科学领域,Python是数据科学家的首选工具,其强大的数据处理能力和丰

    毕设源码-基于python的西西家居全屋定制系统的设计与实现_ijsj--论文-期末大作业+说明文档.rar

    本项目是基于Python语言开发的西西家居全屋定制系统,旨在为家居行业提供一个高效、智能的定制解决方案。项目涵盖了从客户需求分析、设计方案生成、材料选购到最终订单生成的全过程,力求实现家居定制的数字化和智能化。 在主要功能方面,系统具备强大的客户管理模块,能够详细记录和分析客户的定制需求。设计模块则采用先进的三维建模技术,为客户提供直观、真实的家居设计方案预览。此外,系统还整合了丰富的材料数据库,方便客户根据自身喜好和预算进行材料选择。 框架方面,项目采用了B/S架构,确保了系统的稳定性和可扩展性。后端使用Python的Django框架,前端则结合了HTML、CSS和JavaScript等技术,实现了用户界面的友好和响应速度。 开发此项目的目的,不仅是为了满足家居行业对个性化定制的需求,也为计算机相关专业的学生提供了一个实践和学习的平台,有助于提升他们的实际开发能力。

    简单连接到 Binance Public API.zip

    Binance公共API连接器Python 这是一个轻量级库,可作为Binance 公共 API的连接器支持的 API/api/*/sapi/*现货 Websocket 市场动态现货用户数据流现货 WebSocket API包含测试用例和示例可定制的基本 URL、请求超时和 HTTP 代理可以显示响应元数据安装pip install binance-connector文档https://binance-connector.readthedocs.ioRESTful API使用示例from binance.spot import Spotclient = Spot()# Get server timestampprint(client.time())# Get klines of BTCUSDT at 1m intervalprint(client.klines("BTCUSDT", "1m"))# Get last 10 klines of BNBUSDT at 1h intervalprint(client.k

    离线安装eclipse的aptana插件详细中文最新版本

    Aptana是一个非常强大,开源,JavaScript-focused的AJAX开发IDE。 Aptana的特点包括: 1JavaScript,HTML,CSS语言的Code Assist功能。 2Outliner(大纲):显示JavaScript,HTML和CSS的代码结构。

    学习自律养成小程序 微信小程序+SSM毕业设计 源码+数据库+论文+启动教程.zip

    学习自律养成小程序 微信小程序+SSM毕业设计 源码+数据库+论文+启动教程 项目启动教程:https://www.bilibili.com/video/BV1BfB2YYEnS

    认知能力评估表.docx

    认知能力评估表.docx

    数学建模学习资料 粒子群算法 先进算法讲义.pdf

    数学建模学习资料 粒子群算法 先进算法讲义.pdf

Global site tag (gtag.js) - Google Analytics