Tomcat內存溢出(java.lang.OutOfMemoryError: PermGen space)的問題常常會在開發中遇到,面試中也有會問到的時候,因此我決定要記一下。javascript
首先,優化tomcat我給出三個方向:css
a) 內存優化:主要是對Tomcat啓動參數進行優化,咱們能夠在Tomcat啓動腳本中修改它的最大內存數等等。html
b) 線程數優化:Tomcat的併發鏈接參數,主要在Tomcat配置文件中server.xml中配置,好比修改最小空閒鏈接線程數,用於提升系統處理性能等等。java
c) 優化緩存:打開壓縮功能,修改參數,好比壓縮的輸出內容大小默認爲2KB,能夠適當的修改。面試
針對第一點,咱們須要瞭解如下6個參數:數據庫
1:-Xms 堆空間初始大小緩存
2:-Xmx 堆空間最大數值tomcat
3:-Xmn 年輕代的堆大小安全
4:-Xss 每一個線程堆大小服務器
5:-XX: PermSize 內存永久保留區域
6:-XX:MaxPermSize 內存最大永久保留區域
這幾個參數都是在Tomcat 啓動腳本(bin/catalina.bat)中進行配置的,通常開發環境配置時咱們就會預先配置好。只要把如下代碼加入文件中,而且重啓Tomcat便可。
JAVA_OPTS="-Xms2048m -Xmx2048m -Xmn1024m -Xss1024K -XX:PermSize=128m -XX:MaxPermSize=512m"(具體數值按公司服務器大小以及實際自行配置)
第二第三點合在一塊兒優化也是修改Tomcat的配置文件server.xml。初學者通常使用這個文件都是爲了修改端口號。
咱們打開server.xml找到Connector 標籤項,能夠看到默認配置以下:
1 <Connector port="8080" protocol="HTTP/1.1"
2 connectionTimeout="20000"
3 redirectPort="8443" />
其中port表明服務端口;protocol表明協議類型;connectionTimeout表明鏈接超時時間,單位爲毫秒;redirectPort表明安全通訊(https)轉發端口,通常配置成443。
咱們須要調整鏈接器connector的併發處理能力,好比這樣:
1 <Connector port="8080"
2 protocol="HTTP/1.1"
3 maxHttpHeaderSize="8192"
4 maxThreads="1000"
5 minSpareThreads="100"
6 maxSpareThreads="1000"
7 minProcessors="100"
8 maxProcessors="1000"
9 enableLookups="false"
10 compression="on"
11 compressionMinSize="2048"
12 compressableMimeType="text/html,text/xml,text/javascript,text/css,text/plain"
13 connectionTimeout="20000"
14 URIEncoding="utf-8"
15 acceptCount="1000"
16 redirectPort="8443"
17 disableUploadTimeout="true"/>
參數說明:
maxThreads 客戶請求最大線程數
minSpareThreads Tomcat初始化時建立的 socket 線程數
maxSpareThreads Tomcat鏈接器的最大空閒 socket 線程數
enableLookups 若設爲true, 則支持域名解析,可把 ip 地址解析爲主機名
redirectPort 在須要基於安全通道的場合,把客戶請求轉發到基於SSL 的 redirectPort 端口
acceptAccount 監聽端口隊列最大數,滿了以後客戶請求會被拒絕(不能小於maxSpareThreads )
minProcessors 服務器建立時的最小處理線程數
maxProcessors 服務器同時最大處理線程數
URIEncoding URL統一編碼
compression 打開壓縮功能
compressionMinSize 啓用壓縮的輸出內容大小,這裏面默認爲2KB
compressableMimeType 壓縮類型
connectionTimeout 定義創建客戶鏈接超時的時間. 若是爲 -1, 表示不限制創建客戶鏈接的時間
推薦一篇好文章:http://blog.csdn.net/kally_wang/article/details/74989885 目前我仍是有點看不大明白的,還得學習下,其中提到了線程池,我在想這和數據庫鏈接池是應該有殊途同歸之處。