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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

代理模式详解(包含原理详解)

發(fā)布時間:2024/4/17 编程问答 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 代理模式详解(包含原理详解) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

http://www.cnblogs.com/zuoxiaolong/p/pattern3.html

作者:zuoxiaolong8810(左瀟龍),轉(zhuǎn)載請注明出處,特別說明:本博文來自博主原博客,為保證新博客中博文的完整性,特復(fù)制到此留存,如需轉(zhuǎn)載請注明新博客地址即可。

? ? ? ? ? ? ? ?我特意將本系列改了下名字,原名是《設(shè)計模式學(xué)習之路》,原因是因為之前寫過一篇《spring源碼學(xué)習之路》,但是我感覺本次寫二十三種設(shè)計模式,更多的還是分享給各位自己的理解,所以感覺學(xué)習之路有點不合適,另外,從本章開始,正式啟用本人稱呼,LZ。

? ? ? ? ? ? ? ?好了,廢話至此,本章接著討論第二種要介紹的設(shè)計模式,代理模式。

? ? ? ? ? ? ? ?LZ不希望寫的東西與網(wǎng)絡(luò)上的資料千篇一律,所以這一系列不會像很多典型文章一章,只是列出這個模式的定義以及一堆適用的情況,然后就是一堆這個模式的各個角色,對于這種羅列LZ并不反對,相比之下會比較清晰,但是如果脫離了實際,就會導(dǎo)致看的人特別是初學(xué)者覺得設(shè)計模式很陌生很遙遠。

? ? ? ? ? ? ? ?LZ并不反對這種教學(xué)式的標準模式,但說實話,LZ本人看這種帖子從來都感覺收獲不大,看一遍看一遍,到現(xiàn)在都沒記住那些各個適用的情況與一堆亂七八糟的角色。

? ? ? ? ? ? ? ?所以LZ探討代理模式,不會再按這個步驟進行,而是跟著自己的思維進行。

? ? ? ? ? ? ? ?首先代理模式,可以分為兩種,一種是靜態(tài)代理,一種是動態(tài)代理。

? ? ? ? ? ? ? ?兩種代理從虛擬機加載類的角度來講,本質(zhì)上都是一樣的,都是在原有類的行為基礎(chǔ)上,加入一些多出的行為,甚至完全替換原有的行為。

? ? ? ? ? ? ? ?靜態(tài)代理采用的方式就是我們手動的將這些行為換進去,然后讓編譯器幫我們編譯,同時也就將字節(jié)碼在原有類的基礎(chǔ)上加入一些其他的東西或者替換原有的東西,產(chǎn)生一個新的與原有類接口相同卻行為不同的類型。

? ? ? ? ? ? ? ?說歸說,我們來真實的去試驗一下,實驗的話需要找一個示例,就拿我們的數(shù)據(jù)庫連接來做例子吧。

? ? ? ? ? ? ? ?我們都知道,數(shù)據(jù)庫連接是很珍貴的資源,頻繁的開關(guān)數(shù)據(jù)庫連接是非常浪費服務(wù)器的CPU資源以及內(nèi)存的,所以我們一般都是使用數(shù)據(jù)庫連接池來解決這一問題,即創(chuàng)造一堆等待被使用的連接,等到用的時候就從池里取一個,不用了再放回去,數(shù)據(jù)庫連接在整個應(yīng)用啟動期間,幾乎是不關(guān)閉的,除非是超過了最大閑置時間。

? ? ? ? ? ? ? ?但是在程序員編寫程序的時候,會經(jīng)常使用connection.close()這樣的方法,去關(guān)閉數(shù)據(jù)庫連接,而且這樣做是對的,所以你并不能告訴程序員們說,你們使用連接都不要關(guān)了,去調(diào)用一個其他的類似歸還給連接池的方法吧。這是不符合程序員的編程思維的,也很勉強,而且具有風險性,因為程序員會忘的。

? ? ? ? ? ? ? ?解決這一問題的辦法就是使用代理模式,因為代理模式可以替代原有類的行為,所以我們要做的就是替換掉connection的close行為。

? ? ? ? ? ? ? ?下面是connection接口原有的樣子,我去掉了很多方法,因為都類似,全貼上來占地方。

import java.sql.SQLException; import java.sql.Statement; import java.sql.Wrapper;public interface Connection extends Wrapper {Statement createStatement() throws SQLException;void close() throws SQLException;}

? ? ? ? ? ? ? 這里只貼了兩個方法,但是我們代理的精髓只要兩個方法就能掌握,下面使用靜態(tài)代理,采用靜態(tài)代理我們通常會使用組合的方式,為了保持對程序猿是透明的,我們實現(xiàn)Connection這個接口,?如下所示。

import java.sql.SQLException; import java.sql.Statement;public class ConnectionProxy implements Connection{private Connection connection;public ConnectionProxy(Connection connection) {super();this.connection = connection;}public Statement createStatement() throws SQLException{return connection.createStatement();}public void close() throws SQLException{System.out.println("不真正關(guān)閉連接,歸還給連接池");}}

? ? ? ? ? ? ? ? 我們在構(gòu)造方法中讓調(diào)用者強行傳入一個原有的連接,接下來我們將我們不關(guān)心的方法,交給真正的Connection接口去處理,就像createStatement方法一樣,而我們將真正關(guān)心的close方法用我們自己希望的方式去進行。

? ? ? ? ? ? ? ? 此處為了更形象,LZ給出一個本人寫的非常簡單的連接池,意圖在于表明實現(xiàn)的思路。下面我們來看一下連接池的變化,在里面注明了變化點。

import java.sql.Connection; import java.sql.DriverManager; import java.sql.SQLException; import java.util.LinkedList;public class DataSource {private static LinkedList<Connection> connectionList = new LinkedList<Connection>();static{try {Class.forName("com.mysql.jdbc.Driver");} catch (ClassNotFoundException e) {e.printStackTrace();}}private static Connection createNewConnection() throws SQLException{return DriverManager.getConnection("url","username", "password");}private DataSource(){if (connectionList == null || connectionList.size() == 0) {for (int i = 0; i < 10; i++) {try {connectionList.add(createNewConnection());} catch (SQLException e) {e.printStackTrace();}}}}public Connection getConnection() throws Exception{if (connectionList.size() > 0) {//return connectionList.remove(); 這是原有的方式,直接返回連接,這樣可能會被程序員把連接給關(guān)閉掉//下面是使用代理的方式,程序員再調(diào)用close時,就會歸還到連接池return new ConnectionProxy(connectionList.remove());}return null;}public void recoveryConnection(Connection connection){connectionList.add(connection);}public static DataSource getInstance(){return DataSourceInstance.dataSource;}private static class DataSourceInstance{private static DataSource dataSource = new DataSource();}}

? ? ? ? ? ? ? ?連接池我們把它做成單例,所以假設(shè)是上述連接池的話,我們代理中的close方法可以再具體化一點,就像下面這樣,用歸還給連接池的動作取代關(guān)閉連接的動作。

public void close() throws SQLException{DataSource.getInstance().recoveryConnection(connection);}

? ? ? ? ? ? ? ?好了,這下我們的連接池返回的連接全是代理,就算程序員調(diào)用了close方法也只會歸還給連接池了。

? ? ? ? ? ? ? ?我們使用代理模式解決了上述問題,從靜態(tài)代理的使用上來看,我們一般是這么做的。

? ? ? ? ? ? ? ?1,代理類一般要持有一個被代理的對象的引用。

? ? ? ? ? ? ? ?2,對于我們不關(guān)心的方法,全部委托給被代理的對象處理。

? ? ? ? ? ? ? ?3,自己處理我們關(guān)心的方法。

? ? ? ? ? ? ? ?這種代理是死的,不會在運行時動態(tài)創(chuàng)建,因為我們相當于在編譯期,也就是你按下CTRL+S的那一刻,就給被代理的對象生成了一個不可動態(tài)改變的代理類。

? ? ? ? ? ? ? ?靜態(tài)代理對于這種,被代理的對象很固定,我們只需要去代理一個類或者若干固定的類,數(shù)量不是太多的時候,可以使用,而且其實效果比動態(tài)代理更好,因為動態(tài)代理就是在運行期間動態(tài)生成代理類,所以需要消耗的時間會更久一點。就像上述的情況,其實就比較適合使用靜態(tài)代理。

? ? ? ? ? ? ? ?下面介紹下動態(tài)代理,動態(tài)代理是JDK自帶的功能,它需要你去實現(xiàn)一個InvocationHandler接口,并且調(diào)用Proxy的靜態(tài)方法去產(chǎn)生代理類。

? ? ? ? ? ? ? ?接下來我們依然使用上面的示例,但是這次該用動態(tài)代理處理,我們來試一下看如何做。

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.sql.Connection;public class ConnectionProxy implements InvocationHandler{private Connection connection;public ConnectionProxy(Connection connection) {super();this.connection = connection;}public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//這里判斷是Connection接口的close方法的話if (Connection.class.isAssignableFrom(proxy.getClass()) && method.getName().equals("close")) {//我們不執(zhí)行真正的close方法//method.invoke(connection, args);//將連接歸還連接池 DataSource.getInstance().recoveryConnection(connection);return null;}else {return method.invoke(connection, args);}}public Connection getConnectionProxy(){return (Connection) Proxy.newProxyInstance(getClass().getClassLoader(), new Class[]{Connection.class}, this);}}

? ? ? ? ? ? ? ?上面是我們針對connection寫的動態(tài)代理,InvocationHandler接口只有一個invoke方法需要實現(xiàn),這個方法是用來在生成的代理類用回調(diào)使用的,關(guān)于動態(tài)代理的原理一會做詳細的分析,這里我們先只關(guān)注用法。很顯然,動態(tài)代理是將每個方法的具體執(zhí)行過程交給了我們在invoke方法里處理。而具體的使用方法,我們只需要創(chuàng)造一個ConnectionProxy的實例,并且將調(diào)用getConnectionProxy方法的返回結(jié)果作為數(shù)據(jù)庫連接池返回的連接就可以了。

? ? ? ? ? ? ? ?上述便是我們針對connection做動態(tài)代理的方式,但是我們從中得不到任何好處,除了能少寫點代碼以外,因為這個動態(tài)代理還是只能代理Connection這一個接口,如果我們寫出這種動態(tài)代理的方式的話,說明我們應(yīng)該使用靜態(tài)代理處理這個問題,因為它代表我們其實只希望代理一個類就好。從重構(gòu)的角度來說,其實更簡單點,那就是在你發(fā)現(xiàn)你使用靜態(tài)代理的時候,需要寫一大堆重復(fù)代碼的時候,就請改用動態(tài)代理試試吧。

? ? ? ? ? ? ? ?通常情況下,動態(tài)代理的使用是為了解決這樣一種問題,就是我們需要代理一系列類的某一些方法,最典型的應(yīng)用就是我們前段時間討論過的springAOP,我們需要創(chuàng)造出一批代理類,切入到一系列類當中的某一些方法中。下面給出一個經(jīng)常使用的動態(tài)代理方式。

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy;public class DynamicProxy implements InvocationHandler{private Object source;public DynamicProxy(Object source) {super();this.source = source;}public void before(){System.out.println("在方法前做一些事,比如打開事務(wù)");}public void after(){System.out.println("在方法返回前做一些事,比如提交事務(wù)");}public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//假設(shè)我們切入toString方法,其他其實也是類似的,一般我們這里大部分是針對特定的方法做事情的,通常不會對類的全部方法切入//比如我們常用的事務(wù)管理器,我們通常配置的就是對save,update,delete等方法才打開事務(wù)if (method.getName().equals("toString")) {before();}Object result = method.invoke(source, args);if (method.getName().equals("toString")) {after();}return result;}public Object getProxy(){return Proxy.newProxyInstance(getClass().getClassLoader(), source.getClass().getInterfaces(), this);}}

? ? ? ? ? ? ? ? 上述我做了一些注釋,其實已經(jīng)說明一些問題,這個代理類的作用是可以代理任何類,因為它被傳入的對象是Object,而不再是具體的類,比如剛才的Connection,這些產(chǎn)生的代理類在調(diào)用toString方法時會被插入before方法和after方法。

? ? ? ? ? ? ? ? 動態(tài)代理有一個強制性要求,就是被代理的類必須實現(xiàn)了某一個接口,或者本身就是接口,就像我們的Connection。

? ? ? ? ? ? ? ? 道理其實很簡單,這是因為動態(tài)代理生成的代理類是繼承Proxy類的,并且會實現(xiàn)被你傳入newProxyInstance方法的所有接口,所以我們可以將生成的代理強轉(zhuǎn)為任意一個代理的接口或者Proxy去使用,但是Proxy里面幾乎全是靜態(tài)方法,沒有實例方法,所以轉(zhuǎn)換成Proxy意義不大,幾乎沒什么用。假設(shè)我們的類沒有實現(xiàn)任何接口,那么就意味著你只能將生成的代理類轉(zhuǎn)換成Proxy,那么就算生成了,其實也沒什么用,而且就算你傳入了接口,可以強轉(zhuǎn),你也用不了這個沒有實現(xiàn)你傳入接口的這個類的方法。

? ? ? ? ? ? ? ?你可能會說,假設(shè)有個接口A,那我將接口A傳給newProxyInstance方法,并代理一個沒實現(xiàn)接口A的類B,但類B與接口A有一樣的方法可以嗎?

? ? ? ? ? ? ? ?答案是可以的,并且JDK的動態(tài)代理只認你傳入的接口,只要你傳入,你就可以強轉(zhuǎn)成這個接口,這個一會解釋,但是你無法在invoke方法里調(diào)用method.invoke方法,也就是說,你只能全部替換A接口的方法,而不能使用類B中原有與接口A方法描述相同的方法,這是因為invoke中傳入的Method的class信息是接口A,而類B因為沒實現(xiàn)接口A,所以無法執(zhí)行傳入的Method,會拋出非法參數(shù)異常。

? ? ? ? ? ? ? ?下面我貼出測試代碼,各位可以自己試一下,具體為何會這樣是在后面解釋的,這里不再多做解釋。

? ? ? ? ? ? ? 先是一個普通接口。

public interface TestInterface {void method1();void method2();void method3(); }

? ? ? ? ? ? 然后是一個類,和接口一模一樣的方法,但是就是沒實現(xiàn)這個接口。

public class TestClass{public void method1() {System.out.println("TestClass.method1");}public void method2() {System.out.println("TestClass.method2");}public void method3() {System.out.println("TestClass.method3");}}

? ? ? ? ? ? ? ?下面是測試類。

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy;public class DynamicProxy implements InvocationHandler{private Object source;public DynamicProxy(Object source) {super();this.source = source;}public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("接口的方法全部變成這樣了");//這里source是TestClass,但是我們不能使用反射調(diào)用它的方法,像下面這樣,放開這一行會拋異常//return method.invoke(source, args);return null;}public static void main(String[] args) {//只要你傳入就可以強轉(zhuǎn)成功TestInterface object = (TestInterface) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{TestInterface.class}, new DynamicProxy(new TestClass()));object.method1();object.method2();object.method3();} }

? ? ? ? ? ? ? ?上面我們運行就會發(fā)現(xiàn)接口的方法全部都只能輸出一個很2的字符串了。如果是要繼續(xù)使用TestClass的方法也不是不行,只要你確認你傳入的類包括了所有你傳入的接口的方法,只是沒實現(xiàn)這些接口而已,那么你可以在invoke中這樣使用。

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("before");Method sourceMethod = source.getClass().getDeclaredMethod(method.getName(), method.getParameterTypes());sourceMethod.setAccessible(true);Object result = sourceMethod.invoke(source, args);System.out.println("after");return result;}

? ? ? ? ? ? ? ? 這就與你實現(xiàn)接口的表現(xiàn)行為一致了,但是我們本來就只需要一句method.invoke就可以了,就因為沒實現(xiàn)接口就要多寫兩行,所以這種突破JDK動態(tài)代理必須實現(xiàn)接口的行為就有點畫蛇添足了。因為你本來就實現(xiàn)了該接口的方法,只是差了那一句implements而已。

? ? ? ? ? ? ? ?上面寫這個例子只是為了解釋LZ當初的疑惑,因為LZ曾一度認為不實現(xiàn)接口就不能使用動態(tài)代理,現(xiàn)在想想那時候LZ有點2,呵呵。

? ? ? ? ? ? ? ?好了,從現(xiàn)在開始,我們開始詳細講解動態(tài)代理的原理,這算是進階篇,如果是新手的話,可以跳過下面的內(nèi)容,因為現(xiàn)在還沒必要知道這些,而且弄不好會越看越蒙,不過僅僅是LZ個人建議,你要有耐心,完全可以繼續(xù)看下去。

? ? ? ? ? ? ? ?接下來我們結(jié)合源碼去看一下,代理類是如何產(chǎn)生的,首先當然就是要進入Proxy的newProxyInstance方法,這里是產(chǎn)生代理的入口,源碼如下。

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)throws IllegalArgumentException{if (h == null) {throw new NullPointerException();}/** Look up or generate the designated proxy class.*/Class cl = getProxyClass(loader, interfaces);/** Invoke its constructor with the designated invocation handler.*/try {Constructor cons = cl.getConstructor(constructorParams);return (Object) cons.newInstance(new Object[] { h });} catch (NoSuchMethodException e) {throw new InternalError(e.toString());} catch (IllegalAccessException e) {throw new InternalError(e.toString());} catch (InstantiationException e) {throw new InternalError(e.toString());} catch (InvocationTargetException e) {throw new InternalError(e.toString());}}

? ? ? ? ? ? ? ?這個方法其實很簡單,首先獲取了代理類的運行時Class引用,然后調(diào)用了這個Class中的構(gòu)造方法,這個構(gòu)造方法只有一個參數(shù),正是InvocationHandler接口,由此產(chǎn)生了一個代理類的實例。那么關(guān)鍵的地方就在于如何獲取的代理類運行時的class信息的呢?我們進入getProxyClass方法看一下。為了方便起見,我直接加注釋,這個方法需要解釋的地方比較多。

public static Class<?> getProxyClass(ClassLoader loader,Class<?>... interfaces) throws IllegalArgumentException {//如果傳入的接口長度大于65535就拋出異常,我去你妹。。。if (interfaces.length > 65535) {throw new IllegalArgumentException("interface limit exceeded");}Class proxyClass = null;/* collect interface names to use as key for proxy class cache */String[] interfaceNames = new String[interfaces.length];Set interfaceSet = new HashSet(); // for detecting duplicatesfor (int i = 0; i < interfaces.length; i++) {/** Verify that the class loader resolves the name of this interface* to the same Class object.*/String interfaceName = interfaces[i].getName();Class interfaceClass = null;try {//加載每一個接口的運行時Class信息interfaceClass = Class.forName(interfaceName, false, loader);} catch (ClassNotFoundException e) {}//如果采用你傳入的類加載器載入的Class和你傳入的Class不相等則拋出異常if (interfaceClass != interfaces[i]) {throw new IllegalArgumentException(interfaces[i]+ " is not visible from class loader");}//如果你傳入的不是接口拋出異常/** Verify that the Class object actually represents an interface.*/if (!interfaceClass.isInterface()) {throw new IllegalArgumentException(interfaceClass.getName()+ " is not an interface");}//如果你傳入的接口重復(fù)拋異常/** Verify that this interface is not a duplicate.*/if (interfaceSet.contains(interfaceClass)) {throw new IllegalArgumentException("repeated interface: "+ interfaceClass.getName());}interfaceSet.add(interfaceClass);interfaceNames[i] = interfaceName;}/** Using string representations of the proxy interfaces as keys in the* proxy class cache (instead of their Class objects) is sufficient* because we require the proxy interfaces to be resolvable by name* through the supplied class loader, and it has the advantage that* using a string representation of a class makes for an implicit weak* reference to the class.*/Object key = Arrays.asList(interfaceNames);/** Find or create the proxy class cache for the class loader.*/Map cache;synchronized (loaderToCache) {//這個是為了存儲每一個類加載器所載入過的代理接口的代理類cache = (Map) loaderToCache.get(loader);if (cache == null) {cache = new HashMap();loaderToCache.put(loader, cache);}/** This mapping will remain valid for the duration of this method,* without further synchronization, because the mapping will only be* removed if the class loader becomes unreachable.*/}/** Look up the list of interfaces in the proxy class cache using the* key. This lookup will result in one of three possible kinds of* values: null, if there is currently no proxy class for the list of* interfaces in the class loader, the pendingGenerationMarker object,* if a proxy class for the list of interfaces is currently being* generated, or a weak reference to a Class object, if a proxy class* for the list of interfaces has already been generated.*/synchronized (cache) {/** Note that we need not worry about reaping the cache for entries* with cleared weak references because if a proxy class has been* garbage collected, its class loader will have been garbage* collected as well, so the entire cache will be reaped from the* loaderToCache map.*/do {//檢查是否有生成好的代理Object value = cache.get(key);if (value instanceof Reference) {proxyClass = (Class) ((Reference) value).get();}//有的話直接返回if (proxyClass != null) {// proxy class already generated: return itreturn proxyClass;//否則看一下這個代理類是不是正在構(gòu)造中,是的話就在cache對象上等待} else if (value == pendingGenerationMarker) {// proxy class being generated: wait for ittry {cache.wait();} catch (InterruptedException e) {/** The class generation that we are waiting for should* take a small, bounded time, so we can safely ignore* thread interrupts here.*/}continue;//如果沒有現(xiàn)成的,也沒有創(chuàng)造中的,那就開始創(chuàng)造代理類} else {/** No proxy class for this list of interfaces has been* generated or is being generated, so we will go and* generate it now. Mark it as pending generation.*///將當前代理類置為正在構(gòu)造中,并直接退出循環(huán) cache.put(key, pendingGenerationMarker);break;}} while (true);}try {String proxyPkg = null; // package to define proxy class in//這一段是看你傳入的接口中有沒有不是public的接口,如果有,這些接口必須全部在一個包里定義的,否則拋異常/** Record the package of a non-public proxy interface so that the* proxy class will be defined in the same package. Verify that all* non-public proxy interfaces are in the same package.*/for (int i = 0; i < interfaces.length; i++) {int flags = interfaces[i].getModifiers();if (!Modifier.isPublic(flags)) {String name = interfaces[i].getName();int n = name.lastIndexOf('.');String pkg = ((n == -1) ? "" : name.substring(0, n + 1));if (proxyPkg == null) {proxyPkg = pkg;} else if (!pkg.equals(proxyPkg)) {throw new IllegalArgumentException("non-public interfaces from different packages");}}}if (proxyPkg == null) { // if no non-public proxy interfaces,proxyPkg = ""; // use the unnamed package }{/** Choose a name for the proxy class to generate.*/long num;synchronized (nextUniqueNumberLock) {num = nextUniqueNumber++;}//生成一個隨機代理類名String proxyName = proxyPkg + proxyClassNamePrefix + num;/** Verify that the class loader hasn't already defined a class* with the chosen name.*///這一句就是重中之重了,生成代理類的class文件,這就是JDK動態(tài)代理的原理了,通過動態(tài)生成class文件來產(chǎn)生的代理類//這個generateProxyClass方法下面會著重介紹/** Generate the specified proxy class.*/byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces);try {//得到class文件二進制流后,直接載入代理類proxyClass = defineClass0(loader, proxyName,proxyClassFile, 0, proxyClassFile.length);} catch (ClassFormatError e) {/** A ClassFormatError here means that (barring bugs in the* proxy class generation code) there was some other invalid* aspect of the arguments supplied to the proxy class* creation (such as virtual machine limitations exceeded).*/throw new IllegalArgumentException(e.toString());}}//proxyClasses這個Map是為了來判斷是不是代理的Class// add to set of all generated proxy classes, for isProxyClassproxyClasses.put(proxyClass, null);} finally {/** We must clean up the "pending generation" state of the proxy* class cache entry somehow. If a proxy class was successfully* generated, store it in the cache (with a weak reference);* otherwise, remove the reserved entry. In all cases, notify all* waiters on reserved entries in this cache.*/synchronized (cache) {if (proxyClass != null) {//最終將生成的代理用弱引用包裝起來放到cache當中cache.put(key, new WeakReference(proxyClass));} else {//如果代理類是空則移除代理的接口所代表的key值 cache.remove(key);}//通知正在等待在cache對象上的線程,告訴他們可以繼續(xù)了,代理Class加載完畢了 cache.notifyAll();}}return proxyClass;}

? ? ? ? ? ? ? ? 上面基本上已經(jīng)解釋的很清楚了,下面就是去看一下byte[] proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces)這句話是如何處理的,也就是如何得到的代理類的class文件的,我們進去源碼看一下,我依然會加上注釋。

public static byte[] generateProxyClass(String paramString, Class[] paramArrayOfClass) {//新建一個ProxyGenerator實例,傳入類名和接口數(shù)組ProxyGenerator localProxyGenerator = new ProxyGenerator(paramString, paramArrayOfClass);//這個才是真正生成class文件的地方final byte[] arrayOfByte = localProxyGenerator.generateClassFile();//看保存生成文件的標志是否為真,如果是就將class文件生成到本地,生成時要檢查權(quán)限if (saveGeneratedFiles) {AccessController.doPrivileged(new PrivilegedAction() {public Object run() {try {FileOutputStream localFileOutputStream = new FileOutputStream( ProxyGenerator.dotToSlash(this.val$name) + ".class");localFileOutputStream.write(arrayOfByte);localFileOutputStream.close();return null;} catch (IOException localIOException) {throw new InternalError( "I/O exception saving generated file: " + localIOException);}}});}return arrayOfByte;}

? ? ? ? ? ? ? ? 我們繼續(xù)跟蹤到localProxyGenerator.generateClassFile()這一句當中,依然會加上注釋。

private byte[] generateClassFile() {//addProxyMethod方法,就是將方法都加入到一個列表中,并與對應(yīng)的class對應(yīng)起來//這里給Object對應(yīng)了三個方法hashCode,toString和equalsaddProxyMethod(hashCodeMethod, Object.class);addProxyMethod(equalsMethod, Object.class);addProxyMethod(toStringMethod, Object.class);//同樣將接口與接口下的方法對應(yīng)起來for (int i = 0; i < this.interfaces.length; i++) {localObject1 = this.interfaces[i].getMethods();for (int k = 0; k < localObject1.length; k++) {addProxyMethod(localObject1[k], this.interfaces[i]);}}//檢查所有代理方法的返回類型for (Iterator localIterator1 = this.proxyMethods.values().iterator(); localIterator1.hasNext();) {localObject1 = (List) localIterator1.next();checkReturnTypes((List) localObject1);}Object localObject2;try {//方法中加入構(gòu)造方法,這個構(gòu)造方法只有一個,就是一個帶有InvocationHandler接口的構(gòu)造方法//這個才是真正給class文件,也就是代理類加入方法了,不過還沒真正處理,只是先加進來等待循環(huán),構(gòu)造方法在class文件中的名稱描述是<init>this.methods.add(generateConstructor());//循環(huán)代理方法for (localIterator1 = this.proxyMethods.values().iterator(); localIterator1.hasNext();) {localObject1 = (List) localIterator1.next();for (localIterator2 = ((List) localObject1).iterator(); localIterator2.hasNext();) {localObject2 = (ProxyMethod) localIterator2.next();//給每一個代理方法加一個Method類型的屬性,數(shù)字10是class文件的標識符,代表這些屬性都是private static的this.fields.add(new FieldInfo(((ProxyMethod) localObject2).methodFieldName,"Ljava/lang/reflect/Method;", 10));//將每一個代理方法都加到代理類的方法中this.methods.add(((ProxyMethod) localObject2).generateMethod());}}Iterator localIterator2;//加入一個靜態(tài)初始化塊,將每一個屬性都初始化,這里靜態(tài)代碼塊也叫類構(gòu)造方法,其實就是名稱為<clinit>的方法,所以加到方法列表this.methods.add(generateStaticInitializer());} catch (IOException localIOException1) {throw new InternalError("unexpected I/O Exception");}//方法和屬性個數(shù)都不能超過65535,包括剛才的接口個數(shù)也是這樣,//這是因為在class文件中,這些個數(shù)都是用4位16進制表示的,所以最大值是2的16次方-1if (this.methods.size() > 65535) {throw new IllegalArgumentException("method limit exceeded");}if (this.fields.size() > 65535) {throw new IllegalArgumentException("field limit exceeded");}//這里是將類名中的.轉(zhuǎn)成成斜線為了寫入class文件。this.cp.getClass(dotToSlash(this.className));this.cp.getClass("java/lang/reflect/Proxy");for (int j = 0; j < this.interfaces.length; j++) {this.cp.getClass(dotToSlash(this.interfaces[j].getName()));}this.cp.setReadOnly();//這里開始真正的寫class文件ByteArrayOutputStream localByteArrayOutputStream = new ByteArrayOutputStream();Object localObject1 = new DataOutputStream(localByteArrayOutputStream);try {//寫入class文件的標識號,標識這是一個class文件((DataOutputStream) localObject1).writeInt(-889275714);//次版本號0((DataOutputStream) localObject1).writeShort(0);//主版本號,49代表的是JDK1.5((DataOutputStream) localObject1).writeShort(49);//這里寫入的是常量池,包括一些屬性名稱,類名稱,方法描述符,屬性描述符等等,常量池在加載時會被放到方法區(qū)或者說永久代。this.cp.write((OutputStream) localObject1);//這里寫入的是這個類的訪問標識,49代表的是public final,也就是說JDK動態(tài)代理生成的代理類都是final的((DataOutputStream) localObject1).writeShort(49);//寫入代理類的類名((DataOutputStream) localObject1).writeShort(this.cp.getClass(dotToSlash(this.className)));//寫入代理類的父類類名,也就是Proxy類,這個位置的類如果說是JAVA文件,相當于extend后面的類,也就是父類((DataOutputStream) localObject1).writeShort(this.cp.getClass("java/lang/reflect/Proxy"));//寫入代理類所實現(xiàn)的接口數(shù)量 ((DataOutputStream) localObject1).writeShort(this.interfaces.length);//寫入代理類所實現(xiàn)的接口類名,同樣的,對于JAVA文件來說,相當于implements后面的接口,也就是實現(xiàn)的接口for (int m = 0; m < this.interfaces.length; m++) {((DataOutputStream) localObject1).writeShort(this.cp.getClass(dotToSlash(this.interfaces[m].getName())));}//寫入屬性個數(shù)((DataOutputStream) localObject1).writeShort(this.fields.size());//寫入屬性描述for (Iterator localIterator3 = this.fields.iterator(); localIterator3.hasNext();) {localObject2 = (FieldInfo) localIterator3.next();((FieldInfo) localObject2).write((DataOutputStream) localObject1);}//寫入方法個數(shù)((DataOutputStream) localObject1).writeShort(this.methods.size());//寫入方法描述,方法的code屬性,以及構(gòu)造方法和類構(gòu)造方法都在這里被寫入了。for (localIterator3 = this.methods.iterator(); localIterator3.hasNext();) {localObject2 = (MethodInfo) localIterator3.next();((MethodInfo) localObject2).write((DataOutputStream) localObject1);}//結(jié)束((DataOutputStream) localObject1).writeShort(0);} catch (IOException localIOException2) {throw new InternalError("unexpected I/O Exception");}return localByteArrayOutputStream.toByteArray();}

? ? ? ? ? ? ? ? 其實代理類的class文件并不復(fù)雜,還是有很多規(guī)律可循的,所以上述過程基本上可以讓各位了解下JDK動態(tài)代理生成代理類時都生成了什么東西。

? ? ? ? ? ? ? ? 下面我們可以調(diào)用下JDK中生成Class文件的方法,并且寫入到本地文件,然后使用反編譯工具來看一下生成的代理類到底是什么樣子的。下面是生成文件的測試類。我們暫且將生成的類名寫成TestProxy,代理的接口就是我們上面的TestInterface。如下。

import java.io.File; import java.io.FileOutputStream; import java.io.IOException;import sun.misc.ProxyGenerator;public class CreateClassTest {public static void main(String[] args) throws IOException {byte[] classFile = ProxyGenerator.generateProxyClass("TestProxy", new Class[]{TestInterface.class});File file = new File("F:/TestProxy.class");FileOutputStream fos = new FileOutputStream(file);fos.write(classFile);fos.flush();fos.close();}}

? ? ? ? ? ? ? ? 生成后,我們反編譯過來會是如下格式的JAVA文件。我加入了注釋,大致說明了下文件中生成的部分與剛才分析的時候?qū)懭氲倪^程的對應(yīng)關(guān)系。

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.lang.reflect.UndeclaredThrowableException;//public final的,繼承Proxy,實現(xiàn)你傳入的接口 public final class TestProxy extends Proxyimplements TestInterface {//private static 的Method屬性,對應(yīng)所有方法private static Method m1;private static Method m5;private static Method m3;private static Method m4;private static Method m0;private static Method m2;//唯一的構(gòu)造方法,需要一個InvocationHandler接口傳入public TestProxy(InvocationHandler paramInvocationHandler)throws {super(paramInvocationHandler);}//重寫Object的三個方法public final boolean equals(Object paramObject)throws {try{return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}public final void method3()throws {try{this.h.invoke(this, m5, null);return;}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}//代理的三個方法,回調(diào)傳入的InvocationHandler的invoke方法public final void method1()throws {try{this.h.invoke(this, m3, null);return;}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}public final void method2()throws {try{this.h.invoke(this, m4, null);return;}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}public final int hashCode()throws {try{return ((Integer)this.h.invoke(this, m0, null)).intValue();}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}public final String toString()throws {try{return (String)this.h.invoke(this, m2, null);}catch (Error|RuntimeException localError){throw localError;}catch (Throwable localThrowable){throw new UndeclaredThrowableException(localThrowable);}}//這個就是剛才this.methods.add(generateStaticInitializer());這一句話所加入的靜態(tài)初始化塊,初始化每一個屬性static{try{//每一個屬性所代表的Method都是與上面加入代理方法列表時與固定類綁定的,這是class文件中的格式,方法要與固定的類綁定m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });m5 = Class.forName("TestInterface").getMethod("method3", new Class[0]);m3 = Class.forName("TestInterface").getMethod("method1", new Class[0]);m4 = Class.forName("TestInterface").getMethod("method2", new Class[0]);m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);return;}catch (NoSuchMethodException localNoSuchMethodException){throw new NoSuchMethodError(localNoSuchMethodException.getMessage());}catch (ClassNotFoundException localClassNotFoundException){throw new NoClassDefFoundError(localClassNotFoundException.getMessage());}} }

? ? ? ? 看到這里就知道invoke方法是干嘛的了,其實就是生成的代理類對每一個方法的處理就是回調(diào)invoke方法。從生成的代理類源文件中也可以發(fā)現(xiàn),每一個Method除了hashCode,toString和equals外,都是與所屬的接口綁定的,所以這也就解釋了為什么我們不實現(xiàn)這個接口,只傳入進入的話,不能直接使用method.invoke,而是要轉(zhuǎn)成source對應(yīng)的method才可以調(diào)用。

? ? ? ? 好了,代理模式就分析到這里了,這里講的更多的是代理模式的原理,對于如何使用并沒有講述太多,是因為代理模式在平時工作中用的雖然很多,但我們大多是使用的現(xiàn)成的,原因很簡單,就是因為spring的AOP已經(jīng)給我們弄了一個很好的動態(tài)代理的框架,所以我們幾乎不需要自己去寫,只要明白其原理,知道動態(tài)代理和靜態(tài)代理主要處理的問題是那種的,知道在何處用,也能夠用起來得心應(yīng)手就可以了,當然這只是LZ個人之見,僅供參考。

? ? ? ? 下期預(yù)告,簡單工廠模式。

?

總結(jié)

以上是生活随笔為你收集整理的代理模式详解(包含原理详解)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

激情欧美网 | 国产成人61精品免费看片 | 久久精品一二三区白丝高潮 | 日韩动态视频 | 三级黄色片在线观看 | 国产馆在线播放 | 97成人精品区在线播放 | 国产不卡在线观看 | 久久高清国产 | 最近免费中文字幕mv在线视频3 | 国产成人精品久久二区二区 | 狠狠综合| 国产一区二区三区免费在线观看 | 特级aaa毛片| 麻豆国产视频下载 | 欧美男同视频网站 | 乱子伦av | 日本中文字幕影院 | 亚洲a在线观看 | www.久久精品视频 | 国产日产精品一区二区三区四区的观看方式 | 免费观看福利视频 | 日韩美av在线 | 香蕉久久久久久av成人 | 亚洲v欧美v国产v在线观看 | 在线精品视频免费播放 | 亚洲国产精品女人久久久 | 免费观看9x视频网站在线观看 | 日本高清免费中文字幕 | 亚洲一区日韩 | 在线视频app | 日韩精品亚洲专区在线观看 | 日本论理电影 | 久久人人精 | 国产精选在线观看 | 亚洲精品在线电影 | 久久久久久久免费看 | www.97视频 | 一级淫片在线观看 | 久久精品网址 | 日韩三级在线 | 天天操天天曰 | 日本中文字幕在线电影 | 美女黄视频免费 | 超碰97人人在线 | 91九色在线观看视频 | 狠狠色丁香婷婷 | 久久综合九色99 | 黄色综合| 天天狠狠| 久久国产精品99久久久久久老狼 | 狠狠狠狠狠狠操 | 九九导航 | 欧美大片mv免费 | 亚洲九九九 | 91成人短视频在线观看 | 亚洲a网 | 欧美日韩一区二区视频在线观看 | 在线 精品 国产 | 999国产在线 | 人成电影网 | 国产91在线播放 | 中文字幕电影高清在线观看 | 九九热只有精品 | 日本黄色免费大片 | 午夜视频免费播放 | 欧美激情综合五月色丁香小说 | 狠狠躁夜夜av | 韩国av免费在线 | 国产一区二区综合 | 在线电影 你懂得 | 国产又粗又猛又爽 | 麻豆va一区二区三区久久浪 | 国产无遮挡猛进猛出免费软件 | 亚洲精品字幕在线 | 日韩精品无 | 毛片在线网 | 国产美女无遮挡永久免费 | 国产成人一区二区啪在线观看 | 中文字幕日本特黄aa毛片 | 久久成人综合视频 | 国产高清免费观看 | 日韩一区二区三免费高清在线观看 | 国产精品一区电影 | 久久综合狠狠 | 成年人国产精品 | 亚洲精品国产自产拍在线观看 | 又黄又爽的视频在线观看网站 | 国产在线观看h | 日本三级久久 | 在线免费性生活片 | 在线观看视频一区二区三区 | 深爱婷婷 | 成人91免费视频 | 国产又粗又硬又爽视频 | 91色国产在线 | 一区二区视频欧美 | 一区二区三区电影 | 激情伊人五月天久久综合 | 91视频a| www色片| 成人免费观看视频网站 | 精品视频国产 | 免费观看一级特黄欧美大片 | 亚洲国产精选 | 日本最新一区二区三区 | 国产精品一区二区三区99 | 欧美电影黄色 | 青青五月天 | av福利超碰网站 | 美女黄频免费 | 欧美激情精品久久久久 | 午夜一级免费电影 | 97精品电影院 | 免费观看特级毛片 | 99久久久久久久久久 | 一区二区三区精品久久久 | 狠狠色狠狠色综合日日小说 | 成人免费视频a | 国产一区在线视频 | 9在线观看免费 | 麻豆视频在线 | 成人福利在线播放 | 精品国产视频一区 | 欧美日韩aa | 狠狠狠色 | 亚洲欧美少妇 | 精品一区精品二区高清 | 啪啪激情网| 亚洲免费在线看 | 99精品欧美一区二区 | 91.dizhi永久地址最新 | 欧美 日韩 性 | 久久久高清| 成人免费xxx在线观看 | 久久久久久久久久久久久久免费看 | 日本系列中文字幕 | 国精产品一二三线999 | 精品96久久久久久中文字幕无 | 国产中文字幕三区 | 国产黄 | 日韩一区正在播放 | 国内精品视频在线 | 一区二区三区播放 | 国产一区在线视频播放 | 色综合久久五月天 | 成年人在线免费看片 | 免费观看福利视频 | 特级西西人体444是什么意思 | 一级片免费视频 | 日日夜色| 国产在线观看,日本 | 国产精品一区二区免费在线观看 | 亚洲日本va在线观看 | 精品一区 在线 | av成人在线看 | 成人影视免费看 | 日韩在线免费视频观看 | 国产一级二级视频 | 丁香婷婷色月天 | 国产精品欧美激情在线观看 | 在线看片91 | 激情五月色播五月 | 一区二区欧美激情 | 97精品国产97久久久久久 | 99精品欧美一区二区蜜桃免费 | 在线免费中文字幕 | 欧美成人精品欧美一级乱 | 五月天久久久 | 毛片3| 久久99精品国产一区二区三区 | 欧美 日韩 国产 成人 在线 | 在线观看一 | 久久天堂精品视频 | 国产亚洲成人精品 | 精品久久久久久久久亚洲 | 亚洲精品视频在线观看网站 | 国产精品99精品久久免费 | 色婷婷综合久久久久中文字幕1 | 操高跟美女 | 美女视频久久久 | 成人h视频在线 | 久久精品99视频 | 伊人永久在线 | 国内精品久久久久久久影视麻豆 | 中文字幕中文字幕 | 伊人网综合在线观看 | 精品久久久成人 | 在线观看久 | 国产福利在线 | 日韩免费中文 | 日韩理论在线观看 | 午夜12点| 99免费精品| 久久国产麻豆 | 九九精品久久久 | 一区二区三区精品在线 | 日韩字幕 | 在线观看免费黄色 | 天天射天天射天天 | 一区国产精品 | 久久久黄色av | 视频在线观看国产 | 午夜久久网站 | 婷婷六月天综合 | 国产免费不卡 | 国产在线免费av | 麻豆精品传媒视频 | 亚洲激精日韩激精欧美精品 | 久久久免费少妇 | 日韩特级黄色片 | 亚洲午夜精品久久久久久久久久久久 | 最近2019中文免费高清视频观看www99 | 天天摸日日摸人人看 | 韩日精品在线 | 亚洲日韩中文字幕 | 午夜久久精品 | 国产精品免费在线 | 免费观看视频的网站 | 欧美最爽乱淫视频播放 | 97在线视频免费 | 欧美精品在线观看免费 | 999成人网 | av大全免费在线观看 | 成人国产亚洲 | 亚洲精品久久久蜜臀下载官网 | 亚洲成人免费在线观看 | 伊人久久av | 又污又黄网站 | 久久五月天婷婷 | 久久夜靖品| 欧美日韩亚洲在线观看 | 欧美性做爰猛烈叫床潮 | www一起操| 精品国产欧美一区二区 | 国产剧情av在线播放 | 国产中文字幕在线观看 | 免费视频久久久久 | 日一日干一干 | 亚洲激情视频在线观看 | 亚洲第一区精品 | 久久九九国产精品 | 久久好看 | 日本在线观看中文字幕 | 久久99国产精品视频 | 九九九九热精品免费视频点播观看 | 国产成人精品久久久 | 欧美日韩国产一区二区三区在线观看 | 91视频在线免费观看 | 久久观看免费视频 | 日韩动态视频 | 国产色婷婷精品综合在线手机播放 | 亚洲精品九九 | 日本在线免费看 | 懂色av懂色av粉嫩av分享吧 | 亚洲天堂视频在线 | 中文在线免费看视频 | 日韩一区二区免费在线观看 | 久久婷婷国产色一区二区三区 | 国产亚洲成人网 | 欧美色图视频一区 | 狠狠操狠狠 | 亚洲精品mv在线观看 | 91亚洲国产成人 | 久久国产影视 | 激情一区二区三区欧美 | 国产黄色美女 | 国产精品久久久久毛片大屁完整版 | 韩国一区视频 | 国产在线精品国自产拍影院 | 免费在线观看的av网站 | www夜夜 | 99视频网址| 久久99国产精品久久99 | 黄色小说视频在线 | 伊香蕉大综综综合久久啪 | 婷婷久久五月 | 国产一级大片免费看 | 日本少妇高清做爰视频 | 色综合久久五月天 | 草在线视频 | 在线观看日本高清mv视频 | 国产精品成人久久久久 | av官网 | 欧美日韩在线免费观看 | 国产精品一区一区三区 | 99精品在线观看 | 国产一区二区在线免费播放 | 欧美日韩视频在线一区 | 亚洲一区二区视频在线播放 | 国产在线精品播放 | 欧美性生活免费 | 一区二区亚洲精品 | 天天干夜夜爽 | 超碰人人做 | 欧美色操 | 青青草久草在线 | av免费在线看网站 | 国产在线看一区 | 亚洲国产精品va在线看 | 国产精品九九久久99视频 | 91精品国产自产在线观看 | 免费av在线播放 | 天天干天天天天 | 菠萝菠萝在线精品视频 | 在线观看国产成人av片 | 久久av黄色| 精品一区 在线 | 蜜臀久久99精品久久久酒店新书 | 天天操综合网站 | 色综合久久网 | 久久久久久国产精品999 | 夜夜夜夜夜夜操 | 天天躁日日 | 中文字幕欧美日韩va免费视频 | 久久99精品国产一区二区三区 | 日本久久久久久久久久久 | 国产精品久久久久久欧美 | 午夜视频久久久 | 欧美日韩调教 | 日韩精品一区二区三区免费观看视频 | 91最新网址| 久久久久久久久影院 | 一区二区三区日韩在线观看 | 丰满少妇在线观看 | www.狠狠操.com| 在线电影日韩 | 激情综合五月网 | 六月色婷 | 天天综合色 | 国产免费片 | 在线精品视频在线观看高清 | 久草在线免费在线观看 | 久久性生活片 | 香蕉在线视频播放网站 | 人人干人人做 | 福利二区视频 | 国产精成人品免费观看 | 91精品网站 | 国产裸体视频网站 | 久久高清视频免费 | 中文字幕在线观 | 有码一区二区三区 | 日韩高清不卡一区二区三区 | 欧美日韩亚洲第一页 | 国产精品一级在线 | 天天爽综合网 | 亚洲天天综合 | 干天天 | 国产精品欧美日韩在线观看 | 97香蕉久久超级碰碰高清版 | 国产在线观看 | 99自拍视频在线观看 | 色 中文字幕| 黄色大片免费播放 | 日韩精品亚洲专区在线观看 | 天天撸夜夜操 | 国产精品免费在线 | 超碰免费观看 | 欧美性生活久久 | 91视频免费视频 | 天天要夜夜操 | 婷婷六月综合亚洲 | 国产一级性生活视频 | 国产精品久久久999 国产91九色视频 | 三级黄色免费片 | 精品主播网红福利资源观看 | 欧美二区在线播放 | 免费观看不卡av | 久热精品国产 | 国产一级在线播放 | 久久精品久久精品 | 免费又黄又爽视频 | 久久精品视频免费观看 | 日韩精品一区二区三区在线播放 | 麻豆视频免费在线播放 | 日韩中文字幕免费看 | 久久国产精品影视 | 免费av片在线 | 亚洲精品乱码久久久久久高潮 | 91看片在线 | 丁香免费视频 | 一级久久精品 | h文在线观看免费 | 国产91影院| 国产精品免费久久久久久 | 91精品国产自产在线观看永久 | 日韩免费在线视频 | 免费在线观看黄 | 国产少妇在线观看 | 亚洲精品国产综合99久久夜夜嗨 | 中文字幕丝袜制服 | 91人人澡 | 香蕉视频在线看 | 色永久免费视频 | japanesexxxhd奶水 91在线精品一区二区 | 久久综合色8888 | 91天堂影院 | 亚洲精品国产日韩 | 免费毛片aaaaaa | 欧美男男激情videos | 久久精品视频在线免费观看 | 天天曰天天曰 | 日本久久久久久久久 | 97视频人人免费看 | 久久资源在线 | 在线电影日韩 | 国产精品igao视频网网址 | 婷婷精品国产欧美精品亚洲人人爽 | 亚州国产精品 | 久久久久久欧美二区电影网 | bayu135国产精品视频 | 色婷婷成人 | 国产亚洲综合在线 | 亚洲精品玖玖玖av在线看 | 91九色视频国产 | 人人澡超碰碰97碰碰碰软件 | 日韩av免费在线电影 | 亚洲视频在线观看免费 | 日韩理论在线观看 | 婷婷资源站 | 欧美日产一区 | 国产涩涩在线观看 | 日日操操| 久久手机免费观看 | 91一区二区三区在线观看 | 青青河边草观看完整版高清 | 麻豆视频国产在线观看 | 免费国产在线精品 | 久草视频在线免费播放 | 久久a v电影 | 99精品视频网站 | 麻豆视频在线免费观看 | 手机av电影在线 | 在线最新av | 欧美日韩一区久久 | 99人久久精品视频最新地址 | 久青草视频在线观看 | 亚洲国产欧洲综合997久久, | 91高清免费观看 | 91精品国产99久久久久 | 天堂va欧美va亚洲va老司机 | 久久久久国产成人精品亚洲午夜 | 91在线视频免费 | 久久久国产精品电影 | 日本久久电影网 | 中国老女人日b | 在线观看黄a | 久久涩涩网站 | 亚洲高清激情 | 亚洲一区二区观看 | 久久激情电影 | 久草网在线视频 | 欧美日韩有码 | 亚洲干| 99免费在线播放99久久免费 | 成人午夜电影免费在线观看 | 久久久久亚洲精品男人的天堂 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 精品久久久久久亚洲综合网站 | 国产精品999久久久 久产久精国产品 | 日批视频在线播放 | 又色又爽的网站 | 日韩视频一二三区 | www黄色com | 国产永久免费高清在线观看视频 | 久久综合色婷婷 | 午夜精品久久一牛影视 | 国产精品成人自拍 | 成人午夜精品 | 欧美福利在线播放 | 视频一区视频二区在线观看 | 久久在线看| 又黄又爽的免费高潮视频 | 久久这里只有精品1 | 国产亚洲无 | 亚洲丝袜一区二区 | 婷婷香蕉 | 国产午夜麻豆影院在线观看 | 久久不射网站 | 亚洲国产网站 | 婷婷社区五月天 | 精品国产一区二区三区四区在线观看 | 国产精品久久久久三级 | 久久成人国产精品一区二区 | 中午字幕在线 | 一区二区三区四区影院 | 国产亚州精品视频 | 最近最新中文字幕视频 | 国产日韩欧美精品在线观看 | 日韩免费电影一区二区三区 | av片在线看 | 欧美韩国日本在线观看 | 一本一本久久a久久 | 在线观看日韩精品视频 | 日日干天天插 | 在线观看精品国产 | 97国产电影 | 狠狠干狠狠操 | www.狠狠插.com | 五月婷婷婷婷婷 | 西西44人体做爰大胆视频 | 亚洲黄色成人av | 国产精品9999 | 91丨九色丨首页 | 精品中文字幕在线观看 | 高清一区二区 | av电影一区二区三区 | 热久在线 | 999久久久免费精品国产 | 手机看片1042 | 亚洲色图 校园春色 | 中文字幕高清在线播放 | 亚洲欧洲一级 | 激情五月看片 | 黄色av成人在线观看 | 国产一区二区在线免费观看 | 五月天天色 | 久久国产免费看 | 视频在线观看91 | 久久久资源网 | 亚洲综合日韩在线 | 久久久久久蜜av免费网站 | 91麻豆精品一区二区三区 | www操操操 | 96视频免费在线观看 | 96av视频| 午夜精品福利一区二区 | 国产精品永久久久久久久久久 | 亚洲第五色综合网 | 碰碰影院 | 日韩av高清在线观看 | 99久久毛片 | 欧美日韩18| 中文字幕日韩伦理 | av天天色 | www.色综合.com | 欧美精品v国产精品v日韩精品 | 午夜黄网 | 7777xxxx| 亚洲天堂精品 | 在线亚洲欧美日韩 | 色婷婷久久 | 97色在线观看免费视频 | 久久精品香蕉 | 伊人va| 亚洲三级网站 | 久久久免费毛片 | 1区2区视频 | 久久久久国 | 国产麻豆精品一区二区 | 国产成人在线免费观看 | 日韩在线视频网站 | 草久久久| 久久一区二区三区国产精品 | 中文av一区二区 | 99性视频| 一区二区三区在线免费播放 | 欧美一级高清片 | 国产免费视频一区二区裸体 | 国产一区二区久久久 | 欧美在线aa | 国产小视频在线免费观看视频 | 青青草国产成人99久久 | 久久精品a| 久久夜色精品国产欧美乱极品 | 手机看片中文字幕 | 成人黄色大片在线免费观看 | 欧美视频二区 | 久久中文字幕导航 | 久久精品精品电影网 | 午夜精品一区二区国产 | 六月色婷 | 色婷婷伊人 | 99r在线 | 色综合色综合色综合 | 久草在线视频资源 | 国产在线小视频 | 国产精品视频久久 | 国产伦精品一区二区三区在线 | 国产一区久久 | 国产精品在线看 | 天天伊人狠狠 | 欧美日韩视频一区二区三区 | 亚洲春色综合另类校园电影 | 免费成人av网站 | 日本久久综合视频 | 日韩av一区二区在线影视 | 国产剧情在线一区 | 97av免费视频 | 美女网站免费福利视频 | 国产小视频国产精品 | 国产精品美女久久久网av | 免费的黄色av | av在线短片 | 精品国产一区二区三区av性色 | 久久精品伊人 | 999男人的天堂 | 99久久精品国产免费看不卡 | 亚洲国产精品电影 | 久久久久在线 | 婷婷久久网站 | 五月婷综合网 | 亚洲精品在线看 | 婷婷五天天在线视频 | 亚洲国产精品成人精品 | 日日操操| 国产va在线 | 日韩欧美大片免费观看 | 国产小视频免费在线网址 | 免费久久99精品国产婷婷六月 | 国产精品videossex国产高清 | 人人网av | 激情图片久久 | 日日干夜夜草 | 天天天天天天操 | 午夜精品久久久久久99热明星 | 国产一在线精品一区在线观看 | 日韩亚洲国产中文字幕 | 天天天天天天天天操 | 精品国产一区二区在线 | 国产福利91精品一区 | 最新av网址大全 | 一级一级一片免费 | 激情久久婷婷 | 六月婷婷久香在线视频 | 国产精品久久久久久久久毛片 | 亚洲精品456在线播放 | 日韩免费在线观看 | 国产中文字幕在线看 | 激情综合五月婷婷 | 日日日日干 | 亚洲精品视频在线免费播放 | 国产自制av | 成年人电影毛片 | 欧美aa在线 | 久久国内精品 | 99久久日韩精品视频免费在线观看 | 嫩草av在线 | 不卡的av在线播放 | 97视频网站 | 国产美腿白丝袜足在线av | 在线中文字幕一区二区 | 五月花激情 | 91激情视频在线观看 | 免费a v在线| 亚洲免费在线视频 | 有码中文字幕在线观看 | 亚洲日本韩国一区二区 | 中文字幕成人网 | av线上免费看 | 婷婷六月网 | 免费特级黄色片 | 欧美一进一出抽搐大尺度视频 | 麻豆视频免费在线 | 久久视频免费 | 国产精品99爱 | 色五月成人 | 国产日韩欧美视频在线观看 | 在线看黄色的网站 | 五月婷婷在线视频 | a天堂最新版中文在线地址 久久99久久精品国产 | 国产在线精品国自产拍影院 | 欧美日韩国产一区 | 久久久久免费精品国产 | 国产精品不卡视频 | 伊人天堂av| 色av男人的天堂免费在线 | 国产精品麻豆免费版 | 日本黄色免费观看 | h动漫中文字幕 | 美女黄频视频大全 | 丁香午夜| 黄色成人在线 | 超碰在线观看av.com | 嫩小bbbb摸bbb摸bbb | 欧美亚洲精品在线观看 | 国产片免费在线观看视频 | 国产97碰免费视频 | 天天添夜夜操 | 超碰在线色| 视频一区在线播放 | 久久免费视频7 | 91免费在线视频 | 亚洲欧美成人 | 久久国产精品精品国产色婷婷 | 精品久久久成人 | 黄色电影在线免费观看 | 天天干,天天草 | 在线亚洲成人 | 日韩精品高清视频 | 中文字幕在线观看一区 | 国产在线观看午夜 | www免费黄色| 欧美激情片在线观看 | 日韩视频一区二区三区 | 久久综合久久久 | 成人av影视观看 | 日本公妇色中文字幕 | 亚洲 在线 | 免费a视频在线 | 午夜男人影院 | 国产精品国产三级国产aⅴ无密码 | 综合色影院 | 国产精品第一页在线 | 精品96久久久久久中文字幕无 | 婷婷av色综合 | 欧美性生交大片免网 | 亚洲精品一区二区三区四区高清 | 色噜噜日韩精品一区二区三区视频 | 免费在线观看午夜视频 | 午夜国产在线观看 | 久久8精品 | 久久超级碰视频 | 亚洲精品乱码久久久久久蜜桃不爽 | 日韩免费一级电影 | 久久高清免费视频 | 久久艹人人| 韩国一区二区在线观看 | 九九热免费观看 | 97视频人人澡人人爽 | 久久免费美女视频 | 九九九九九九精品 | 人人爽人人爽人人爽人人爽 | 国产黄色特级片 | 欧美一级电影免费观看 | 亚洲精品电影在线 | 色综合天天色 | 日本午夜在线观看 | 99久久精品国产毛片 | 欧美日韩高清一区二区 国产亚洲免费看 | 久久韩国免费视频 | 四虎欧美 | 免费观看黄色12片一级视频 | 国内外成人免费在线视频 | 亚洲永久精品视频 | 超级碰碰免费视频 | 国产不卡在线看 | 久久婷亚洲五月一区天天躁 | 天天综合亚洲 | 一级黄色片在线免费看 | 人人草在线观看 | 高清日韩一区二区 | 欧美大片在线观看一区 | 97超在线视频 | 久久国产视频网站 | 色国产视频 | 久久久久中文字幕 | 精品成人在线 | 亚洲免费专区 | 国产一区二区三精品久久久无广告 | 日韩三级av | 国产1区2区 | 国产婷婷久久 | 久久精品第一页 | 日本久久视频 | 亚洲精品久久在线 | 一区二区精品久久 | 91污污 | 国产免费黄视频在线观看 | 国产精品久久久久国产精品日日 | 久久99电影 | 亚洲国产成人在线 | 99精品国产一区二区 | 免费看的黄网站 | 久久看片网 | 久久久久久久久艹 | 天天色天天综合网 | 成人在线播放免费观看 | 99视屏| 中文字幕在线影视资源 | 亚洲欧洲国产精品 | 天天操天天舔天天爽 | 欧洲一区精品 | 日本成人a | 91porny九色在线播放 | 久久亚洲热| 操操操影院| 欧美一二三视频 | 在线电影91 | 亚洲欧美视屏 | 黄色特级毛片 | 国产成人精品日本亚洲999 | 国产一区欧美一区 | 96国产精品视频 | 天天综合视频在线观看 | av在线免费网 | 免费在线一区二区 | 国产午夜一级毛片 | 日韩免费电影一区二区 | 天干啦夜天干天干在线线 | 国产日女人 | 日韩美女高潮 | 亚洲成a人片77777潘金莲 | 中字幕视频在线永久在线观看免费 | 亚洲日日日 | 亚洲精品黄色在线观看 | 国产一区二区播放 | 午夜精品久久久久久中宇69 | 91麻豆文化传媒在线观看 | a级片在线播放 | 91九色porny在线 | 精品久久久久久久久久久久久 | 天天干天天射天天爽 | 国产一在线精品一区在线观看 | 免费人做人爱www的视 | 成人黄色在线视频 | 亚洲精品理论片 | 精品国产一区二区三区久久影院 | 国产一级大片在线观看 | 97超碰色 | 丁香激情综合久久伊人久久 | 久久久资源 | 国产麻豆视频 | 911香蕉 | 日韩精品一区电影 | 在线探花 | 岛国av在线不卡 | 婷婷成人在线 | 少妇按摩av | 操天天操 | 色婷婷激情四射 | 日韩xxxx视频 | 日韩无在线 | 国产在线不卡精品 | 欧美色久| 不卡的av在线播放 | 国产高清福利在线 | 99精品视频在线播放免费 | 亚洲日本精品 | 2022国产精品视频 | 久久福利剧场 | 高清av在线免费观看 | 精品一区二区在线免费观看 | h动漫中文字幕 | 日韩一区二区三区不卡 | 人人藻人人澡人人爽 | 最近中文字幕 | 国产999精品久久久久久绿帽 | 久久ww| 国产精品成久久久久 | 97av在线视频免费播放 | 久久日韩精品 | 狠狠色噜噜狠狠 | 精品国自产在线观看 | 91成人国产| 免费av在线播放 | 欧美一区二区在线免费看 | 三上悠亚一区二区在线观看 | 国产艹b视频 | 99久久久国产精品美女 | 大片网站久久 | 亚洲精品色视频 | 婷婷精品国产欧美精品亚洲人人爽 | 中文字幕成人一区 | 国产精品视频专区 | 中文在线资源 | 久久国产精品久久久久 | 日韩久久久 | 99精品乱码国产在线观看 | 亚洲成av人影片在线观看 | 久久免费视频国产 | 91精品专区 | 国产一区二区精品久久 | 激情婷婷av| 精品视频成人 | 精品国产精品久久 | 中文字幕精品一区 | 98久9在线 | 免费 | 在线中文字幕播放 | 久草在线最新免费 | 久久手机免费观看 | 亚洲精品资源在线 | 99视频+国产日韩欧美 | 亚洲一区二区精品3399 | 亚洲一区二区观看 | 欧美精品中文在线免费观看 | 欧美久久成人 | 中文字幕在线观看一区二区三区 | 亚洲欧美国产日韩在线观看 | 天天综合网久久 | 亚洲第一区在线播放 | 成人在线视频观看 | 国产精品 亚洲精品 | 亚洲区精品视频 | 欧美精品九九99久久 | 国产成人精品免高潮在线观看 | 香蕉网在线观看 | 亚洲免费av在线播放 | 亚洲黄色片在线 | 久久99国产一区二区三区 | 中文字幕在线免费观看视频 | 最近中文字幕免费观看 | 午夜在线观看影院 | 亚洲精品乱码久久久久久高潮 | 国产精品嫩草影院9 | 黄色一级片视频 | 成人在线免费小视频 | 日韩一区二区在线免费观看 | 在线中文字幕播放 | 一本到视频在线观看 | 国产超碰97 | 亚洲精品国产精品99久久 | 在线观看视频免费大全 | 国产精品成人久久久久久久 | 中文字幕成人一区 | av三级在线看 | 婷婷综合视频 | 天天干夜夜想 | 免费午夜视频在线观看 | 久草在线视频在线观看 | 久久久久高清毛片一级 | 欧女人精69xxxxxx | 国产精品毛片久久久久久久久久99999999 | 91一区二区三区久久久久国产乱 | 国产裸体bbb视频 | 久99久在线视频 | 国产美女网站在线观看 | 99这里有精品 | 亚洲视频在线视频 | 国产精品国产毛片 | 8090yy亚洲精品久久 | 视频一区二区在线观看 | 婷婷av综合 | av解说在线 | 午夜黄网| 国产精品大片在线观看 | 亚洲三级精品 | 久草在线视频资源 | 在线欧美日韩 | 91精品久久久久久综合乱菊 | 青春草国产视频 | 丁香免费视频 | 日韩视频欧美视频 | 免费看一级特黄a大片 | 亚洲黄色大片 | 欧洲一区二区三区精品 | 成人性生交大片免费观看网站 | 色资源在线 | 亚洲最大成人免费网站 | 丝袜足交在线 | 欧美最猛性xxxxx免费 | 婷婷视频在线播放 | 九色免费视频 | 天天干天天干天天射 | 国产一区二区高清 | 超碰免费成人 | 狠狠干我 | 日韩一区二区三 | 免费看污的网站 | 亚洲一区不卡视频 | 天天拍天天操 | 婷婷在线网| 日本黄色大片免费 | 国产不卡免费av | 日韩三级在线 | 毛片99| 天天操天天射天天添 | 热久在线 | 日本特黄特色aaa大片免费 | 免费在线观看成人 | 亚洲高清视频一区二区三区 | 日韩精品一区二区在线视频 | 国产.精品.日韩.另类.中文.在线.播放 | 国产一二区视频 | 精品亚洲va在线va天堂资源站 | 亚洲天堂色婷婷 | 97视频在线免费观看 | 天天操天天曰 | 一级片色播影院 | 精品在线不卡 | 亚洲天堂精品视频 | 激情伊人五月天 | 午夜精品视频一区 | 中文字幕免费观看 | 国产经典三级 | 国产裸体bbb视频 | 天天操 夜夜操 | 成年一级片 | 国产在线看一区 | 91少妇精拍在线播放 | 中文字幕久久精品一区 | 婷婷久久国产 | 久久电影中文字幕视频 | 五月天婷婷综合 | 婷婷五月在线视频 | 亚洲精品字幕在线观看 | 国产在线色 | 欧美韩国在线 | 摸阴视频 | 中文字幕亚洲综合久久五月天色无吗'' | 99久热在线精品视频观看 | www.91国产 | 久久国产日韩 | 精品久久一区二区三区 | 国产在线免费 | 国产一区二区久久久久 | 日本午夜在线亚洲.国产 |