日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

tomcat(9)Session管理

發布時間:2023/12/3 编程问答 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 tomcat(9)Session管理 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
【0】README 0.0)本文部分描述轉自“深入剖析tomcat”,旨在學習tomcat-Session管理” 的基礎知識; 0.1)Catalina通過一個稱為Session 管理器的組件來管理建立的Session對象,該組件由org.apache.catalina.Manager接口來表示。(干貨——catalina通過Session管理器組件來管理Session對象) 0.2)Session管理器:需要與一個Context容器相關聯,且必須與一個Context容器關聯;(干貨——intro to Session Manager) 0.3)Session管理器:負責創建,更新,銷毀Session對象,當有請求到來時,要會返回一個有效的Session對象; 0.4)default case:Session管理器會將其所管理的 Session對象存放在內存中。但在tomcat中,Session管理器也可以將Session 對象進行持久化,存儲到文件存儲器或通過jdbc寫入到database; 0.5)for complete source code, please visit ?https://github.com/pacosonTang/HowTomcatWorks/tree/master/chapter9;
【1】Session對象 1)intro to Session object:Session對象由 javax.servlet.http.HttpSession接口表示; 2)UML類圖實例:
【1.1】Session接口 1)Session接口是作為 Catalina內部的外觀類使用的。Session 接口的標準實現 StandardSession類也實現了 javax.servlet.http.HttpSession 接口; public interface Session { public static final String SESSION_CREATED_EVENT = "createSession"; public static final String SESSION_DESTROYED_EVENT = "destroySession"; public String getAuthType(); public void setAuthType(String authType); public long getCreationTime(); public void setCreationTime(long time); public String getId(); public void setId(String id); public String getInfo(); public long getLastAccessedTime(); public Manager getManager(); public void setManager(Manager manager); public int getMaxInactiveInterval(); public void setMaxInactiveInterval(int interval); public void setNew(boolean isNew); public Principal getPrincipal(); public void setPrincipal(Principal principal); public HttpSession getSession(); public void setValid(boolean isValid); public boolean isValid(); public void access(); public void addSessionListener(SessionListener listener); public void expire(); public Object getNote(String name); public Iterator getNoteNames(); public void recycle();public void removeNote(String name);public void removeSessionListener(SessionListener listener);public void setNote(String name, Object value); } 2)Session對象總是存在于Session管理器中的:可以使用 getManager()方法或 setManager() 方法將Session實例和某個 Session管理器相關聯; 2.1)對某個Session實例來說:在與其Session 管理器相關聯的某個Context實例內,該Session對象有一個唯一的標識符,可以通過 setId() 方法和 getId() 方法來訪問該 Session 標識符; 2.2)Session管理器會調用 getLastAccessedTime()方法:根據返回值來判斷一個Session對象的有效性;(Session管理器會調用setValid()方法來設置或重置其有效性) 2.3)每當訪問一個Session實例時: 會調用其 access()方法 來修改 Session對象的最后訪問時間; 2.4)最后:Session管理器會調用Session對象的 expire()方法將其設置為過期, 也可以通過getSession() 方法獲取一個經過 Session 外觀類包裝的 HttpSession 對象;
【1.2】StandardSession類 1)StandardSession類實現了 java.lang.Serilizable接口,便于序列化Session對象;(干貨——StandardSession類實現了 java.lang.Serilizable接口,便于序列化Session對象) 2)StandardSession類的構造函數接收Manager接口的一個實例:迫使一個Session對象必須擁有一個Session管理器實例;(干貨——注意迫使二字) <span style="font-size:18px;">public StandardSession(Manager manager);</span> 3)下面是StandardSession實例的一些比較重要的私有變量,用于維護該StandardSession實例的一些狀態。注意,帶有transient關鍵字的變量是無法序列化的; Attention)在tomcat5中,上述變量是受保護的,而tomcat4中,它們都是私有變量。每個變量都有與之對應的一個訪問器和一個轉變器(修改器)(get 和 set 方法) class StandardSessionimplements HttpSession, Session, Serializable { public StandardSession(Manager manager) { super();this.manager = manager;if (manager instanceof ManagerBase)this.debug = ((ManagerBase) manager).getDebug();}private static final String NOT_SERIALIZED ="___NOT_SERIALIZABLE_EXCEPTION___"; private HashMap attributes = new HashMap();private transient String authType = null; private transient Method containerEventMethod = null; private static final Class containerEventTypes[] = { String.class, Object.class }; private long creationTime = 0L; private transient int debug = 0;private transient boolean expiring = false;private transient StandardSessionFacade facade = null;private String id = null;private static final String info = "StandardSession/1.0";private long lastAccessedTime = creationTime;private transient ArrayList listeners = new ArrayList(); private Manager manager = null;private int maxInactiveInterval = -1;private boolean isNew = false;private boolean isValid = false;private transient HashMap notes = new HashMap();private transient Principal principal = null;private static StringManager sm =StringManager.getManager(Constants.Package); private static HttpSessionContext sessionContext = null;private transient PropertyChangeSupport support =new PropertyChangeSupport(this); private long lastUsedTime = creationTime; } 4)其中 getSession()方法會通過一個自身實例來創建 StandardSessionFacade 類的一個實例,并將其返回;(干貨——如果是單例模式的話,該方法寫的就很有問題,當然只是小生我一個小小的猜想) public HttpSession getSession() {if(facade==null){facade = new StandardSessionFacade(this);}return facade; }5)設置Session為過期:若Session管理器中的某個Session對象 在某個時間長度內都沒有被訪問 的話,會被 Session 管理器設置為過期,這個時間長度由變量 maxInactiveInterval 的值來指定;(干貨——將一個Session對象設置為過期主要是通過Session接口的expire() 方法來完成的); public void expire(boolean notify) {// Mark this session as "being expired" if neededif (expiring)return;expiring = true;setValid(false);// Remove this session from our manager's active sessionsif (manager != null)manager.remove(this);// Unbind any objects associated with this sessionString keys[] = keys();for (int i = 0; i < keys.length; i++)removeAttribute(keys[i], notify);// Notify interested session event listenersif (notify) {fireSessionEvent(Session.SESSION_DESTROYED_EVENT, null);}// Notify interested application event listeners// FIXME - Assumes we call listeners in reverse orderContext context = (Context) manager.getContainer();Object listeners[] = context.getApplicationListeners();if (notify && (listeners != null)) {HttpSessionEvent event =new HttpSessionEvent(getSession());for (int i = 0; i < listeners.length; i++) {int j = (listeners.length - 1) - i;if (!(listeners[j] instanceof HttpSessionListener))continue;HttpSessionListener listener =(HttpSessionListener) listeners[j]; try {fireContainerEvent(context, // highlight line."beforeSessionDestroyed",listener);listener.sessionDestroyed(event); // destroy session.fireContainerEvent(context, // also highlight line."afterSessionDestroyed",listener);} catch (Throwable t) {try {fireContainerEvent(context,"afterSessionDestroyed",listener);} catch (Exception e) {;}// FIXME - should we do anything besides log these?log(sm.getString("standardSession.sessionEvent"), t);}}}// We have completed expire of this sessionexpiring = false;if ((manager != null) && (manager instanceof ManagerBase)) {recycle();} }
private void fireContainerEvent(Context context,String type, Object data)throws Exception {if (!"org.apache.catalina.core.StandardContext".equals(context.getClass().getName())) {return; // Container events are not supported}// NOTE: Race condition is harmless, so do not synchronizeif (containerEventMethod == null) {containerEventMethod =context.getClass().getMethod("fireContainerEvent", // terrific line.containerEventTypes);}Object containerEventParams[] = new Object[2];containerEventParams[0] = type;containerEventParams[1] = data;containerEventMethod.invoke(context, containerEventParams);}
【1.3】StandardSessionFacade類 1)為了傳遞一個 Session對象給servlet實例,Catalina會實例化 StandardSession 類,填充該Session對象,然后再將其傳給 servlet實例;(干貨——但是,請注意下面的但是) 2)但是(呵呵):Catalina傳遞的是Session的 外觀類 StandardSessionFacade的實例,該類僅僅實現了 javax.servlet.http.HttpSession接口中的方法; 3)這樣,servlet程序員就不能將 HttpSession對象向下轉換為 ?StandardSession 類型,也阻止了 servlet程序員訪問一些敏感方法;
【2】Manager 1)Catalina通過一個稱為Session 管理器的組件來管理建立的Session對象,該組件由org.apache.catalina.Manager接口來表示。 2)Catalina中:org.apache.catalina.session包中有一個名為 ManagerBase的工具類,該類提供了常見功能的實現。ManagerBase類有兩個直接子類,分別是 StandardManager 類 和 PersistentManagerBase類;(干貨——引入ManagerBase的工具類) 3)當Catalina運行時:StandardManager實例會將Session對象存儲在內存中,但是,當Catalina關閉時,它(Manager)會將當前內存中的所有Session對象存儲到一個文件中。當再次啟動Catalina時,又會將這些Session 對象重新載入到內存中; 4)PersistentManagerBase類:該類會將 Session對象存儲到輔助存儲器中,其類圖如下:(干貨——PersistentManagerBase類會將 Session對象存儲到輔助存儲器中)
【2.1】Manager接口 1)intro to Manager interface:Session管理器是 Manager接口的實例; 2)Manager接口的源代碼 public interface Manager { public Container getContainer(); public void setContainer(Container container); public DefaultContext getDefaultContext(); public void setDefaultContext(DefaultContext defaultContext); public boolean getDistributable(); public void setDistributable(boolean distributable); public String getInfo(); public int getMaxInactiveInterval(); public void setMaxInactiveInterval(int interval); public void add(Session session); public Session createEmptySession(); public void addPropertyChangeListener(PropertyChangeListener listener); public Session createSession(); public Session findSession(String id) throws IOException; public Session[] findSessions(); public void load() throws ClassNotFoundException, IOException; public void remove(Session session); public void removePropertyChangeListener(PropertyChangeListener listener); public void unload() throws IOException; } 對上述代碼的分析(Analysis):
A1)getContainer()方法和setContainer()方法:以便將一個Manager實現與一個 Context容器相關聯; A2)createSession()方法:用來創建一個Session實例; A3)add()方法:會將一個 Session實例添加到 Session池中; A4)remove()方法:則會將一個 Session實例從 Session池中移除; A5)getMaxInactiveInterval()方法和 setMaxInactiveInterval()方法:用來獲取或設置一個時間長度,單位為秒; A6)Session管理器: 會以此作為一個Session對象的最長存活時間; A7)load()方法和unload() 方法:用來將Session對象持久化到輔助存儲器中(干貨——load方法和unload方法) A7.1)load方法:該方法會將介質中的Session對象重新載入到內存中; A7.2)unload方法:該方法會將當前活動的 Session對象存儲到 Manager 實現指定的介質中; 【2.2】ManagerBase類(org.apache.catalina.session.ManagerBase) 1)ManagerBase.createSession()方法:該方法會創建一個新的Session對象, 每個Session對象都有一個 唯一的標識符; 2)generateSessionId()方法:可以通過 ManagerBase類受保護的方法generateSessionId() 方法來返回一個唯一的標識符; Attention)一個活動的Session對象:指的是有效的,還未過期的Session對象;(干貨——活動的Session對象的定義) 3)某個Context容器的 Session管理器會管理該 Context容器中所有活動的Session對象。這些活動的Session對象都存儲在一個名為sessions 的 HashMap變量中: protected HashMap sessions = new HashMap(); 4)add方法,remove方法,findSession方法 public void add(Session session) {synchronized (sessions) {sessions.put(session.getId(), session);if( sessions.size() > maxActive ) {maxActive=sessions.size();}}} public void remove(Session session) {synchronized (sessions) {sessions.remove(session.getId());}} public Session[] findSessions() {Session results[] = null;synchronized (sessions) {results = new Session[sessions.size()];results = (Session[]) sessions.values().toArray(results);}return (results);} public Session findSession(String id) throws IOException {if (id == null)return (null);synchronized (sessions) {Session session = (Session) sessions.get(id);return (session);}} 【2.3】StandardManger類 1)intro to StandardManager:StandardManager類是Manager接口的標準實現,該類將 Session 存儲于內存中; 1.1)該類還實現了Lifecycle接口:這樣就可以由與其關聯的Context容器來啟動和關閉;其中stop方法的實現會調用 unload() 方法,以便將有效的Session對象序列化到一個名為"SESSION.ser"的文件中,而且每個Context容器都會產生這樣一個文件;(干貨——引入了SESSION.ser文件,unload() 方法持久化Session對象到文件 1.2)SESSION.ser文件:位于環境變量 CATALINA_HOME指定的目錄下的work目錄中; 看個荔枝)在tomcat4或tomcat5中,如果運行了實例應用程序,就可以在 CATALINA_HOME/work/Standalone/localhost/examples 目錄下找到 SESSION.ser文件; 1.3)當StandardManager實例再次啟動時:這些Session 對象會通過調用load() 方法重新讀入內存中;(干貨——load()方法將持久化文件中的Session對象重新讀入內存) 2)Session管理器:還要負責銷毀那些已經失效的Session對象,在tomcat4中的 StandardManager類中,這項工作是由一個專門的線程來完成的; public void run() { //org.apache.catalina.session.StandardManager// Loop until the termination semaphore is setwhile (!threadDone) {threadSleep();processExpires();}} 對上述代碼的分析(Analysis): A1)threadSleep()方法:會使線程休眠一段時間,長度由checkInterval指定,default==60 秒; A2)processExpir()方法:會遍歷由 Session管理器管理的所有Session對象,將Session實例的lastAccessedTime屬性值與當前時間進行比較,如果兩者之間的差值超過了變量 maxInactiveInterval 指定的數值,則會調用 Session接口的 expire() 方法使這個Session實例過期; private void processExpires() { // org.apache.catalina.session.StandardManager.processExpires()long timeNow = System.currentTimeMillis();Session sessions[] = findSessions();for (int i = 0; i < sessions.length; i++) {StandardSession session = (StandardSession) sessions[i];if (!session.isValid())continue;int maxInactiveInterval = session.getMaxInactiveInterval();if (maxInactiveInterval < 0)continue;int timeIdle = // Truncate, do not round up(int) ((timeNow - session.getLastUsedTime()) / 1000L);if (timeIdle >= maxInactiveInterval) {try {expiredSessions++;session.expire();// highlight line.} catch (Throwable t) {log(sm.getString("standardManager.expireException"), t);}}}} Attention) A1)在tomcat5中,StandardManager類已不再實現 java.lang.Runnable接口。其中的 backgroundporocess() 方法會直接調用 tomcat5 中StandardManager對象的 processExpires()方法; public voiid backgroundProcess() { //org.apache.catalina.session.StandardManager.backgroundProcess()processExpire(); } public void expire() { //org.apache.catalina.sessoin.StandardManager.expire()expire(true);} public void expire(boolean notify) { // the same as the above.// Mark this session as "being expired" if neededif (expiring)return;expiring = true;setValid(false);// Remove this session from our manager's active sessionsif (manager != null)manager.remove(this);// Unbind any objects associated with this sessionString keys[] = keys();for (int i = 0; i < keys.length; i++)removeAttribute(keys[i], notify);// Notify interested session event listenersif (notify) {fireSessionEvent(Session.SESSION_DESTROYED_EVENT, null);}// Notify interested application event listeners// FIXME - Assumes we call listeners in reverse orderContext context = (Context) manager.getContainer();Object listeners[] = context.getApplicationListeners();if (notify && (listeners != null)) {HttpSessionEvent event =new HttpSessionEvent(getSession());for (int i = 0; i < listeners.length; i++) {int j = (listeners.length - 1) - i;if (!(listeners[j] instanceof HttpSessionListener))continue;HttpSessionListener listener =(HttpSessionListener) listeners[j];try {fireContainerEvent(context,"beforeSessionDestroyed",listener);listener.sessionDestroyed(event);fireContainerEvent(context,"afterSessionDestroyed",listener);} catch (Throwable t) {try {fireContainerEvent(context,"afterSessionDestroyed",listener);} catch (Exception e) {;}// FIXME - should we do anything besides log these?log(sm.getString("standardSession.sessionEvent"), t);}}}// We have completed expire of this sessionexpiring = false;if ((manager != null) && (manager instanceof ManagerBase)) {recycle();}} 【2.4】PersistentManagerBase類(參見上圖) 1)intro to PersistentManagerBase:該類是所有持久化Session管理器的 父類; 1.1)StandardManager實例 和 持久化Session管理器(PersistentManagerBase)的區別:在于后者中存儲器的表示形式,即存儲Session 對象的輔助存儲器的形式,使用Store對象來表示; 2)在持久化Session管理器中:Session對象可以備份,也可以換出;(干貨——Session對象可以備份,也可以換出);如果server destroyed,就可以從存儲器中獲取活動的Session對象; 2.1)當Session對象被換出:他會被移動到存儲器中,因為當前活動的Session對象數超過了上限值,或者這個Session對象閑置了過長時間;(干貨——換出是為了節省內存空間) 2.2)tomcat4中,PersistentManagerBase類實現了java.lang.Runnable 接口:使用一個專門的線程來執行備份和換出活動Session對象的任務;(run() 方法如下 )(干貨——注意是tomcat4,使用一個專門的線程來執行備份和換出活動Session對象的任務) public void run() { //org.apache.catalina.session.PersistentManagerBase.run()while(!threadDone) {threadSleep();processExpires(); // 檢查Session對象是否過期.processPersistenceChecks();} } public void processPersistenceChecks(){ // org.apache.catalina.session.PersistentManagerBase.processPersistenceChecks()processMaxIdleSwaps(); // highlight line.(下面我們只以processMaxIdleSwaps方法為例進行調用過程的分析,其他兩個方法類似)processMaxActiveSwaps(); // highlight line.processMaxIdleBackups(); // highlight line. } /*** Swap idle sessions out to Store if they are idle too long.若session空閑太久則換出*/protected void processMaxIdleSwaps() {if (!isStarted() || maxIdleSwap < 0)return;Session sessions[] = findSessions();long timeNow = System.currentTimeMillis();// Swap out all sessions idle longer than maxIdleSwap// FIXME: What's preventing us from mangling a session during// a request?if (maxIdleSwap >= 0) {for (int i = 0; i < sessions.length; i++) {StandardSession session = (StandardSession) sessions[i];if (!session.isValid())continue;long lastAccessed = ((StandardSession)session).getLastUsedTime();int timeIdle = // Truncate, do not round up(int) ((timeNow - lastAccessed) / 1000L);if (timeIdle > maxIdleSwap && timeIdle > minIdleSwap) {if (debug > 1)log(sm.getString("persistentManager.swapMaxIdle",session.getId(), new Integer(timeIdle)));try {swapOut(session); // highlight line.} catch (IOException e) {; // This is logged in writeSession()}}}}}protected void swapOut(Session session) throws IOException { //換出操作if (store == null ||!session.isValid() ||isSessionStale(session, System.currentTimeMillis()))return;((StandardSession)session).passivate();writeSession(session); // highlight line.super.remove(session);session.recycle();}protected void writeSession(Session session) throws IOException {if (store == null ||!session.isValid() ||isSessionStale(session, System.currentTimeMillis()))return;try {store.save(session); //highlight line.sessionSwapIgnore.remove(session.getId());} catch (IOException e) {log(sm.getString("persistentManager.serializeError", session.getId(), e));throw e;}}
3)tomcat5中,PersistentManagerBase類不再實現 java.lang.Runnable 接口。備份和換出Session對象的任務由其 backgroundProcess() 管理器完成,后者由其相關聯的 StandardContext 實例周期性地調用; (下面分析 換出和備份Session實例的實現過程)
4)PersistentManagerBase類換出Session實例 4.1)換出條件:只有當活動Session對象的數量超過了變量 maxActiveSessions指定的上限值,或者該Session對象閑置了很長時間后,才會被換出;(干貨——Session對象被換出的condition) 4.1.1)當活動的Session實例數量過多時:PersistentManagerBase會將Session對象換出直到數量等于?maxActiveSessions;(參見 processMaxActiveSwaps()方法) 4.1.2)當某個Session對象閑置了過長時間:PersistentManagerBase會依據兩個變量來決定是否將這個Session換出,這兩個變量是minIdleSwap and maxIdleSwap;如果某個Session對象的 lastAccessedTime 屬性的值超過了 minIdleSwap 和 maxIdleSwap 的值,就會將這個Session換出;(干貨——為了防止換出Session對象,將maxIdleSwap設置為負數,參見 processMaxIdleSwaps() 方法) 4.2)如何查找Session實例:因為Session對象可以被換出,所以他既可能駐留在內存中,也可能駐留在存儲器中,因此,findSession(String id ) 方法首先會在內存中查找是否存在該Session對象,若沒有找到,才會在存儲器中繼續find。(PersistentManagerBase.findSession() 方法的源代碼如下)(干貨——Session對象被換出后,可能駐留在內存中,也可能駐留在存儲器中) public Session findSession(String id) throws IOException { //org.apache.catalina.session.PersistentManagerBase.findSession()Session session = super.findSession(id);if (session != null)return (session);// See if the Session is in the Storesession = swapIn(id); // highlight line.return (session);}protected Session swapIn(String id) throws IOException { // the same as the above.if (store == null)return null;if (sessionSwapIgnore.contains(id)) {return null;}Session session = null;try {session = store.load(id); // highlight line.} catch (ClassNotFoundException e) {log(sm.getString("persistentManager.deserializeError", id, e));throw new IllegalStateException(sm.getString("persistentManager.deserializeError", id, e));}if (session == null) {sessionSwapIgnore.put(id,id);return (null);}if (!session.isValid()|| isSessionStale(session, System.currentTimeMillis())) {log("session swapped in is invalid or expired");session.expire(); // highlight line.store.remove(id); // highlight line.sessionSwapIgnore.put(id,id);return (null);}if(debug > 2)log(sm.getString("persistentManager.swapIn", id));session.setManager(this);// To make sure the listener knows about it.((StandardSession)session).tellNew();add(session);((StandardSession)session).activate();return (session);}
5)PersistentManagerBase類備份Session實例 5.1)備份條件:PersistentManagerBase實例 僅僅會備份那些空閑時間超過了變量 maxIdleBackup 的Session對象。 5.2)processMaxIdleBackups()方法:負責對該Session對象進行備份操作;
【2.5】PersistentManager類 1)intro to?PersistentManager類:該類繼承了?PersistentManagerBase類,僅僅是多添加了兩個屬性而已; public final class PersistentManager extends PersistentManagerBase { //org.apache.catalina.session.PersistentManager private static final String info = "PersistentManager/1.0"; protected static String name = "PersistentManager"; public String getInfo() {return (this.info);} public String getName() {return (name);}} 【2.6】DistributedManager類(應用與集群環境) 1)intro to?DistributedManager類:繼承自PersistentManagerBase類,用于兩個或多個節點的集群環境; 1.1)一個節點表示部署的一臺 tomcat server:而集群中的節點 可以在同一臺機器上,也可以在不同的machine上 1.2)每個節點必須使用 DistributedManager實例作為其Session管理器:這樣才能支持復制 Session對象,這也是DistributedManager的主要功能?;(干貨——DistributedManager的主要功能) 2)DistributedManager會向其他節點發送消息:為了實現copy Session對象的目的,當創建或銷毀Session對象時,DistributedManager會向其他節點發送消息;(干貨——DistributedManager會向其他節點發送消息) 2.2)而且:集群中的節點也必須能夠接受其他節點發送的消息。這樣,http 請求才能到達集群中的任意節點; 3)catalina在org.apache.catlina.cluster中提供了一些工具:這些工具用于 與集群中其他節點的DistributedManager實例發送和接收消息;其中,ClusterSender 類用于向集群中的其他節點發送消息,ClusterReceiver 則用于接收集群中其他節點發送的消息;(干貨——catalina在org.apache.catlina.cluster中提供了一些工具,如ClusterSender and ClusterReceiver) 4)DistributedManager.createSession()方法: 要創建一個Sesssion對象,存儲在當前?DistributedManager 實例中,并使用 ClusterSender實例向其他節點發送消息; public Session createSession() {Session session = super.createSession(); // step1.ObjectOutputStream oos = null;ByteArrayOutputStream bos = null;ByteArrayInputStream bis = null;try {bos = new ByteArrayOutputStream();oos = new ObjectOutputStream(new BufferedOutputStream(bos));((StandardSession)session).writeObjectData(oos);oos.close();byte[] obs = bos.toByteArray();clusterSender.send(obs); // step2.if(debug > 0)log("Replicating Session: "+session.getId());} catch (IOException e) {log("An error occurred when replicating Session: "+session.getId());}return (session);} 對以上代碼的分析(Analysis): step1)調用超類的 createSession()方法為自身創建一個 Session對象; step2)使用ClusterSender實例,以字節數組的形式將該Session對象發送到集群中的其他節點; 5)DistributedManager類還實現了 java.lang.Runnable 接口: 這樣就可以使用一個 專門的thread 來檢查 Session對象是否過期,并從集群中的其他節點接收消息;(其run() 方法的源代碼如下)? public void run() { // org.apache.catalina.session.DistributedManager.run()// Loop until the termination semaphore is setwhile (!threadDone) {threadSleep();processClusterReceiver(); // 以該方法為例進行說明.processExpires();processPersistenceChecks();}} /*** Called from our background thread to process new received Sessions**/public void processClusterReceiver() { //org.apache.catalina.session.DistributedManager.processClusterReceiver()Object[] objs = clusterReceiver.getObjects();StandardSession _session = null;ByteArrayInputStream bis = null;Loader loader = null;ClassLoader classLoader = null;ObjectInputStream ois = null;byte[] buf = new byte[5000];ReplicationWrapper repObj = null;for(int i=0; i < objs.length;i++) {try {bis = new ByteArrayInputStream(buf);repObj = (ReplicationWrapper)objs[i];buf = repObj.getDataStream();bis = new ByteArrayInputStream(buf, 0, buf.length);if (container != null)loader = container.getLoader();if (loader != null)classLoader = loader.getClassLoader();if (classLoader != null)ois = new CustomObjectInputStream(bis,classLoader);elseois = new ObjectInputStream(bis);_session = (StandardSession) super.createSession();_session.readObjectData(ois);_session.setManager(this);if (debug > 0)log("Loading replicated session: "+_session.getId());} catch (IOException e) {log("Error occurred when trying to read replicated session: "+e.toString());} catch (ClassNotFoundException e) {log("Error occurred when trying to read replicated session: "+e.toString());} finally {if (ois != null) {try {ois.close();bis = null;} catch (IOException e) {;}}}}} 【3】存儲器(存儲器是org.apache.catalina.Store接口的實例) 1)intro to 存儲器是為Session管理器管理的 Session對象提供持久化存儲器的一個組件;(干貨——intro to 存儲器——org.apache.catalina.Store接口的實例) 2)Store接口的源代碼如下: public interface Store { // org.apache.catalina.Storepublic String getInfo(); public Manager getManager(); public void setManager(Manager manager); public int getSize() throws IOException; public void addPropertyChangeListener(PropertyChangeListener listener); public String[] keys() throws IOException; public Session load(String id) throws ClassNotFoundException, IOException;public void remove(String id) throws IOException; public void clear() throws IOException; public void removePropertyChangeListener(PropertyChangeListener listener); public void save(Session session) throws IOException; } 3)Store接口中有兩個方法:save()方法和 load() 方法(干貨——注意save方法和 load方法,它們的作用分別是什么) 3.1)save方法:用于將指定的 Session對象存儲到某種持久性存儲器中; 3.2)load方法:會從存儲器中,依據Session對象 的標識符將該Session對象載入(加載)到內存中; 補充)keys()方法:會以字符串數組的形式返回所有 Session 對象的標識符; 4)其UML 類圖如下所示:
【3.1】StoreBase類圖(見上圖) 1)intro to?StoreBase:該類是一個抽象類,提供了一些基本功能,有兩個子類; 2)StoreBase類:并沒有實現Store接口的save()方法 和 load()方法,因為,這兩個方法依賴于持久化Session對象的存儲器的類型; 3)在tomcat4中:StoreBase類使用另一個線程周期性地檢查 Session對象:從活動的Session的集合中移除過期的Session對象;(下面是 tomcat4 中 StoreBase.run()方法的源代碼,注意是tomcat4 public void run() { // org.apache.catalina.session.StoreBase.run()// Loop until the termination semaphore is setwhile (!threadDone) {threadSleep();processExpires();}} 對以上代碼的分析(Analysis): A1)processExpires方法:會獲取所有活動的 Session對象,檢查 每個Session對象的 lastAccessedTime屬性值,刪除那些長時間不活動的Session對象;源代碼如下:(干貨——再次提及了processExpires方法,注意其功能) protected void processExpires() {// org.apache.catalina.session.StoreBase.processExpires()long timeNow = System.currentTimeMillis();String[] keys = null;if(!started) {return;}try {keys = keys();} catch (IOException e) {log (e.toString());e.printStackTrace();return;}for (int i = 0; i < keys.length; i++) {try {StandardSession session = (StandardSession) load(keys[i]);if (session == null) {continue;}if (!session.isValid()) {continue;}int maxInactiveInterval = session.getMaxInactiveInterval();if (maxInactiveInterval < 0) {continue;}int timeIdle = // Truncate, do not round up(int) ((timeNow - session.getLastUsedTime()) / 1000L);if (timeIdle >= maxInactiveInterval) {if ( ( (PersistentManagerBase) manager).isLoaded( keys[i] )) {// recycle old backup sessionsession.recycle();} else {// expire swapped out sessionsession.expire();}remove(session.getId());}} catch (Exception e) {log ("Session: "+keys[i]+"; "+e.toString());try {remove(keys[i]);} catch (IOException e2) {log (e2.toString());e2.printStackTrace();}}}}
Attention)在tomcat5中,不再使用專用的線程調用 processExpire()方法 ,而是,相關聯的 PersistentManagerBase實例的 backgroundProcess()方法會周期性地調用 processExpires() 方法;
【3.2】FileStore類 1)intro to FileStore:FileStore類會將 Session對象存儲到某個文件中,文件名會使用Session對象的 標識符再加上一個后綴 ".session" 構成;(干貨——FileStore類的作用) 2)文件位于臨時的工作dir下:可以調用FileStore.setDirectory() 方法進行修改臨時目錄的位置; (干貨——理清FileStore.save and FileStore.load方法的作用)
3)save方法:使用 java.io.ObjectOutputStream類將 Session對象進行序列化,故所有存儲在Session實例中的對象都要實現 java.lang.Serializable接口; 4)load方法:該方法使用 java.io.ObjectInputStream 類實現 Session 對象的反序列化;
【3.3】JDBCStore類 1)intro to JDBCStore:該類將Session對象通過JDBC 存入 db中;(干貨——JDBCStore類的作用)
【4】app program 1)intro to app program該應用程序中的Context容器使用StandardManager實例來管理Session對象,為了測試該應用程序,可以使用第3個示例servlet:SessionServlet; public class SessionServlet extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {System.out.println("SessionServlet -- service");response.setContentType("text/html");PrintWriter out = response.getWriter();out.println("<html>");out.println("<head><title>SessionServlet</title></head>");out.println("<body>");String value = request.getParameter("value");HttpSession session = request.getSession(true);out.println("<br>the previous value is " +(String) session.getAttribute("value"));out.println("<br>the current value is " + value);session.setAttribute("value", value);out.println("<br><hr>");out.println("<form>");out.println("New Value: <input name=value>");out.println("<input type=submit>");out.println("</form>");out.println("</body>");out.println("</html>");} }
【4.1】Bootstrap類 1)源代碼如下
public static void main(String[] args) {//invoke: http://localhost:8080/myApp/SessionSystem.setProperty("catalina.base", System.getProperty("user.dir"));Connector connector = new HttpConnector();Wrapper wrapper1 = new SimpleWrapper();wrapper1.setName("Session");wrapper1.setServletClass("SessionServlet");Context context = new StandardContext();// StandardContext's start method adds a default mappercontext.setPath("/myApp");context.setDocBase("myApp");context.addChild(wrapper1);// context.addServletMapping(pattern, name);// note that we must use /myApp/Session, not just /Session// because the /myApp section must be the same as the path, so the cookie will// be sent back.context.addServletMapping("/myApp/Session", "Session");// add ContextConfig. This listener is important because it configures// StandardContext (sets configured to true), otherwise StandardContext// won't startLifecycleListener listener = new SimpleContextConfig();((Lifecycle) context).addLifecycleListener(listener);// here is our loaderLoader loader = new WebappLoader();// associate the loader with the Contextcontext.setLoader(loader);connector.setContainer(context);// add a ManagerManager manager = new StandardManager(); // highlight line.context.setManager(manager); // highlight line.try {connector.initialize();((Lifecycle) connector).start();((Lifecycle) context).start();// make the application wait until we press a key.System.in.read();((Lifecycle) context).stop();}catch (Exception e) {e.printStackTrace();}} } 【4.2】SimpleWrapperValve類 1)servlet示例可以調用javax.servlet.http.HttpServletRequest接口的getSession() 方法獲取Session對象:當調用 getSession() 方法時,request對象必須調用與Context 容器相關聯的Session管理器。Session管理器組件要么創建一個新的session 對象,要么返回一個已經存在的session對象。request對象為了能夠訪問Session管理器,他必須能夠訪問Context容器。 2)為了達到這個目的:在 SimpleWrapperValve類的invoke方法中,需要調用 org.apache.catalina.Request接口的setContext()方法,并傳入 Context實例; Attention)SimpleWrapperValve.invoke方法會調用被請求的servlet實例的service方法,所以,必須在調用service方法之前設置 Context實例; public void invoke(Request request, Response response, ValveContext valveContext)throws IOException, ServletException {SimpleWrapper wrapper = (SimpleWrapper) getContainer();ServletRequest sreq = request.getRequest();ServletResponse sres = response.getResponse();Servlet servlet = null;HttpServletRequest hreq = null;if (sreq instanceof HttpServletRequest)hreq = (HttpServletRequest) sreq;HttpServletResponse hres = null;if (sres instanceof HttpServletResponse)hres = (HttpServletResponse) sres;//-- new addition -----------------------------------Context context = (Context) wrapper.getParent(); // highlight line.request.setContext(context); // highlight line.//-------------------------------------// Allocate a servlet instance to process this requesttry {servlet = wrapper.allocate();if (hres!=null && hreq!=null) {servlet.service(hreq, hres);}else {servlet.service(sreq, sres);}}catch (ServletException e) {}} 補充)org.apache.catalina.connector.HttpRequestBase類的私有方法:doGetSession() 會調用Context接口的 getManager() 方法來獲取 Session管理器對象:(當獲取了Session管理器對象之后,就可以獲取到Session對象了,或直接創建一個新Session) Manager manager = null; if(contect != null)manager = context.getManager();
【4.3】打印結果 1)console info E:\bench-cluster\cloud-data-preprocess\HowTomcatWorks\src>java -cp .;lib/servlet.jar;lib/catalina_4_1_24.jar;lib/catalina-5.5.4.jar;lib/naming-common. jar;lib/commons-collections.jar;lib/naming-resources.jar;lib/;lib/catalina.jar;E:\bench-cluster\cloud-data-preprocess\HowTomcatWorks\webroot com/tomca t/chapter9/startup/Bootstrap HttpConnector Opening server socket on all host IP addresses HttpConnector[8080] Starting background thread WebappLoader[/myApp]: Deploying class repositories to work directory E:\bench-cluster\cloud-data-preprocess\HowTomcatWorks\src\work\_\_\myApp Starting Wrapper Session StandardManager[/myApp]: Seeding random number generator class java.security.SecureRandom StandardManager[/myApp]: Seeding of random number generator has been completed SessionServlet -- service SessionServlet -- service SessionServlet -- service SessionServlet -- service 2)訪問信息


創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎

總結

以上是生活随笔為你收集整理的tomcat(9)Session管理的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

91av在线免费看 | 正在播放五月婷婷狠狠干 | 韩国av一区二区 | 日韩网站免费观看 | 久久免费视频一区 | 五月婷婷天堂 | 日日夜夜天天操 | 日韩在线观看 | 午夜视频黄 | 色99在线| 日韩和的一区二在线 | 美女精品在线观看 | 97在线精品国自产拍中文 | 99成人精品 | 国产在线97| 国产一级二级三级在线观看 | 亚洲激情在线播放 | 激情网站免费观看 | 在线看片视频 | 欧美一级视频在线观看 | 精品国产aⅴ麻豆 | 婷婷99| 深爱开心激情 | 最新成人在线 | a视频免费在线观看 | 色网免费观看 | 在线观看www视频 | 麻豆传媒电影在线观看 | 中文字幕av有码 | 天天骚夜夜操 | 欧美一二三专区 | 全久久久久久久久久久电影 | 亚洲国产精品成人va在线观看 | 国产在线视频不卡 | 欧美日韩在线免费观看 | av 一区二区三区四区 | 久久精品高清视频 | 黄色免费视频在线观看 | 国产精品久久久久久69 | 九九久久影院 | 亚洲综合视频在线观看 | 国产综合在线视频 | 亚州精品在线视频 | 欧美俄罗斯性视频 | 亚洲国产精品第一区二区 | 中文字幕亚洲五码 | 久久久久欧美精品999 | 国产精品久久久久一区二区三区共 | 久久精品欧美一区二区三区麻豆 | 人人狠狠综合久久亚洲 | 国产精品第一 | 中文字幕 国产 一区 | 麻豆视频一区 | 欧美一级久久 | 国产精品久久二区 | 性色大片在线观看 | 国产高清在线 | 国产精品一区二区久久久 | 欧美电影在线观看 | 久久视频二区 | 人人超在线公开视频 | 超碰免费97 | 精品久久久久久亚洲 | 不卡国产在线 | 91丨九色丨国产丨porny精品 | 人人爽人人爽人人 | 91亚洲精品久久久蜜桃 | 在线视频欧美精品 | 欧美精品久久久久久久免费 | 在线中文字幕观看 | 99久久99久久精品国产片 | 亚洲天堂精品视频在线观看 | 99久久婷婷国产综合亚洲 | 91 中文字幕 | aaa日本高清在线播放免费观看 | 欧美日韩高清一区二区 国产亚洲免费看 | 国产成人av网 | 国产手机在线观看视频 | 美女一区网站 | 激情久久久| 午夜久久久久久久 | 少妇按摩av | 91av片| 中文av不卡| 国产成人精品一区二区在线观看 | 亚洲九九影院 | 日韩一二区在线 | a黄色 | 亚洲 中文 欧美 日韩vr 在线 | 日韩欧美在线不卡 | 中文字幕的 | 在线视频日韩 | 久久九九久久精品 | 国产精品久久免费看 | 在线日本看片免费人成视久网 | a天堂中文在线 | 91人人视频在线观看 | 91精品国产乱码久久 | 国产高清视频在线观看 | 色综合婷婷 | 天天骚夜夜操 | 久久久麻豆 | 蜜桃视频成人在线观看 | 天天干天天插伊人网 | 精品久久久精品 | av一区二区三区在线观看 | 91成人天堂久久成人 | 国产精品二区在线 | 天天躁日日躁狠狠躁av中文 | 欧美色就是色 | 黄色一集片 | 亚洲成人免费观看 | av不卡在线看 | 免费黄av | 欧美日韩伦理一区 | 麻豆视频免费在线播放 | 久久久久草 | 911久久香蕉国产线看观看 | 天天草天天草 | 久久激情综合网 | a级成人毛片 | 欧美-第1页-屁屁影院 | 久久无码av一区二区三区电影网 | 在线视频 影院 | 国产成人免费在线 | 韩国一区二区av | 69国产盗摄一区二区三区五区 | 国产中文字幕在线 | 午夜精品视频在线 | 美女精品网站 | 国产精品美 | 99九九99九九九视频精品 | 久久色在线播放 | 日日干夜夜草 | 91九色精品女同系列 | 99久热在线精品视频观看 | 免费观看成人网 | 欧美福利精品 | 午夜影院一级片 | 欧美成人a在线 | 97精品视频在线播放 | 五月综合婷 | 欧美日韩精品在线免费观看 | 国产成人一区二区啪在线观看 | 婷婷在线色 | 超碰.com| 一区二区三区久久精品 | 久久久久久久影院 | 九色91在线视频 | 精品久久一| 久99久视频 | 人人插人人射 | www日日| 国产成人精品国内自产拍免费看 | 亚洲黑丝少妇 | 最新av电影网址 | 在线看免费| 国产黄在线看 | 美女黄频在线观看 | 亚洲人成免费网站 | 天天综合色网 | 亚洲激情综合 | 在线观看国产www | 久久久伦理 | 免费的国产精品 | 在线91av| 日韩av一区二区在线播放 | 国产成人在线观看 | 色婷婷av一区二 | 黄色软件大全网站 | 国产亚洲久一区二区 | 亚洲综合成人婷婷小说 | 国产成人精品999 | 日韩免费观看一区二区 | 91麻豆精品一区二区三区 | 久久不射电影院 | 高清不卡一区二区三区 | 精品国产免费av | 97视频免费看 | 狠狠色伊人亚洲综合网站色 | 中文一区在线 | 婷婷六月色 | 国产精品18久久久久久久网站 | 91桃色在线播放 | 国产综合片| 国产精品一区二区免费 | 欧美色综合天天久久综合精品 | 亚洲综合视频网 | 中文字幕在线视频精品 | 欧美日韩观看 | 日韩在线视 | 四虎影视欧美 | 色综合久久88色综合天天 | 国内精品久久久久国产 | 日韩视频免费 | 亚洲激情 欧美激情 | 在线电影av | 91精品成人 | 99精品国产在热久久下载 | 91在线视频观看免费 | 国产精品综合在线 | 国产在线一区二区 | 免费在线黄色av | 成人 国产 在线 | 99精品视频网 | 久久精品一区二区三 | 婷婷干五月 | 丁香五月亚洲综合在线 | 国产午夜精品福利视频 | 九色精品在线 | 国产一区二区日本 | 亚洲精品成人免费 | 亚洲欧美日韩不卡 | 九九热精品视频在线观看 | 亚洲夜夜网 | 免费91麻豆精品国产自产在线观看 | 久久夜夜操 | 天堂av中文字幕 | 欧美在线视频二区 | 日韩视频免费观看高清完整版在线 | 国产精品私人影院 | 久久久久亚洲精品 | 欧洲激情综合 | 国产专区欧美专区 | 美女中文字幕 | 国产又粗又长的视频 | 91av网址| 成人网444ppp| 99视频在线观看一区三区 | 免费成人av网站 | 欧美性爽爽 | 久久综合五月婷婷 | 天天色天天综合 | 亚洲色图22p| 在线观看中文字幕av | 久久久久女人精品毛片 | 97干com| 国内精品久久久久久久久久久 | av直接看| 国产美女在线免费观看 | 夜夜看av | 欧美日韩亚洲一 | 欧美特一级 | 日韩极品视频在线观看 | 国产黄在线播放 | 国产精品视频永久免费播放 | 国产精品午夜久久 | 免费视频97| 欧美一区日韩精品 | 亚洲成aⅴ人片久久青草影院 | av在线播放观看 | 四虎影视成人精品 | 久久人人爽人人人人片 | 一区二区激情 | 丁香影院在线 | 国产亚洲永久域名 | 欧美成人h版电影 | 91亚洲综合 | 日日碰狠狠添天天爽超碰97久久 | 精品国产一区二区三区不卡 | 成人av一区二区三区 | 色在线免费 | 亚洲黄色区 | 亚洲黄色av网址 | 亚洲伦理一区 | 8090yy亚洲精品久久 | 狠狠色丁香婷婷综合久久片 | 午夜在线看片 | 国产午夜三级 | av在线网站观看 | 国产高清无av久久 | 婷婷在线免费视频 | 亚洲成a人片综合在线 | 国产精品1024 | 日韩美精品视频 | 午夜手机电影 | 亚洲乱亚洲乱亚洲 | 天天搞夜夜骑 | 波多野结衣电影一区二区 | 综合视频在线 | 五月天婷婷视频 | 亚洲永久精品在线 | av色影院 | 欧美成人91 | 久久激情久久 | 亚洲人成人在线 | 久久亚洲影院 | 欧美性大胆 | 欧美日韩大片在线观看 | 精品国产精品久久一区免费式 | av成人免费观看 | 久久精品网址 | 久久欧洲视频 | 亚洲精品国产精品久久99热 | 久久婷婷色综合 | 国产香蕉97碰碰碰视频在线观看 | 久久综合狠狠综合久久综合88 | 免费观看的黄色片 | 伊甸园永久入口www 99热 精品在线 | 久久午夜色播影院免费高清 | 中文区中文字幕免费看 | 久久精品国亚洲 | 亚洲午夜精品久久久久久久久 | 九九热视频在线 | 波多野结衣视频一区二区 | 国产在线精品国自产拍影院 | 国产精品免费一区二区 | 黄色app网站在线观看 | 日本一区二区三区视频在线播放 | 日本一区二区不卡高清 | 狠狠色丁香婷婷综合久小说久 | wwwww.国产 | 欧美激情综合五月色丁香小说 | 日韩精品久久一区二区 | 中文字幕一区二区三区在线观看 | 欧美日韩国产在线精品 | 成年人免费观看国产 | 日批视频在线观看免费 | 99精品久久久久久久 | 色六月婷婷 | 国产成人av | 国产精品视屏 | 国产在线观看 | 免费av的网站 | 欧美国产精品一区二区 | 韩国精品福利一区二区三区 | 亚洲精品美女久久久久网站 | av亚洲产国偷v产偷v自拍小说 | 狠狠久久婷婷 | 亚洲人毛片 | 麻豆国产精品va在线观看不卡 | 日三级在线 | 日韩久久激情 | 精品主播网红福利资源观看 | 国产精品一区二区三区在线看 | 99视频免费播放 | 色婷婷激情五月 | 国产精品久久久久久久毛片 | 狠狠狠色狠狠色综合 | 欧美综合在线视频 | 成年人黄色免费看 | 国产黄色精品网站 | 综合久久久久久久久 | av一区二区在线观看中文字幕 | 日日久视频 | 91精品久久久久久久久久久久久 | 欧美午夜精品久久久久久浪潮 | 丁香九月婷婷综合 | av一级片在线观看 | 久草在线视频网站 | 日本一区二区三区免费观看 | 日韩av中文字幕在线免费观看 | 国产成人av网址 | 成人禁用看黄a在线 | 久久精品网 | 久久综合射 | 欧美一级日韩免费不卡 | 草莓视频在线观看免费观看 | 色综合色综合久久综合频道88 | 一区免费观看 | 91九色综合 | 天天天天色综合 | 91综合久久一区二区 | 日韩大片在线看 | 综合色综合色 | 欧美有色| 国产精品区二区三区日本 | 婷婷在线精品视频 | 亚洲日本va中文字幕 | 精品久久国产精品 | 欧美精品一区二区蜜臀亚洲 | 国产色婷婷精品综合在线手机播放 | 黄色av电影| 亚洲一区二区视频在线 | 在线不卡中文字幕播放 | 国产在线探花 | 日韩理论片中文字幕 | 97电影在线 | 狠狠色婷婷丁香六月 | 国产色一区 | 人人舔人人插 | 激情xxxx| 国产精品久久久99 | 正在播放亚洲精品 | 91亚洲激情 | 成人va在线观看 | 色91av| 中文字幕av最新更新 | 亚洲黄网站 | 色wwww| 在线一二三区 | 欧美精品一区二区免费 | 国产手机在线精品 | 久草精品视频在线观看 | 日韩视频一区二区三区 | 久久毛片网 | 成人 国产 在线 | 五月丁色 | 国产精品久久久久久久久久新婚 | 国产精品爽爽久久久久久蜜臀 | 国产成人一区二区啪在线观看 | 97人人精品 | 中国一级片视频 | 精品国产美女在线 | 久久99九九99精品 | 久久国产美女视频 | 亚洲一级在线观看 | 天天操偷偷干 | 国产麻豆精品在线观看 | 成人av在线亚洲 | 99在线精品观看 | 啪啪肉肉污av国网站 | 日日干天天操 | 黄色av在| 欧美激情h| 国产一及片 | 日韩在线观看网站 | av线上免费看 | 日韩在线电影一区二区 | 亚洲黄色成人 | 在线看av的网址 | 三上悠亚一区二区在线观看 | 免费麻豆视频 | 免费无遮挡动漫网站 | 伊人黄色网 | 韩国精品在线观看 | 久久精品老司机 | 免费看国产一级片 | 精品国产精品国产偷麻豆 | 免费看国产曰批40分钟 | 嫩草91影院 | 久草视频在线免费看 | 久久久久久久久久网站 | 最新国产精品拍自在线播放 | 九九免费视频 | 天天色草| 欧美午夜寂寞影院 | 午夜三级福利 | 亚洲国产精品免费 | 成人va天堂 | 97av精品 | 日韩精品一区二区在线观看 | 国产精品不卡在线观看 | 九九av | 国产无限资源在线观看 | 在线av资源 | av资源网在线播放 | 久久精品99久久 | 午夜视频在线瓜伦 | 91精品色 | 亚洲成人家庭影院 | 国产一区观看 | 中文字幕一区二区在线观看 | 一区二区三区久久精品 | 不卡的av在线播放 | 97视频在线免费 | 美女视频一区二区 | 五月天av在线 | 亚洲永久精品在线观看 | 国产精品99久久久久久小说 | 九九九九热精品免费视频点播观看 | 亚洲最新视频在线播放 | 久久伊99综合婷婷久久伊 | 精品在线观看一区二区 | 亚洲狠狠婷婷综合久久久 | 999视频在线播放 | 国产精品一区二区视频 | av五月婷婷 | 九九免费在线观看 | 日本中文字幕在线电影 | 亚洲一区精品人人爽人人躁 | 日本精品久久久久中文字幕5 | 国产精品初高中精品久久 | 国产精品美女久久久久久免费 | 久草在线视频免赞 | 亚洲午夜精品福利 | 色婷婷啪啪免费在线电影观看 | 欧美在线视频一区二区 | 国产成人777777 | 在线观看精品视频 | 欧美日韩视频在线 | 青草视频在线免费 | 中文字幕高清在线播放 | 毛片精品免费在线观看 | 中文字幕在线观看三区 | 亚洲一区二区三区91 | 亚洲砖区区免费 | 日本激情视频中文字幕 | 亚洲精品免费在线观看 | 亚州性色| 亚洲 欧美 变态 国产 另类 | 中文字幕一区二区在线播放 | 亚洲精品久久久蜜臀下载官网 | 在线观看视频黄 | 国产黄av | 五月激情av | 色免费在线| 成人一区二区三区在线观看 | 天天综合天天做天天综合 | 国产专区精品视频 | 免费国产在线精品 | 国产美女久久 | 亚洲波多野结衣 | 日韩精品一区二区不卡 | 成人午夜电影网站 | 人人爽人人看 | 婷婷日日| 日韩免费成人av | av成人免费在线看 | 日本久久高清视频 | 欧美激情视频一区二区三区免费 | 麻豆国产网站入口 | 国产精品久久久久久久久久久不卡 | 在线观看免费中文字幕 | 99精品欧美一区二区蜜桃免费 | 国内小视频在线观看 | 国产精品一区二区三区久久久 | 亚洲视频一级 | x99av成人免费 | 毛片视频电影 | www欧美色 | 欧美福利片在线观看 | 国产日韩精品一区二区三区在线 | 久久综合天天 | 福利一区二区三区四区 | 一区二区av| 久久情侣偷拍 | av福利第一导航 | 日韩色综合网 | 亚洲更新最快 | 精品视频在线视频 | 久久久久色 | 成人高清在线观看 | 蜜臀av性久久久久蜜臀aⅴ涩爱 | 韩国精品一区二区三区六区色诱 | 香蕉网在线观看 | 99久久久免费视频 | 国产精品久久久久免费观看 | 日韩专区一区二区 | 久影院| 日日操操 | 久久黄色a级片 | 91在线看视频免费 | 免费久久久久久 | 少妇性xxx | 亚洲精品黄| 婷婷久久精品 | 婷婷 综合 色 | 中文字幕在线观看视频免费 | 九九热久久免费视频 | 丝袜美女在线 | 欧美一区二区三区在线播放 | 午夜精品一二三区 | 久久九九国产精品 | 亚洲国产综合在线 | 亚洲国产精品久久久久婷婷884 | 99精品国产高清在线观看 | 91精品导航 | 97av在线| 丁香综合 | 欧美一二区视频 | 欧美极度另类性三渗透 | 婷婷色综 | 正在播放国产精品 | 日韩激情片在线观看 | 国产伦精品一区二区三区免费 | 亚洲成人精品影院 | 久久中文精品视频 | 高清一区二区三区 | 在线成人免费 | 免费观看一级 | 在线观看免费观看在线91 | 91色亚洲 | 久久午夜电影 | 久草视频免费在线播放 | 久久视频免费 | 国产精品久久久久久久婷婷 | 黄色大片国产 | 涩av在线| 国精产品999国精产品岳 | 九九九国产 | 在线观看视频你懂得 | 久久久国产在线视频 | 一区二区三区国产精品 | 天天干天天拍天天操 | 九九久久久久久久久激情 | 黄色精品国产 | 亚洲电影久久 | 国产在线精品二区 | 日韩一区二区免费在线观看 | 一区二区三区在线影院 | 黄色字幕网 | 成人在线观看资源 | 久久精品亚洲一区二区三区观看模式 | 国内外成人免费在线视频 | 国内精品久久天天躁人人爽 | 在线观看视频中文字幕 | 在线观看免费 | 娇妻呻吟一区二区三区 | 亚洲干| 久久影院中文字幕 | 久久精品视频网站 | 久久综合偷偷噜噜噜色 | av黄网站 | 国产精品久久久精品 | 国产在线美女 | 国产精品1区2区在线观看 | 日韩精品一区二区三区免费观看视频 | 亚洲三级av | 98超碰人人 | ww亚洲ww亚在线观看 | 五月开心六月婷婷 | 国产精品高潮呻吟久久久久 | 国产精品aⅴ | 国产精品免费观看在线 | 色在线高清 | 久久久久久久久福利 | 国产精品99免视看9 国产精品毛片一区视频 | 成年人视频免费在线播放 | 9i看片成人免费看片 | 久久天堂亚洲 | 欧美日韩性视频在线 | 中文字幕丰满人伦在线 | 91爱在线 | 中文字幕免费播放 | 超碰在线最新网址 | 精品天堂av| 国产精品中文在线 | 久久久久久久久影视 | 免费美女久久99 | 国产真实精品久久二三区 | 女人高潮特级毛片 | 国产va饥渴难耐女保洁员在线观看 | 色99中文字幕 | 亚洲精品网址在线观看 | 精品亚洲成a人在线观看 | www免费网站在线观看 | 日本激情视频中文字幕 | 精品久久久久久久久亚洲 | 欧美精品久久天天躁 | 成人资源站 | 久久精品一区二区三区四区 | 欧美精品一区二区在线播放 | 国产日韩欧美视频 | 极品嫩模被强到高潮呻吟91 | 九九精品视频在线观看 | 婷婷色在线| 热99久久精品 | 国产毛片久久久 | 日韩一区二区三免费高清在线观看 | 亚洲va欧美va人人爽春色影视 | 九九99 | 国产偷国产偷亚洲清高 | 成人精品视频 | 国产精品毛片久久蜜 | 97人人模人人爽人人喊网 | 国产玖玖在线 | 91超级碰| 91在线视频 | 黄色av成人在线观看 | 欧美日韩高清在线观看 | 欧美a级成人淫片免费看 | 99免费在线视频 | www.狠狠插.com | 99精品久久久久久久 | 国产成人av网站 | 国产69精品久久99的直播节目 | 久草干 | 欧美成人理伦片 | 久久成人毛片 | 欧美日韩视频在线观看一区二区 | 欧美日韩精品在线免费观看 | 夜夜嗨av色一区二区不卡 | av在线中文 | 欧美91精品久久久久国产性生爱 | 丁香久久五月 | 亚洲黄色在线观看 | 久精品视频 | bayu135国产精品视频 | 国产婷婷 | 91丨九色丨国产在线观看 | 欧美成人va| 五月开心激情网 | 激情五月六月婷婷 | 韩国视频一区二区三区 | 国产精品久久久久一区 | 久久久久久久久久久福利 | 不卡的av电影在线观看 | 午夜av在线电影 | 天堂网av在线 | 国产日韩欧美在线观看视频 | 人人干狠狠干 | 国产一区自拍视频 | 91精品国产九九九久久久亚洲 | 国产午夜三级 | 黄色一级免费网站 | 精品无人国产偷自产在线 | 超碰个人在线 | 一 级 黄 色 片免费看的 | 99精品欧美一区二区三区黑人哦 | 悠悠av资源片 | 久久久高清免费视频 | 懂色av一区二区三区蜜臀 | 午夜精品一二三区 | 国产 欧美 日本 | www.色就是色 | av一区在线播放 | 少妇bbw揉bbb欧美 | 免费在线观看成人 | 一区二区不卡高清 | 精品国产免费人成在线观看 | 香蕉视频网站在线观看 | 99综合影院在线 | 国产精品免费观看视频 | 香蕉视频色 | a色视频| 网址你懂的在线观看 | 久久不射电影院 | 国产免费区 | 日韩精品一区二区久久 | 五月婷婷六月丁香 | 久久手机在线视频 | 久久天天躁| 日韩色一区二区三区 | 久草视频中文在线 | 日韩中文字幕在线不卡 | 99国产精品视频免费观看一公开 | 欧美久久久久久久久久 | 欧美午夜精品久久久久 | 精品视频资源站 | 亚洲影院色 | 婷婷深爱激情 | 国产网站在线免费观看 | 操操操夜夜操 | 五月天久久久 | 天天操天天色天天 | 91视频免费看网站 | av丝袜在线 | 夜夜视频欧洲 | 狠狠狠干狠狠 | 国产午夜三级一二三区 | 摸阴视频 | 久久黄色免费视频 | 国产精品一区二区三区久久 | 免费视频黄色 | 久久免费试看 | 狠狠色丁香久久婷婷综 | 在线观看免费黄视频 | 91精品久久久久久综合乱菊 | 日本视频网 | 91女人18片女毛片60分钟 | 狠狠狠色丁香婷婷综合久久88 | 精品在线播放 | 永久黄网站色视频免费观看w | 中文网丁香综合网 | 丁香久久综合 | 日韩高清一区在线 | 国产精品va最新国产精品视频 | 国产精品久久网站 | 波多野结衣久久精品 | 国产一区二区三区高清播放 | 99久久久国产精品免费99 | 成人久久国产 | 日本三级久久 | 韩国精品福利一区二区三区 | 亚洲国产精品一区二区尤物区 | 国产精品1000 | 国产 一区二区三区 在线 | 国产精品伦一区二区三区视频 | 涩涩成人在线 | 91经典在线 | 超碰人人超 | 国产一区免费 | 久久久久久蜜av免费网站 | 国产视频精品在线 | 人人干人人模 | 精品久久在线 | 日韩免费二区 | 久久久久高清 | 男女激情免费网站 | 国产在线播放一区 | 久久精品视频网站 | 福利视频精品 | 国产专区一 | 亚洲精品麻豆 | 高清国产午夜精品久久久久久 | 欧洲亚洲国产视频 | 国产精品精品久久久久久 | 黄网站色欧美视频 | 中文字幕在线高清 | 一本一道波多野毛片中文在线 | 国产无限资源在线观看 | 国产成人精品日本亚洲999 | 欧美在线观看视频一区二区三区 | 久久久人 | 亚洲理论片 | 日韩手机在线观看 | 国产中文字幕视频在线 | 久久久国产精品网站 | 久久在线精品 | 成人免费大片黄在线播放 | 久久精品视频99 | 性色va | 亚洲欧美激情精品一区二区 | 97干com| 日本韩国在线不卡 | 操操操干干干 | 香蕉视频国产在线观看 | 免费看三级 | 日日干精品 | free,性欧美 九九交易行官网 | 久久好看| 在线观看免费日韩 | 天天干天天想 | 国产精品久久久久久久久蜜臀 | 国产日韩精品一区二区三区 | 色妞久久福利网 | 色综合天天综合网国产成人网 | 蜜臀久久99精品久久久无需会员 | av免费网 | 国产精品午夜久久 | 久久精品99视频 | 97超碰成人 | 色欧美成人精品a∨在线观看 | 国产一级黄 | 成人精品一区二区三区中文字幕 | www黄com| 国产精品igao视频网入口 | 人人澡人摸人人添学生av | 蜜臀av免费一区二区三区 | 亚洲国产精品va在线 | 国产96精品 | 在线观看日韩中文字幕 | 久久亚洲成人网 | 公与妇乱理三级xxx 在线观看视频在线观看 | 国产裸体视频bbbbb | 精品国产中文字幕 | 精品在线观看一区二区三区 | 国产视频在线免费 | 日韩精品资源 | 久视频在线 | 国模一区二区三区四区 | 国产麻豆视频免费观看 | 9999在线视频 | 一区二区不卡视频在线观看 | 婷婷中文字幕在线观看 | 成人资源网 | 中文av一区二区 | 97人人人人| 99久久99| 韩国一区二区三区视频 | 欧美精品久久久久久久亚洲调教 | 男女激情片在线观看 | 亚洲精品美女久久17c | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 97操操| 视频国产 | 国产精品久久精品 | 天天插天天 | 免费看片成人 | 国产日韩欧美视频 | av激情五月| 在线国产小视频 | 国产日韩视频在线观看 | 久久久国产精品一区二区三区 | www黄色| av在线8 | 成人免费xyz网站 | 天天射射天天 | 国产精品久久久久久久久毛片 | 麻豆精品传媒视频 | 成人一级片免费看 | 天天综合五月天 | 精品人人人 | 四虎国产视频 | 人人爽人人爽人人片av | 色先锋资源网 | 精品福利视频在线 | 国产精品9999 | 久久69av| 98涩涩国产露脸精品国产网 | 91黄视频在线 | 97国产精品亚洲精品 | 亚州人成在线播放 | 国产精品久久久久一区二区国产 | 五月婷婷综合在线观看 | 免费久久久久久久 | www狠狠操 | 美女久久久久久久 | 日韩手机在线 | 四虎在线观看视频 | 国产精品不卡在线观看 | 在线观看亚洲专区 | 久久综合色天天久久综合图片 | 成人超碰在线 | 亚洲综合色av | 国产日产av| 中文字幕资源在线 | 狠狠躁夜夜躁人人爽视频 | 日日干 天天干 | 亚洲欧美国产日韩在线观看 | 人人澡人人爽 | 国产又黄又硬又爽 | 天天操婷婷 | 中文字幕在线观看不卡 | 欧美激情精品久久久久久免费 | 久久综合九色99 | 99免费| 国产午夜精品在线 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 五月婷婷欧美 | 久久综合精品国产一区二区三区 | 97精品视频在线 | 欧美性色19p | 久草免费在线 | 97精产国品一二三产区在线 | 久久96国产精品久久99软件 | 国产伦精品一区二区三区高清 | 亚洲国产欧美在线看片xxoo | 99久视频| 91视频在线免费下载 | 亚洲97在线 | 麻豆一二三精选视频 | 成人av中文字幕在线观看 | 天天曰天天 | 青青草在久久免费久久免费 | 国产91在线免费视频 | 91香蕉视频在线下载 | 97超碰资源站 | 日本精品视频在线播放 | 久久久福利影院 | 美女激情影院 | 人人干干人人 | 久久精品男人的天堂 | 日本中文字幕电影在线免费观看 | 欧美日韩国产一二三区 | 免费在线观看视频一区 | 国产精品麻豆99久久久久久 | 精品国产久 | 久久精品三 | 精品久久久久久久久久 | 天天干天天在线 | 69视频永久免费观看 | 最新午夜电影 | 91久久国产露脸精品国产闺蜜 | 日本在线观看黄色 | 免费观看成年人视频 | 少妇视频一区 | 91在线看| 国产精品第54页 | 91豆花在线观看 | 一区二区中文字幕在线播放 | 五月婷婷中文网 | 免费看国产视频 | 国产精品美女999 | 精品国产一区在线观看 | 久久露脸国产精品 | 国产精品不卡av | 美女免费网站 | 日韩大片在线看 | 精品久久1 | 国产91对白在线播 | 涩涩在线 | 999久久久久久久久久久 | 欧美精品乱码久久久久 | 91成年人网站 | 韩日三级在线 | www日韩在线观看 | 国产裸体视频bbbbb | 国产中出在线观看 | 亚洲电影第一页av | 91一区在线观看 | 国产自产高清不卡 | 日本最新中文字幕 | 伊人首页| 久色婷婷| 视频在线观看91 | 亚洲最新在线 | 91热精品| 一区二区精品在线 | 日韩三级视频 | 免费看片网址 | 国产丝袜制服在线 | 日韩一区二区在线免费观看 | 黄色h在线观看 | 久草在线这里只有精品 | 五月婷婷综合在线视频 | 不卡电影免费在线播放一区 | 亚洲视频在线观看网站 | 伊人资源视频在线 | 中文字幕影视 | 国产精品免费在线 | 日韩一区二区免费播放 | 91精品国产成人 | 婷婷视频在线播放 | 公与妇乱理三级xxx 在线观看视频在线观看 | 国产裸体视频网站 | 天天碰天天操视频 |