精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

SringMVC從入門到源碼,這一篇就夠

開發 后端
Java行業的誰人不知SSM框架呢?除非你告訴我剛學Java,我就相信你不知道SpringMVC。

[[341461]]

SpringMVC簡介

Java行業的誰人不知SSM框架呢?除非你告訴我剛學Java,我就相信你不知道SpringMVC。

關于SringMVC的由來和干嘛用的基本都不用介紹了,基本都知道了。但是有一點可以肯定的是:有很多人只停留在SpringMVC使用層面,對于SpringMVC的底層原理和源碼卻沒有深入了解過。

這一期我們就來了解「SpringMVC的底層原理和源碼」,在以前的JSP時代,代碼中前端和后端都混在一起,可能比較老的程序員就寫過下面的代碼,這就是大名鼎鼎的JSP和Servlet時代。

在一些老的項目中可能就會出現這樣的代碼,這樣的代碼是不是看起來非常的帶勁,要是讓你維護這樣的代碼,想死的心都有。

 

這樣的代碼前端和后端混在一起,相互依賴JSP與Java Bean之間嚴重耦合,java代碼和Html代碼混在一起,這要求開發人員既要會前端也要會后端,給測試帶來了很多不方便,代碼也不能復用。

諸如此類的問題,為了解決這樣的問題,首先就是將這些代碼進行嚴格的劃分,前端與后端的代碼分開,逐漸出現代碼的分層架構,各層職責分明。

 

但是,這樣的模型層也還會有問題,首先每個模塊就需要一個Servlet控制器,模塊多的,控制器就會變得很多,這樣會導致控制器復雜。

并且更換視圖技術麻煩,嚴重依賴Servlet API。Java Bean結構包含持久化層以及業務的處理,數據的封裝,這樣就會導致Java Bean結構臃腫。

按照我們現在代碼的分層,可以把Java Bean又分為「持久層(dao)和服務層(Service)」 以及我們的 「應用控制層(Controller)」。

 

SpringMVC原理

為了簡化控制層(Servlet),在SpringMVC框架中使用「DispatcherServlet(前端控制器)」 調度我們自己的「應用控制層(Controller)」。

就這樣逐漸的演變,出現了我們現在真正意義上的Web MVC三層架構,具體的結構圖如下所示:

 

首先來說明一下SpringMVC幾個核心的組件:

  1. DispatcherServlet:前端前端控制器主要負責調度工作,進行全局的流程控制。比如:調度HandlerMapping然后返回執行鏈。
  2. HandlerMapping:處理器映射器會返回一個執行鏈,通俗來講也就是執行的邏輯順序,執行鏈中包含多個「Interceptor(攔截器)」 和一個「Handler(處理器)」。
  3. HandlerAdapter:處理器適配器里面包含了處理器的調用,使用適配器的設計原則,通過反射調用我們自己的Controller。
  4. Handler:處理器也就是我們的Controller,用戶對應的請求URL請求過來,通過請求與我們Controller的映射規則(HandlerMapping)相對應起來,這個就是處理器。
  5. ModelAndView:模型和視圖,模式(Model)也就是我們的數據,通過上面反射調用Handler(Controller)生成的數據,以及邏輯視圖(View)。邏輯視圖并不是真正的視圖名,它只是一個邏輯視圖名,比如:index。
  6. View:視圖,這時候才會通過上面生成的邏輯視圖名生成對應的物理視圖,返回前端呈現用戶。

Hello World上面說了那么多,其實還是要在項目進行實踐中才會有深刻的體會,下面我們通過實際一個案例進行上面的深刻的理解。

這里我使用idea搭建SSM項目,還用Eclipse的同學,建議你該換工具了,首先New - Project

然后左邊選擇Maven,右邊勾選Create from archetype,并且選擇webapp模塊:

 

下面就是填寫一些GroupId以及ArtifactId,這些比較簡單就直接跳過了,不然會被大佬diss死了,創建完項目后的基本目錄結構如下:

 

并且在resource目錄下分別創建下面四個配置文件「applicationContext.xml、jdbc.properties、log4j.properties、spring-mvc.xml」。

applicationContext.xml是Spring的核心配置文件,內容如下:

  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://www.springframework.org/schema/beans" 
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  4.        xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" 
  5.        xsi:schemaLocation="http://www.springframework.org/schema/beans 
  6.                         http://www.springframework.org/schema/beans/spring-beans-3.1.xsd 
  7.                         http://www.springframework.org/schema/tx 
  8.                         http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> 
  9.  
  10.     <!-- 加載properties文件 --> 
  11.     <bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"
  12.         <property name="location" value="classpath:jdbc.properties"/> 
  13.     </bean> 
  14.  
  15.     <!-- 配置數據源 --> 
  16.     <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
  17.         <property name="driverClassName" value="${driver}"/> 
  18.         <property name="url" value="${url}"/> 
  19.         <property name="username" value="${username}"/> 
  20.         <property name="password" value="${password}"/> 
  21.     </bean> 
  22.  
  23.     <!-- mybatis和spring完美整合,不需要mybatis的配置映射文件 --> 
  24.     <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"
  25.         <property name="dataSource" ref="dataSource"/> 
  26.         <!-- 掃描model包 --> 
  27.         <property name="typeAliasesPackage" value="com.ldc.model"/> 
  28.         <!-- 掃描sql配置文件:mapper需要的xml文件 --> 
  29.         <property name="mapperLocations" value="classpath:mapper/*.xml"/> 
  30.     </bean> 
  31.  
  32.     <!-- Mapper動態代理開發,掃描dao接口包--> 
  33.     <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"
  34.         <!-- 注入sqlSessionFactory --> 
  35.         <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/> 
  36.         <!-- 給出需要掃描Dao接口包 --> 
  37.         <property name="basePackage" value="com.ldc.dao"/> 
  38.     </bean> 
  39.  
  40.     <!-- 事務 --> 
  41.     <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
  42.         <property name="dataSource" ref="dataSource"/> 
  43.     </bean> 
  44.  
  45. </beans> 

這個文件主要配置了數據源、數據庫的來連接的配置信息,數據的詳細信息就放在jdbc.properties中:

  1. driver=com.mysql.cj.jdbc.Driver 
  2. url=jdbc:mysql://localhost:3306/test?characterEncoding=utf8&useSSL=false&serverTimezone=Asia/Shanghai 
  3. username=root 
  4. password=user 
  5. initialSize=0 
  6. maxActive=20 
  7. maxIdle=20 
  8. minIdle=1 
  9. maxWait=60000 

這里的數據庫信息,你們只要修改數據庫的用戶名了密碼就行了,其它的作為測試信息基本就不用修改了。

接下來就是日志的配置信息log4j.properties,這里只做簡單的日志配置:

  1. #日志輸出級別 
  2. log4j.rootLogger=debug,stdout,D,E 
  3.  
  4. #設置stdout的日志輸出控制臺 
  5. log4j.appender.stdout=org.apache.log4j.ConsoleAppender 
  6. #輸出日志到控制臺的方式,默認為System.out 
  7. log4j.appender.stdout.Target = System.out 
  8. #設置使用靈活布局 
  9. log4j.appender.stdout.layout=org.apache.log4j.PatternLayout 
  10. #靈活定義輸出格式 
  11. log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss,SSS} -[%p]  method:[%c (%rms)] - %m%n 

配置完日志信息后,接著配置spring-mvc.xml,這個的SpringMVC框架內的信息配置文件:

這和配置信息也很簡單,主要包括「開啟注解驅動、包掃描、視圖解析器的配置」。

配置完SpringMVC后,最后就是配置web.xml,web.xml是前端請求的入口文件:

  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://www.springframework.org/schema/beans" 
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  4.        xmlns:context="http://www.springframework.org/schema/context" 
  5.        xmlns:mvc="http://www.springframework.org/schema/mvc" 
  6.        xsi:schemaLocation="http://www.springframework.org/schema/beans 
  7.        http://www.springframework.org/schema/beans/spring-beans.xsd 
  8.        http://www.springframework.org/schema/context 
  9.        http://www.springframework.org/schema/context/spring-context.xsd 
  10.        http://www.springframework.org/schema/mvc 
  11.        http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"> 
  12.  
  13.     <!-- 掃描注解,這樣com.xjt包下的文件都能被掃描 --> 
  14.     <context:component-scan base-package="com.ldc"/> 
  15.  
  16.     <!-- 開啟SpringMVC注解模式 --> 
  17.     <mvc:annotation-driven/> 
  18.  
  19.     <!-- 靜態資源默認servlet配置 --> 
  20.     <mvc:default-servlet-handler/> 
  21.   
  22.  <!-- 配置返回視圖的路徑,以及識別后綴是jsp文件 --> 
  23.     <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"
  24.         <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/> 
  25.         <property name="prefix" value="/WEB-INF/jsp/"/> 
  26.         <property name="suffix" value=".jsp"/> 
  27.     </bean> 
  28. </beans> 

在web.xml中主要包含:「默認歡迎頁面的配置、字符編碼過濾器的配置、前端控制器、以及指定spring核心配置文件和SpringMVC的配置文件」。

以上就是最基本的配置,其它的配置信息一般是按需配置,這樣配置完后,我們搭建一個簡單的SSM的項目基本已經完成了。

最后的Maven的坐標依賴,如下:

  1. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" 
  2.          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  3.          xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" 
  4.          version="3.1"
  5.  
  6.   <display-name>mvcDemo</display-name
  7.   <!--項目的歡迎頁,項目運行起來后訪問的頁面--> 
  8.   <welcome-file-list> 
  9.     <welcome-file>index.jsp</welcome-file> 
  10.   </welcome-file-list> 
  11.  
  12.   <!-- 注冊ServletContext監聽器,創建容器對象,并且將ApplicationContext對象放到Application域中 --> 
  13.   <listener> 
  14.     <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> 
  15.   </listener> 
  16.  
  17.   <!-- 指定spring核心配置文件 --> 
  18.   <context-param> 
  19.     <param-name>contextConfigLocation</param-name
  20.     <param-value>classpath:applicationContext.xml</param-value> 
  21.   </context-param> 
  22.  
  23.   <!-- 解決亂碼的過濾器 --> 
  24.   <filter> 
  25.     <filter-name>CharacterEncodingFilter</filter-name
  26.     <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> 
  27.     <init-param> 
  28.       <param-name>encoding</param-name
  29.       <param-value>utf-8</param-value> 
  30.     </init-param> 
  31.     <init-param> 
  32.       <param-name>forceEncoding</param-name
  33.       <param-value>true</param-value> 
  34.     </init-param> 
  35.   </filter> 
  36.   <filter-mapping> 
  37.     <filter-name>CharacterEncodingFilter</filter-name
  38.     <url-pattern>/*</url-pattern> 
  39.   </filter-mapping> 
  40.   <!-- 配置前端控制器 --> 
  41.   <servlet> 
  42.     <servlet-name>springmvc</servlet-name
  43.     <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> 
  44.     <!-- 指定配置文件位置和名稱 如果不設置,默認找/WEB-INF/<servlet-name>-servlet.xml --> 
  45.     <init-param> 
  46.       <param-name>contextConfigLocation</param-name
  47.       <param-value>classpath:spring-mvc.xml</param-value> 
  48.     </init-param> 
  49.     <load-on-startup>1</load-on-startup> 
  50.     <async-supported>true</async-supported> 
  51.   </servlet> 
  52.   <servlet-mapping> 
  53.     <servlet-name>springmvc</servlet-name
  54.     <url-pattern>/</url-pattern> 
  55.   </servlet-mapping> 
  56. </web-app> 

maven做表中主要開發包含的依賴數據庫驅動、日志、mybaties、spring坐標、web mvc的坐標、以及繼承JSP的坐標。

這里前端技術可以繼承你們自己想要的:Freemarker或者Thymeleaf,只需要引入相關的Maven坐標,因為JSP已經基本被淘汰了,這里只是為了作測試,并不關心前端用什么技術。

我們在controller包下創建我們自己的測試類:UserController:

  1. @Controller 
  2. @RequestMapping("/user"
  3. public class UserController { 
  4.  
  5.     @Autowired 
  6.     private IUserService userService; 
  7.  
  8.     @RequestMapping("/getUserById"
  9.     public ModelAndView selectUser(@PathVariable("id") Long id) throws Exception { 
  10.         ModelAndView mv = new ModelAndView(); 
  11.         User user = userService.selectUser(id); 
  12.         mv.addObject("user"user); 
  13.         mv.setViewName("user"); 
  14.         return mv; 
  15.     } 

這里簡單解釋一下:

  1. @Controller:標名它是一個控制器,被Spring容器所管理,這個注解是在@Component后面出的,為了表示代碼的分層,于是就有了@Controller、@Service、@Mapper這三個注解,他們的作用是一樣的。
  2. @RequestMapping:表示接受的請求,還是GetMapping、PostMapping等注解表示請求方法的不同。
  3. @Autowired:表示自動注入,前提就是被注入的對象被Spring容器所管理。
  4. ModelAndView:這個前面說過,它裝的就是數據和邏輯視圖名。

這些還是比較簡單的,通過下面配置Tomcat信息進行部署,就可以啟動項目進行測試了:

 

DispatcherServlet源碼解析

這個還是比較簡單的,還不會可以自行百度,啟動項目后我們來測試一下前面,出現下面的界面說明,你搭建SSM項目的基本環境已經成功了:

 

那么我們的前端請求是怎么一步一步的從「前臺->后臺->前臺」的呢?其實前面我們已經說了SpringMVC的基本原理,在這個基本原理的基礎上,從源碼的角度,進行詳細的解析:

上面說到SpringMVC的核心調度器就是DispatcherServlet,負責主流程的調度工作,在DispatcherServlet里面最主要的方法就是doDispatch:

  1. protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {   
  2.         HttpServletRequest processedRequest = request;   
  3.         HandlerExecutionChain mappedHandler = null;   
  4.         int interceptorIndex = -1;   
  5.    
  6.         try {   
  7.             ModelAndView mv;   
  8.             boolean errorView = false;   
  9.    
  10.             try {   
  11.                    //檢查是否是請求是否是multipart(如文件上傳),如果是將通過MultipartResolver解析   
  12.                 processedRequest = checkMultipart(request);   
  13.                    //步驟2、請求到處理器(頁面控制器)的映射,通過HandlerMapping進行映射   
  14.                 mappedHandler = getHandler(processedRequest, false);   
  15.                 if (mappedHandler == null || mappedHandler.getHandler() == null) {   
  16.                     noHandlerFound(processedRequest, response);   
  17.                     return;   
  18.                 }   
  19.                    //步驟3、處理器適配,即將我們的處理器包裝成相應的適配器(從而支持多種類型的處理器)   
  20.                 HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());   
  21.    
  22.                   // 304 Not Modified緩存支持   
  23.                 //此處省略具體代碼   
  24.    
  25.                 // 執行處理器相關的攔截器的預處理(HandlerInterceptor.preHandle)   
  26.                 //此處省略具體代碼   
  27.    
  28.                 // 步驟4、由適配器執行處理器(調用處理器相應功能處理方法)   
  29.                 mv = ha.handle(processedRequest, response, mappedHandler.getHandler());   
  30.    
  31.                 // Do we need view name translation?   
  32.                 if (mv != null && !mv.hasView()) {   
  33.                     mv.setViewName(getDefaultViewName(request));   
  34.                 }   
  35.    
  36.                 // 執行處理器相關的攔截器的后處理(HandlerInterceptor.postHandle)   
  37.                 //此處省略具體代碼   
  38.             }   
  39.             catch (ModelAndViewDefiningException ex) {   
  40.                 logger.debug("ModelAndViewDefiningException encountered", ex);   
  41.                 mv = ex.getModelAndView();   
  42.             }   
  43.             catch (Exception ex) {   
  44.                 Object handler = (mappedHandler != null ? mappedHandler.getHandler() : null);   
  45.                 mv = processHandlerException(processedRequest, response, handler, ex);   
  46.                 errorView = (mv != null);   
  47.             }   
  48.    
  49.             //步驟5 步驟6、解析視圖并進行視圖的渲染   
  50. //步驟5 由ViewResolver解析View(viewResolver.resolveViewName(viewName, locale))   
  51. //步驟6 視圖在渲染時會把Model傳入(view.render(mv.getModelInternal(), request, response);)   
  52.             if (mv != null && !mv.wasCleared()) {   
  53.                 render(mv, processedRequest, response);   
  54.                 if (errorView) {   
  55.                     WebUtils.clearErrorRequestAttributes(request);   
  56.                 }   
  57.             }   
  58.             else {   
  59.                 if (logger.isDebugEnabled()) {   
  60.                     logger.debug("Null ModelAndView returned to DispatcherServlet with name '" + getServletName() +   
  61.                             "': assuming HandlerAdapter completed request handling");   
  62.                 }   
  63.             }   
  64.    
  65.             // 執行處理器相關的攔截器的完成后處理(HandlerInterceptor.afterCompletion)   
  66.             //此處省略具體代碼   
  67.    
  68.         catch (Exception ex) {   
  69.             // Trigger after-completion for thrown exception.   
  70.             triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, ex);   
  71.             throw ex;   
  72.         }   
  73.         catch (Error err) {   
  74.             ServletException ex = new NestedServletException("Handler processing failed", err);   
  75.             // Trigger after-completion for thrown exception.   
  76.             triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, ex);   
  77.             throw ex;   
  78.         }   
  79.    
  80.         finally {   
  81.             // Clean up any resources used by a multipart request.   
  82.             if (processedRequest != request) {   
  83.                 cleanupMultipart(processedRequest);   
  84.             }   
  85.         }   
  86.     }   

這個方法不長,基本就是負責其它方法的調用,從我們上面分析到前端請求第一步到達SpringMVC后調用HandlerMapping(處理器映射器)返回執行鏈HandlerExecutionChain:

 

我們debug啟動項目,打個斷點看看,這個HandlerExecutionChain到底是個什么東西。

 

我們可以看到,當斷點執行到HandlerExecutionChain后,查看HandlerExecutionChain中的handler其實就是我們自己的請求訪問的Controller,比如上面的我們請求登陸操作,handler里面的信息就是我們自己的LoginController。

同時包含LoginController的BeanType,前端要請求的方法,以及參數這個元數據信息,簡單的概括就是:「HandlerExecutionChain里面handler就是我們要請求的Controller以及和一些interceptors信息」。

那么在獲取到這個HandlerExecutionChain之前肯定是有初始化所有的Spring容器中的Bean以及所有的url與Bean對應的HandlerMapping對象。

這個都是在Spring中去完成的,這個我們后面在做了解,我們再進一步的了解HandlerMapping對象存儲的內容,再getHandler方法里面進行打斷點:

 

handlerMapping是一個List對象,里面主要是這七個成員信息,我們比較熟悉的就是BeanNameUrlMapping和SimpleUrlHandlerMapping對象,這些里面可以看出「handlerMapping主要存儲的各種映射規則」,通過beanName或者url映射到對應的Bean對象。

 

繼續往里面看,可以看到這里有個applicationContext對象,這個也就我們的上下文,里面還有beanFactory,也就是Spring管理的Bean對象都在這個工廠里面,包括Spring自己的和我們自己定義Bean信息。

 

這個就是HandlerMapping對象,主要「包含著的Bean映射規則、Bean詳細信息。」

從HandlerMapping->HandlerExecutionChain的過程,用一句通俗易懂的話概括就是:「從茫茫的人海中找到了你(從beanFactory找到了請求對應的Controller以及方法)」。

當獲取完我們的執行鏈后,接著就是獲取我們的「處理器適配器」(HandlerAdapter),

 

從getHandlerAdapter的方法中可以看到,根據返回的handlerMapping對象中的handler對象來獲取對應的HandlerAdapter對象,直接返回。

 

返回HandlerAdapter對象后,通過執行HandlerAdapter的handle方法獲取ModelAndView對象,從這個方法的上面的注釋來看:Actually invoke the handler.。

實際就是通過「反射」的方式動態的執行我們自己的Controller中的方法,也就是前端請求的Controller,因為mappedHandler.getHandler()返回的「handler對象包含著請求Controller的詳細信息,包括全類名」。

 

獲取到ModelAndView之后,接著就執行我們的攔截器的后置處理方法postHandle。

 

從他的源碼可以看出,它是獲取到所有的攔截器,然后一個一個遍歷,執行。

 

執行完所有攔截器的后置處理方法,就是最后䣌視圖的渲染,這里執行的是processDispatchResult方法,并把ModelAndView對象作為參數傳遞進去。

 

在processDispatchResult方法里面最重要的就是render方法了,執行視圖的渲染,最后將渲染的結果呈現給用戶。

 

到這里DispatcherServlet主要執行邏輯就講完了,其實主要講的還是SpringMVC的從前端請求->后臺->前端這樣的一個過程,限于篇幅,從源碼的角度大概講解這個的過程是怎么跑起來的。

 

一篇文章要把SpringMVC的都講清楚是不可能的,SpringMVC所有講下來,都能寫一本書了,后續的源碼我們繼續精進,這篇作為一個大體脈絡的了解。

本文轉載自微信公眾號「非科班的科班」,可以通過以下二維碼關注。轉載本文請聯系非科班的科班公眾號。

 

責任編輯:武曉燕 來源: 非科班的科班
相關推薦

2021-11-24 22:42:15

WorkManagerAPI

2020-10-17 08:48:12

搞懂“智能聯接”

2025-07-04 02:12:00

2022-07-19 19:39:05

RTK技術定位技術

2020-03-09 17:28:51

NoSQLMongoDB數據庫

2023-09-11 08:13:03

分布式跟蹤工具

2025-07-10 02:11:00

毫米波雷達4G

2021-01-14 07:15:19

NginxWeb服務器

2021-05-14 23:31:50

大數據計算機開發

2021-09-29 09:00:19

Linux虛擬機CentOS

2021-03-12 13:46:49

鴻蒙HarmonyOS應用

2019-06-18 08:13:47

內網公網NAT

2020-10-18 07:32:06

SD-WAN網絡傳統廣域網

2025-10-17 07:05:00

Python數據可視化數據科學

2020-08-03 10:00:11

前端登錄服務器

2023-04-24 08:00:00

ES集群容器

2021-04-14 15:54:20

Kubernetes程序工具

2018-01-17 09:32:45

人工智能卷積神經網絡CNN

2018-11-14 11:57:28

2023-10-30 07:12:04

點贊
收藏

51CTO技術棧公眾號

亚洲激情久久久| 亚洲男人天堂av| 国产成人精品久久久| 久久噜噜色综合一区二区| 成人在线啊v| 欧美日韩美女在线| 在线码字幕一区| 日本高清视频网站| 免费精品99久久国产综合精品| 欧美精品一本久久男人的天堂| 国产麻豆天美果冻无码视频 | 首页亚洲中字| 欧美欧美欧美欧美| 亚洲自偷自拍熟女另类| 黄色成人在线观看| 国产亚洲短视频| 91九色露脸| 国产精品sm调教免费专区| 亚洲天堂黄色| 久久综合久久八八| 摸摸摸bbb毛毛毛片| 精品国产午夜肉伦伦影院| 欧美日本在线看| 国产精品免费入口| 国产91足控脚交在线观看| 国产精品久久三| 欧洲精品久久| 天天干视频在线| 国产福利不卡视频| 国产在线观看精品| 成人小视频在线播放| 国产欧美短视频| 色综合91久久精品中文字幕 | 国产在线观看精品一区二区三区| 天码人妻一区二区三区在线看| 欧美激情一区| 久久久精品国产| 国产 日韩 欧美 综合 一区| 蜜桃视频一区二区| 欧美亚洲免费电影| 久久99久久98精品免观看软件| 欧美freesextv| 伊人av综合网| 中文字幕伦理片| 精品国产91| 亚洲日韩欧美视频| 亚洲 小说 欧美 激情 另类| 亚洲第一福利专区| 亚洲精品美女免费| 最近日本中文字幕| 亚洲国产欧美日韩在线观看第一区 | 一本色道久久88精品综合| 久久丫精品国产亚洲av不卡| 日韩av不卡一区| 日韩成人性视频| 无码国产69精品久久久久网站| 最新国产一区二区| 精品乱码亚洲一区二区不卡| 国产精品果冻传媒| 久久综合五月婷婷| 国产视频亚洲精品| 免费黄色片网站| 欧美www视频在线观看| 久久精品成人一区二区三区| 黄色一级片中国| 欧美日韩理论| 69久久夜色精品国产69| 国产精品久久亚洲| 午夜爱爱毛片xxxx视频免费看| 久久久久国产精品| 欧美人在线观看| 国产在线视频二区| 性xx色xx综合久久久xx| 国产精品久久久久久久久借妻| 中文字幕在线观看视频一区| 国产精品综合一区二区三区| 国产精品播放| 国产视频精品久久| 亚洲码国产岛国毛片在线| 屁屁影院ccyy国产第一页| 大香伊人中文字幕精品| 日韩欧美国产激情| 99热一区二区| 高清日韩欧美| 这里只有精品视频| 国产性一乱一性一伧一色| 一本一本久久| 国产日韩欧美综合| 国模无码一区二区三区| 国产视频视频一区| 国产一区二区三区播放| 午夜伦理福利在线| 欧美一区二区三区日韩| 亚洲欧美视频在线播放| 99热精品久久| 欧美一级免费看| 国产又粗又猛又黄又爽| 91原创在线视频| 欧美一级免费在线观看| 在线天堂资源www在线污| 91精品国产黑色紧身裤美女| 黄色国产在线观看| 欧美 日韩 国产一区二区在线视频| 4438全国成人免费| 国产农村妇女毛片精品久久| 久久久久久久久久久久久久久99 | 精品精品欲导航| 人妻视频一区二区| 影音先锋久久| 亚洲自拍另类欧美丝袜| 电影av在线| 欧美日韩国产色视频| 一级做a爱视频| 国产欧美日韩影院| 91国产在线精品| 国产三级伦理片| 中文字幕成人在线观看| 欧美日韩亚洲一| av日韩精品| 久久香蕉国产线看观看av| 99成人精品视频| 97se狠狠狠综合亚洲狠狠| 超碰超碰超碰超碰超碰| 少妇精品视频在线观看| 国产亚洲精品美女久久久久| 五月婷婷开心网| 成人在线视频一区| 久久久久亚洲av无码专区喷水| 国内自拍亚洲| 在线视频欧美性高潮| 日韩黄色片网站| ww亚洲ww在线观看国产| 黄色免费福利视频| 欧美sss在线视频| 久久久久久久久久久av| 亚洲精品国产精品乱码不卡| 亚洲免费观看高清完整版在线观看| 在线免费视频一区| 欧美久久综合网| 日韩美女在线观看| 久草在线免费福利资源| 色综合天天综合狠狠| 国产熟女高潮一区二区三区| 夜夜夜久久久| 看高清中日韩色视频| 天堂中文av在线资源库| 精品视频在线播放免| 一级黄色免费网站| 久久一区二区视频| www.四虎成人| 精品理论电影在线| 国产精品一区久久久| 免费在线观看av网站| 69堂精品视频| 九九视频在线免费观看| 成人免费黄色在线| 国产在线观看福利| 欧洲毛片在线视频免费观看| 国产精品一区二区性色av| 日本www在线| 日韩欧美一二三四区| 精品午夜福利在线观看| 99免费精品视频| 黄色片视频在线播放| 青青草成人影院| 91香蕉电影院| 91九色porn在线资源| 亚洲精品手机在线观看| 国产videos久久| 国产精品久久色| 国产网友自拍视频导航网站在线观看| 宅男噜噜噜66一区二区66| 一区二区在线观看免费视频| 国产成人精品免费网站| 日韩激情免费视频| 精品久久电影| 亚洲综合第一页| av剧情在线观看| 亚洲图中文字幕| 国产人妖一区二区| 亚洲成人在线观看视频| 欧美人与性囗牲恔配| 国产伦精品一区二区三区在线观看| 91成人综合网| 国产伦一区二区三区| 亚洲xxxxx性| 色一区二区三区| 日韩有码在线观看| 深夜影院在线观看| 69p69国产精品| 制服.丝袜.亚洲.中文.综合懂色| 国产精品无码永久免费888| 中文字幕在线日韩| 性做久久久久久久久久| 色噜噜狠狠成人网p站| 杨钰莹一级淫片aaaaaa播放| 91在线视频18| 激情图片中文字幕| 羞羞视频在线观看欧美| 操bbb操bbb| 精品免费在线| 精品国产免费久久久久久尖叫 | 成人video亚洲精品| 亚洲精品98久久久久久中文字幕| 欧美另类高清videos的特点| 亚洲va欧美va人人爽| 熟女av一区二区| 久久你懂得1024| 亚洲欧美日韩色| 精品一区二区在线视频| 欧美日韩大尺度| 在线看片一区| 99热一区二区三区| 成人av二区| 欧美黑人xxxxx| 精品在线网站观看| 99国内精品久久久久久久软件| 欧美不卡高清一区二区三区| 91av免费观看91av精品在线| 中文字幕有码在线观看| 日韩小视频在线观看| 男人天堂网在线观看| 亚洲国产精品免费| 超碰免费在线97| 欧美精品高清视频| 超碰在线97观看| 一本到高清视频免费精品| 日本少妇激情舌吻| 一区二区三区精密机械公司| 一起操在线播放| 国产精品不卡视频| 你懂得视频在线观看| 国产校园另类小说区| 37p粉嫩大胆色噜噜噜| bt欧美亚洲午夜电影天堂| www.四虎精品| 成人激情校园春色| 国产精品一区二区无码对白| 丰满少妇久久久久久久| 国产sm在线观看| 成人午夜激情在线| 秘密基地免费观看完整版中文| 丰满亚洲少妇av| 亚洲一级Av无码毛片久久精品| 高清不卡在线观看av| 特级特黄刘亦菲aaa级| 成人黄色在线网站| japanese在线观看| av在线一区二区三区| 国产精品无码一区二区三| 91蜜桃传媒精品久久久一区二区| 玖草视频在线观看| 久久精品亚洲一区二区三区浴池| 欧美狂猛xxxxx乱大交3| 国产亚洲短视频| 日韩欧美视频免费观看| 亚洲视频资源在线| 欧美成人三级视频| 亚洲国产日日夜夜| 亚洲伊人成人网| 色欧美88888久久久久久影院| 亚洲av无码乱码国产精品fc2| 欧美日韩亚洲综合在线 | 日韩精品一区二区三区第95| 青青视频在线观| 最近中文字幕2019免费| www.欧美日本韩国| 久久免费精品日本久久中文字幕| 男人天堂视频在线观看| 国产精品免费看久久久香蕉| 成年永久一区二区三区免费视频| 91免费版网站入口| 三妻四妾完整版在线观看电视剧| 亚洲成人网久久久| 青青草在线播放| 在线精品国产成人综合| 成a人片在线观看| 久久免费视频网站| 色综合一本到久久亚洲91| 91九色视频在线| 国产三级精品三级在线观看国产| 欧美日韩高清在线一区| 国产精品黑丝在线播放| 黄色一级视频在线播放| 日本欧美一区二区| 制服.丝袜.亚洲.中文.综合懂| www激情久久| 91porn在线视频| 色婷婷国产精品久久包臀| 国产视频一区二区三| 精品亚洲男同gayvideo网站| 欧美成人三区| 69久久夜色精品国产7777| 日日夜夜综合| 久久精品国产精品国产精品污 | 一个人看的www日本高清视频| 日韩欧美视频一区| 国产高清免费av在线| 国产偷拍一区二区| av不卡免费在线观看| 国产美女免费网站| 亚洲天堂网中文字| 日韩电影在线观看一区二区| 日韩一二三四区| 精品欧美不卡一区二区在线观看 | 日本一区二区三区四区高清视频 | 三区在线观看| 欧美猛交ⅹxxx乱大交视频| 成人国产精品| 精品免费国产| 国户精品久久久久久久久久久不卡| 天天爱天天操天天干| 2019国产精品| 国产午夜视频在线| 欧美一区二区三区男人的天堂 | 欧美大片在线影院| 四虎影视精品永久在线观看| 美女一区视频| 99在线精品视频在线观看| 欧美一区欧美二区| 污免费在线观看| 国产偷v国产偷v亚洲高清| xxxxxx国产| 欧美v日韩v国产v| 日本不卡不卡| 国产精品日日做人人爱| 日韩伦理一区二区三区| 99在线免费视频观看| 国产精品自在在线| 黑人狂躁日本娇小| 欧美亚洲动漫精品| 可以免费看污视频的网站在线| 91精品国产99| 精品福利一区| 国产特级淫片高清视频| 日本欧美韩国| 欧美精品日韩一本| 日韩有码电影| 91干在线观看| 欧美a大片欧美片| 精品无码一区二区三区在线| 成人亚洲一区二区一| 久久这里只有精品免费| 91精品国产色综合久久| 免费观看一区二区三区毛片| fc2成人免费人成在线观看播放| 男人操女人的视频网站| 欧美群妇大交群中文字幕| 成人免费高清在线播放| 日韩免费在线免费观看| 首页亚洲中字| 欧美视频在线播放一区| 久久综合国产精品| 无码人妻丰满熟妇区五十路| 亚洲色图综合网| 91九色综合| 小说区视频区图片区| 国产激情精品久久久第一区二区| 欧美日韩一级在线观看| 精品剧情v国产在线观看在线| 888av在线视频| 欧美久久电影| 看电视剧不卡顿的网站| 一级黄色录像视频| 精品国产制服丝袜高跟| 交100部在线观看| 日韩国产欧美一区| 狠狠久久亚洲欧美| 免费在线一级片| 亚洲毛片在线看| 亚洲老司机网| 国产日本在线播放| 国产丝袜美腿一区二区三区| 国产熟女一区二区丰满| 91精品国产网站| 欧美电影《睫毛膏》| 在线精品视频播放| 免费在线a视频| 加勒比久久高清| 精品一区二区中文字幕| 中文字幕精品在线不卡| 99久久久国产精品无码网爆| 97免费视频在线播放| 日韩伦理视频| av av在线| 欧美日韩国产成人在线免费| 丁香花在线高清完整版视频| 欧美日韩在线精品| 91亚洲大成网污www| 国产又黄又粗又猛又爽的视频| 在线区一区二视频| 99在线播放| 欧美一区二区高清在线观看| 国产一区高清在线| 特级西西444www大精品视频免费看| 日韩视频免费在线观看| 琪琪久久久久日韩精品| 欧美丝袜在线观看| 欧美特黄级在线| 亚洲淫性视频|