1. 程式人生 > >第八章 企業專案開發--分散式快取memcached

第八章 企業專案開發--分散式快取memcached

1、本地快取的問題

  • 本地快取速度一開始高於分散式快取,但是隨著其快取數量的增加,所佔記憶體越來越大,系統執行記憶體越來越小,最後系統會被拖慢(這一點與第二點聯絡起來)
  • 本地快取存於本機,其快取數量與大小受本機記憶體大小限制
  • 本地快取存於本機,其他機器的訪問不到這樣的快取

解決方案:分散式快取

  • Jboss cache:快取還存於本機,但是會同步更新到其他機器(解決了第三個問題,解決不了第一和第二個問題),如果快取機器數量很多,同步更新很耗時
  • memcached:快取存於其他機器,理論上快取數量與大小無限(因為叢集可伸縮),且不需要同步,所以即使快取機器數量很多,也無所謂,但是這樣就會造成單點故障問題,最簡單易行的解決方案是快取備份,即快取至少存兩份。

2、memcached Java客戶端的選用

當下常用的三種memcached Java客戶端:

  • Memcached Client for Java:memcached官方提供,基於Java BIO實現
  • SpyMemcached:基於Java NIO
  • XMemcached:基於Java NIO,併發效能優於XMemcached,實際上SpyMemcached效能也很高

三者的實驗比較結果:

所以,我們選用XMemcached來實現客戶端的編寫。

3、程式碼

在原來的程式碼結構上,我增加了一個ssmm0-cache模組,專門用於放置分散式快取相關(memcached、redis、spring cache)的程式碼。

專案整體結構:

3.1、ssmm0

pom.xml

  1 <?xml version="1.0" encoding="UTF-8"?>
  2 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3     xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
  4 
  5     <
modelVersion>4.0.0</modelVersion> 6 7 <groupId>com.xxx</groupId> 8 <artifactId>ssmm0</artifactId> 9 <version>1.0-SNAPSHOT</version> 10 11 <name>ssmm0</name> 12 <packaging>pom</packaging><!-- 父模組 --> 13 14 <!-- 管理子模組 --> 15 <modules> 16 <module>userManagement</module><!-- 具體業務1-人員管理系統 --> 17 <module>data</module><!-- 封裝資料操作 --> 18 <module>cache</module><!-- 快取模組 --> 19 </modules> 20 21 <properties> 22 <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> 23 <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> 24 </properties> 25 26 <!-- dependencyManagement不會引入實際的依賴,只是作為一個依賴池,供其和其子類使用 --> 27 <dependencyManagement> 28 <dependencies> 29 <!-- json --> 30 <dependency> 31 <groupId>com.alibaba</groupId> 32 <artifactId>fastjson</artifactId> 33 <version>1.1.39</version> 34 </dependency> 35 <!-- servlet --> 36 <dependency> 37 <groupId>javax.servlet</groupId> 38 <artifactId>javax.servlet-api</artifactId> 39 <version>3.0.1</version> 40 <scope>provided</scope> 41 </dependency> 42 <!-- spring --> 43 <dependency> 44 <groupId>org.springframework</groupId> 45 <artifactId>spring-core</artifactId> 46 <version>3.2.6.RELEASE</version> 47 </dependency> 48 <dependency> 49 <groupId>org.springframework</groupId> 50 <artifactId>spring-beans</artifactId> 51 <version>3.2.6.RELEASE</version> 52 </dependency> 53 <dependency> 54 <groupId>org.springframework</groupId> 55 <artifactId>spring-context</artifactId> 56 <version>3.2.6.RELEASE</version> 57 </dependency> 58 <dependency> 59 <groupId>org.springframework</groupId> 60 <artifactId>spring-web</artifactId> 61 <version>3.2.6.RELEASE</version> 62 </dependency> 63 <dependency> 64 <groupId>org.springframework</groupId> 65 <artifactId>spring-webmvc</artifactId> 66 <version>3.2.6.RELEASE</version> 67 </dependency> 68 <!-- 這個是使用velocity的必備包 --> 69 <dependency> 70 <groupId>org.springframework</groupId> 71 <artifactId>spring-context-support</artifactId> 72 <version>3.2.6.RELEASE</version> 73 </dependency> 74 <!-- mysql --> 75 <dependency> 76 <groupId>mysql</groupId> 77 <artifactId>mysql-connector-java</artifactId> 78 <version>5.1.27</version> 79 <scope>runtime</scope> 80 </dependency> 81 <!-- 資料來源 --> 82 <dependency> 83 <groupId>org.apache.tomcat</groupId> 84 <artifactId>tomcat-jdbc</artifactId> 85 <version>7.0.47</version> 86 </dependency> 87 <!-- mybatis --> 88 <dependency> 89 <groupId>org.mybatis</groupId> 90 <artifactId>mybatis</artifactId> 91 <version>3.1.1</version> 92 </dependency> 93 <dependency> 94 <groupId>org.mybatis</groupId> 95 <artifactId>mybatis-spring</artifactId> 96 <version>1.1.1</version> 97 </dependency> 98 <!-- velocity --> 99 <dependency> 100 <groupId>org.apache.velocity</groupId> 101 <artifactId>velocity</artifactId> 102 <version>1.5</version> 103 </dependency> 104 <dependency> 105 <groupId>velocity-tools</groupId> 106 <artifactId>velocity-tools-generic</artifactId> 107 <version>1.2</version> 108 </dependency> 109 <!-- 用於加解密 --> 110 <dependency> 111 <groupId>commons-codec</groupId> 112 <artifactId>commons-codec</artifactId> 113 <version>1.7</version> 114 </dependency> 115 <dependency> 116 <groupId>org.bouncycastle</groupId> 117 <artifactId>bcprov-jdk15on</artifactId> 118 <version>1.47</version> 119 </dependency> 120 <!-- 集合工具類 --> 121 <dependency> 122 <groupId>org.apache.commons</groupId> 123 <artifactId>commons-collections4</artifactId> 124 <version>4.0</version> 125 </dependency> 126 <!-- 字串處理類 --> 127 <dependency> 128 <groupId>org.apache.commons</groupId> 129 <artifactId>commons-lang3</artifactId> 130 <version>3.4</version> 131 </dependency> 132 <!-- http --> 133 <dependency> 134 <groupId>org.apache.httpcomponents</groupId> 135 <artifactId>httpclient</artifactId> 136 <version>4.2.6</version> 137 </dependency> 138 </dependencies> 139 </dependencyManagement> 140 141 <!-- 引入實際依賴 --> 142 <dependencies> 143 <!-- json --> 144 <dependency> 145 <groupId>com.alibaba</groupId> 146 <artifactId>fastjson</artifactId> 147 </dependency> 148 <!-- spring --> 149 <dependency> 150 <groupId>org.springframework</groupId> 151 <artifactId>spring-core</artifactId> 152 </dependency> 153 <dependency> 154 <groupId>org.springframework</groupId> 155 <artifactId>spring-beans</artifactId> 156 </dependency> 157 <dependency> 158 <groupId>org.springframework</groupId> 159 <artifactId>spring-context</artifactId> 160 </dependency> 161 <!-- 集合工具類 --> 162 <dependency> 163 <groupId>org.apache.commons</groupId> 164 <artifactId>commons-collections4</artifactId> 165 </dependency> 166 <!-- 字串處理類 --> 167 <dependency> 168 <groupId>org.apache.commons</groupId> 169 <artifactId>commons-lang3</artifactId> 170 </dependency> 171 </dependencies> 172 173 <build> 174 <resources> 175 <!-- 這裡配置了這一塊兒true,才可以讓指定檔案(這裡是src/main/resources/spring-data.xml)讀到pom.xml中的配置資訊 176 , 值得注意的是,如果src/main/resources下還有其他檔案,而你不想讓其讀pom.xml, 你還必須得把src/main/resources下的其餘檔案再配置一遍,配置為false(不可讀pom.xml), 177 如下邊的註釋那樣,否則,會報這些檔案找不到的錯誤 178 --> 179 <resource> 180 <directory>src/main/resources</directory> 181 <filtering>true</filtering> 182 <includes> 183 <include>*.xml</include> 184 <include>*.properties</include> 185 </includes> 186 </resource> 187 <!-- 188 <resource> 189 <directory>src/main/resources</directory> 190 <filtering>false</filtering> 191 <includes> 192 <include>*.properties</include> 193 </includes> 194 </resource> 195 --> 196 <resource> 197 <directory>src/main/resources</directory> 198 <filtering>false</filtering> 199 <includes> 200 <!-- 這裡如果不加這一條,那麼在spring-data.xml中配置的xml將找不到classpath:mapper/admin/AdminMapper.xml --> 201 <include>mapper/**/*.xml</include> 202 </includes> 203 </resource> 204 </resources> 205 </build> 206 207 <!-- 208 profiles可以定義多個profile,然後每個profile對應不同的啟用條件和配置資訊,從而達到不同環境使用不同配置資訊的效果 209 注意兩點: 210 1)<activeByDefault>true</activeByDefault>這種情況表示伺服器啟動的時候就採用這一套env(在這裡,就是prod) 211 2)當我們啟動伺服器後,想採用開發模式,需切換maven的env為dev,如果env的配置本身就是dev,需要將env換成rc或prod,點選apply,然後再將env切換成dev,點選apply才行 212 --> 213 <profiles> 214 <!-- 開發env --> 215 <profile> 216 <id>dev</id> 217 <activation> 218 <!-- 這裡為了測試方便,改為了true,在上線的時候一定要改成false,否則線上使用的就是這一套dev的環境了 --> 219 <activeByDefault>true</activeByDefault> 220 <property> 221 <name>env</name> 222 <value>dev</value> 223 </property> 224 </activation> 225 <properties> 226 <env>dev</env> 227 228 <jdbc.driverClassName>com.mysql.jdbc.Driver</jdbc.driverClassName> 229 <!-- 230 對於jdbc.url中內容的配置,如果需要配置 &amp;時,有兩種方法: 231 1)如下邊這樣,使用<![CDATA[XXX]]>包起來 232 2)使用jdbc.properties檔案來讀取此pom.xml,然後spring.xml再讀取jdbc.properties檔案 顯然,前者更方便,而且還省了一個jdbc.properties的檔案,但是,有的時候,還是會用後者的; 233 在使用後者的時候,注意三點: 234 1)需要修改上邊的build中的內容 235 2)需要在spring.xml中配置<context:property-placeholder location="classpath:jdbc.properties"/> 236 3)將jdbc.properties放在ssmm0-data專案中,之後需要將ssmm0-data專案的env配置為dev 237 --> 238 <jdbc.url><![CDATA[jdbc:mysql://127.0.0.1:3306/blog?zeroDateTimeBehavior=convertToNull&amp;useUnicode=true&amp;characterEncoding=utf-8]]></jdbc.url> 239 <jdbc.username>root</jdbc.username> 240 <jdbc.password>123456</jdbc.password> 241 242 <!-- memcache,多臺伺服器之間需要使用空格隔開,而不要使用英文逗號隔開,因為Xmemcached的AddrUtil原始碼是根據空格隔開的 --> 243 <memcached.servers><![CDATA[127.0.0.1:11211]]></memcached.servers> 244 <memcached.max.client>10</memcached.max.client><!-- 最多的客戶端數 --> 245 <memcached.expiretime>900</memcached.expiretime><!-- 過期時間900s --> 246 <memcached.hash.consistent>true</memcached.hash.consistent><!-- 是否使用一致性hash演算法 --> 247 <memcached.connection.poolsize>1</memcached.connection.poolsize><!-- 每個客戶端池子的連線數 --> 248 <memcached.op.timeout>2000</memcached.op.timeout><!-- 操作超時時間 --> 249 </properties> 250 </profile> 251 <!-- 預上線env --> 252 <profile> 253 <id>rc</id> 254 <activation> 255 <activeByDefault>false</activeByDefault> 256 <property> 257 <name>env</name>