成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

.NET可逆框架設計

開發 架構 后端
前段時間一直在學習和研究.NET事務處理,慢慢的我發現可以使用事務處理來實現一種可逆的系統框架。這種框架在一些IT社區似乎還沒有見過,但是在我們日常開發中確實有這個需求。所以我花了點時間深入的研究了一下事務的原理和使用,實現了以事務為紐帶,以資源為操作對象的可逆框架。

前段時間一直在學習和研究.NET事務處理,慢慢的我發現可以使用事務處理來實現一種可逆的系統框架。這種框架在一些IT社區似乎還沒有見過,但是在我們日常開發中確實有這個需求。所以我花了點時間深入的研究了一下事務的原理和使用,實現了以事務為紐帶,以資源為操作對象的可逆框架。

這里我假設您對事務有了整體的認識,也對自定義事務管理器有過了解。[王清培版權所有,轉載請給出署名]

(可以參考本人的:.NET簡談事務本質論.NET簡談自定義事務資源管理器)

1.    什么是可逆的程序框架

什么叫可逆的?程序的執行是可以被無限制回滾的。

什么叫可逆的框架?實現了對可逆功能的封裝,并能通過簡單的接口調用進行使用??蚣芸赡苡写笥行。蚁脒@么稱呼它是為了表達它的整體性和重要性。

那么到底可逆的需求在哪里?其實在我們開發程序的時候經常會使用事務來進行業務的控制。比如刪除訂單,然后刪除訂單明細等等,對于這樣的要求很多,我們只能將邏輯控制在一個事務范圍內,不能在沒有事務性的邏輯代碼中編寫這種要求的業務功能。等出現未知錯誤的時候在進行事務的回滾。

你也許會問,使用原來的事務處理不是也能進行回滾嗎?當然不是這么簡單的,我們使用事務回滾時只能將資源回滾到最初未進行事務處理前的狀態。(這里不僅僅指的是數據庫事務,而是全局的事務處理) 我們用圖做個比較。[王清培版權所有,轉載請給出署名]

傳統的事務處理圖:

可逆的事務處理圖:

從這兩幅圖中我們可以很明顯的看出,傳統的事務處理在事務處理的過程當中無法控制中間數據,也就是說無法對事務處理進行分段,然后在進行統一的提交或回滾。

在可逆框架的事務處理里我們就可以控制事務的執行階段,在必要的時候我們只需提交或者回滾某一階段的數據。

1.1環境事務

在可逆框架的事務處理圖中,我們看到事務的開始,然后就進行下一步、下一步這樣的操作。在每進行一個下一步操作的時候,就是進入到了一個子事務里處理,在.NET中是可以進行事務的嵌套,其實也就是依賴事務Dependent Transaction實現。通過使用環境事務可以讓事務性感知代碼能自動的識別出您將要使用事務進行操作。所以在每進行下一步操作的時候,只有將當前環境事務切換為您將依賴的子事務才行。如果只是單純的使用依賴事務對象實例在使用,那么將無法進行諸多其他的事務處理。

2可逆框架的實現原理

由于我們只能控制自定義事務資源管理器的內部實現,所以我們在構建自己的數據處理時問題變的簡單多了。

實現可逆框架的核心技術就是使用依賴事務進行事務的克隆操作。將一個大的事務處理邏輯上切割成多了小的事務操作,然后在進行統一的提交或回滾。

在實現上其實就是將Committable Transaction對象進行包裝,實現簡單的調用接口。這里參照了環境代碼的概念,將對象的生命周期控制在代碼片段中。

2.1自定義資源管理器的實現

我們需要擴展IEnlistmentNotification接口的實現,加入對“上一步”、“下一步”的數據操作。

請看代碼:

  1. View Code   
  2.  /***  
  3.   * author:深度訓練  
  4.   * blog:http://wangqingpei557.blog.51cto.com/  
  5.   * **/ 
  6.  using System;  
  7.  using System.Collections.Generic;  
  8.  using System.Text;  
  9.  using System.Transactions;  
  10.    
  11.  namespace ReversibleLib  
  12.  {  
  13.      /// <summary>  
  14.      /// 可逆范圍內的資源管理器。  
  15.      /// 可以使用該類對易失性資源進行事務范圍內的管理。在事務操作范圍內進行可逆操作。  
  16.      /// </summary>  
  17.      /// <typeparam name="T">需要管理的資源類型</typeparam>  
  18.      /// <typeparam name="Xcopy">資源在使用、恢復過程中的數據復制對象。</typeparam>  
  19.      public class ReResourceManager<T, Xcopy> : IEnlistmentNotification, IReversibleGetResourceData<T>  
  20.          where T : classnew()  
  21.          where Xcopy : class 
  22.      {  
  23.          /// <summary>  
  24.          /// 私有字段。資源的持久引用。  
  25.          /// </summary>  
  26.          T _commitfrontvalue;  
  27.          /// <summary>  
  28.          /// 私有字段。事務性操作數據對象。  
  29.          /// </summary>  
  30.          T _rollbackfrontvalue = new T();  
  31.          /// <summary>  
  32.          /// 保存數據復制對象。  
  33.          /// </summary>  
  34.          Xcopy _copy;  
  35.          /// <summary>  
  36.          /// 泛型約束需要,內部使用。  
  37.          /// </summary>  
  38.          public ReResourceManager() { }  
  39.          /// <summary>  
  40.          /// 資源管理器內部名稱。便于追蹤  
  41.          /// </summary>  
  42.          public string Name { getset; }  
  43.          /// <summary>  
  44.          /// 重載默認構造函數,使用資源類型和數據復制對象初始化資源管理器。  
  45.          /// </summary>  
  46.          public ReResourceManager(T t, Xcopy icopy)  
  47.          {  
  48.              (icopy as IResourceCopy<T>).Copy(_rollbackfrontvalue, t);  
  49.              _commitfrontvalue = t;  
  50.              _copy = icopy;  
  51.          }  
  52.    
  53.          #region IEnlistmentNotification 成員  
  54.          public void Prepare(PreparingEnlistment preparingEnlistment)  
  55.          {  
  56.              preparingEnlistment.Prepared();  
  57.          }  
  58.          public void Commit(Enlistment enlistment)  
  59.          {  
  60.              enlistment.Done();  
  61.          }  
  62.          public void InDoubt(Enlistment enlistment)  
  63.          {  
  64.              enlistment.Done();  
  65.          }  
  66.          public void Rollback(Enlistment enlistment)  
  67.          {  
  68.              (_copy as IResourceCopy<T>).Copy(_commitfrontvalue, _rollbackfrontvalue);//回滾事務  
  69.              enlistment.Done();  
  70.          }  
  71.          #endregion  
  72.    
  73.          #region IReversibleGetResourceData<T> 成員  
  74.          T IReversibleGetResourceData<T>.GetPreviousData()  
  75.          {  
  76.              T result = new T();  
  77.              (_copy as IResourceCopy<T>).Copy(result, _rollbackfrontvalue);  
  78.              return result;  
  79.          }  
  80.          T IReversibleGetResourceData<T>.GetNextData()  
  81.          {  
  82.              T result = new T();  
  83.              (_copy as IResourceCopy<T>).Copy(result, _commitfrontvalue);  
  84.              return result;  
  85.          }  
  86.          #endregion  
  87.      }  
  88.  } 

#p#

2.2可逆框架的入口實現

我們需要簡單的調用就能方便的使用可逆功能,不能以一種新的方式使用。所以這里借鑒了Transaction Scope的設計思想。

請看代碼:

  1. View Code   
  2.  /***  
  3.   * author:深度訓練  
  4.   * blog:http://wangqingpei557.blog.51cto.com/  
  5.   * **/ 
  6.  using System;  
  7.  using System.Collections.Generic;  
  8.  using System.Text;  
  9.  using System.Transactions;  
  10.    
  11.  namespace ReversibleLib  
  12.  {  
  13.      /// <summary>  
  14.      /// 使代碼成為可逆框架的事務性代碼  
  15.      /// </summary>  
  16.      public class ReversibleManagerScope : IDisposable  
  17.      {  
  18.          /// <summary>  
  19.          /// 初始化ReversibleManagerScope新的實例  
  20.          /// </summary>  
  21.          public ReversibleManagerScope()  
  22.          {  
  23.              ReversibleManager._reversibleManager = new ReversibleManager();  
  24.          }  
  25.          /// <summary>  
  26.          /// 使用ReversibleManager對象構造ReversibleManagerScope使用范圍對象  
  27.          /// </summary>  
  28.          /// <param name="manager">ReversibleManager實例</param>  
  29.          public ReversibleManagerScope(ReversibleManager manager)  
  30.          {  
  31.              ReversibleManager._reversibleManager = manager;  
  32.          }  
  33.          /// <summary>  
  34.          /// 使用自定義資源管理器構造ReversibleManagerScope包裝的環境ReversibleManager.Current中的對象實例。  
  35.          /// </summary>  
  36.          /// <param name="source">IEnlistmentNotification資源管理器</param>  
  37.          public ReversibleManagerScope(IEnlistmentNotification source)  
  38.          {  
  39.              ReversibleManager._reversibleManager = new ReversibleManager(source);  
  40.          }  
  41.          /// <summary>  
  42.          /// 全局上下文ReversibleManager對象銷毀  
  43.          /// </summary>  
  44.          public void Dispose()  
  45.          {  
  46.              ReversibleManager._reversibleManager = null;  
  47.          }  
  48.          /// <summary>  
  49.          /// 完成整個操作的提交。該操作將提交事務棧中的所有依賴事務  
  50.          /// </summary>  
  51.          public void Completed()  
  52.          {  
  53.              ReversibleManager.Current.Commit();  
  54.          }  
  55.      }  
  56.      /// <summary>  
  57.      /// 可逆模塊的入口。  
  58.      /// ReversibleManager對事務對象的封裝,實現階段性的事務提交和回滾。  
  59.      /// </summary>  
  60.      public class ReversibleManager  
  61.      {  
  62.          #region 上下文靜態ReversibleManager實例  
  63.          /// <summary>  
  64.          /// 持有對可逆框架的對象引用  
  65.          /// </summary>  
  66.          internal static ReversibleManager _reversibleManager;  
  67.          /// <summary>  
  68.          /// 獲取當前上下文中可逆框架  
  69.          /// </summary>  
  70.          public static ReversibleManager Current  
  71.          {  
  72.              get { return _reversibleManager; }  
  73.          }  
  74.          #endregion  
  75.    
  76.          #region 構造對象  
  77.          /// <summary>  
  78.          /// 默認構造函數  
  79.          /// </summary>  
  80.          public ReversibleManager() { }  
  81.          /// <summary>  
  82.          /// 表示可提交的事務(主事務)  
  83.          /// </summary>  
  84.          private CommittableTransaction _commiTransaction;  
  85.          /// <summary>  
  86.          /// 支持兩階段提交協議的資源管理器(主資源管理器)  
  87.          /// </summary>  
  88.          private IEnlistmentNotification _resourceManager;  
  89.          /// <summary>  
  90.          /// 重載構造函數,使用自定義資源管理器構造可逆模塊的開始。  
  91.          /// </summary>  
  92.          /// <param name="resource">IEnlistmentNotification接口對象</param>  
  93.          public ReversibleManager(IEnlistmentNotification resource)  
  94.          {  
  95.              _resourceManager = resource;  
  96.              InitLoad(IsolationLevel.Serializable);  
  97.          }  
  98.          /// <summary>  
  99.          /// 重載構造函數,使用自定義資源管理器、內部事務范圍的事務隔離級別構造可逆模型的開始。  
  100.          /// </summary>  
  101.          /// <param name="resource">IEnlistmentNotification接口對象</param>  
  102.          /// <param name="isolationlevel">IsolationLevel枚舉成員</param>  
  103.          public ReversibleManager(IEnlistmentNotification resource, IsolationLevel isolationlevel)  
  104.          {  
  105.              _resourceManager = resource;  
  106.              InitLoad(isolationlevel);  
  107.          }  
  108.          /// <summary>  
  109.          /// 事務初始化階段的參數對象  
  110.          /// </summary>  
  111.          TransactionOptions _options;  
  112.          /// <summary>  
  113.          /// 重載構造函數,使用自定義資源管理器、內部事務范圍的事務隔離級別、事務超時時間范圍構造可逆模塊的開始。  
  114.          /// </summary>  
  115.          /// <param name="resource">IEnlistmentNotification接口對象</param>  
  116.          /// <param name="isolationlevel">IsolationLevel枚舉成員</param>  
  117.          /// <param name="span">TimeSpan時間范圍</param>  
  118.          public ReversibleManager(IEnlistmentNotification resource, IsolationLevel isolationlevel, TimeSpan span)  
  119.          {  
  120.              _options = new TransactionOptions();  
  121.              _options.Timeout = span;  
  122.              InitLoad(isolationlevel);  
  123.          }  
  124.          /// <summary>  
  125.          /// 構造CommittableTransaction對象實例。  
  126.          /// </summary>  
  127.          /// <param name="level">事務隔離級別</param>  
  128.          private void InitLoad(IsolationLevel level)  
  129.          {  
  130.              if (_options == null)  
  131.                  _options = new TransactionOptions();  
  132.              _options.IsolationLevel = level;  
  133.              _commiTransaction = new CommittableTransaction(_options);  
  134.              _commiTransaction.EnlistVolatile(_resourceManager, EnlistmentOptions.None);  
  135.              //作為事務棧的頭開始整個可逆結構。  
  136.              _tranStack.Push(_commiTransaction);//壓入事務棧  
  137.              _resourceStack.Push(_resourceManager);//壓入資源棧  
  138.              //設置環境事務,讓所有支持事務性感知框架的代碼都能執行。  
  139.              Transaction.Current = _commiTransaction;  
  140.          }  
  141.          #endregion  
  142.    
  143.          /// <summary>  
  144.          /// 事務棧,依次存放事務。  
  145.          /// </summary>  
  146.          private System.Collections.Generic.Stack<Transaction> _tranStack = new Stack<Transaction>();  
  147.          /// <summary>  
  148.          /// 資源棧,依次存放事務使用的資源。  
  149.          /// </summary>  
  150.          private System.Collections.Generic.Stack<IEnlistmentNotification> _resourceStack = new Stack<IEnlistmentNotification>();  
  151.          /// <summary>  
  152.          /// 階段性事件委托  
  153.          /// </summary>  
  154.          /// <param name="tran">Transaction環境事務</param>  
  155.          public delegate void PhaseHanlder(System.Transactions.Transaction tran);  
  156.          /// <summary>  
  157.          /// 下一步事件  
  158.          /// </summary>  
  159.          public event PhaseHanlder NextEvent;  
  160.          /// <summary>  
  161.          /// 上一步事件  
  162.          /// </summary>  
  163.          public event PhaseHanlder PreviousEvent;  
  164.          /// <summary>  
  165.          /// 開始下一步操作  
  166.          /// </summary>  
  167.          /// <typeparam name="S">IEnlistmentNotification接口實現</typeparam>  
  168.          /// <param name="level">IsolationLevel事務的隔離級別(對全局事務處理設置)</param>  
  169.          /// <param name="source">下一步操作的自定義數據管理器</param>  
  170.          public void Next<S>(IsolationLevel level, S source)  
  171.              where S : class,IEnlistmentNotification, new()  
  172.          {  
  173.              Transaction tran = _tranStack.Peek();//獲取事務棧的頂端事務  
  174.              if (tran == null)  
  175.                  tran = Transaction.Current;//主事務  
  176.              DependentTransaction depentran = tran.DependentClone(DependentCloneOption.BlockCommitUntilComplete);  
  177.              //將本次事務處理的資源管理器壓入資源棧中  
  178.              depentran.EnlistVolatile(source, EnlistmentOptions.None);  
  179.              _tranStack.Push(depentran);  
  180.              _resourceStack.Push(source);  
  181.              //切換環境事務場景  
  182.              Transaction.Current = depentran;  
  183.              if (NextEvent != null)  
  184.                  if (NextEvent.GetInvocationList().Length > 0)  
  185.                      NextEvent(Transaction.Current);  
  186.          }  
  187.          /// <summary>  
  188.          /// 返回上一步操作  
  189.          /// </summary>  
  190.          /// <typeparam name="T">需要接受的數據對象類型</typeparam>  
  191.          /// <param name="refadd">需要接受的數據對象引用</param>  
  192.          public void Previous<T>(out T refadd) where T : class,new()  
  193.          {  
  194.              Transaction tran = _tranStack.Pop();  
  195.              if (tran == null)//頂層事務  
  196.                  Transaction.Current.Rollback();  
  197.              // tran.Rollback();//回滾本事務,將觸發所有克隆事務的回滾。  
  198.              if (PreviousEvent != null)  
  199.                  if (PreviousEvent.GetInvocationList().Length > 0)  
  200.                  {  
  201.                      //設置上一步數據對象  
  202.                      refadd = (_resourceStack.Pop() as IReversibleGetResourceData<T>).GetPreviousData();  
  203.                      PreviousEvent(Transaction.Current);  
  204.                      return;  
  205.                  }  
  206.              refadd = new T();//事務處理異常  
  207.          }  
  208.          /// <summary>  
  209.          /// 提交事物堆棧中的所有事物  
  210.          /// </summary>  
  211.          public void Commit()  
  212.          {  
  213.              if (Transaction.Current is DependentTransaction)  
  214.                  (Transaction.Current as DependentTransaction).Complete();  
  215.              for (int i = 0; i < _tranStack.Count - 1; i++)  
  216.              {  
  217.                  //依賴事務  
  218.                  (_tranStack.Pop() as DependentTransaction).Complete();  
  219.              }  
  220.              //提交事務,主事務。必須進行克隆主體的提交才能完成所有階段的操作。  
  221.              (_tranStack.Pop() as CommittableTransaction).Commit();  
  222.          }  
  223.          /// <summary>  
  224.          /// 回滾事物堆棧中的所有事物  
  225.          /// </summary>  
  226.          public void RollBack()  
  227.          {  
  228.              if (Transaction.Current is DependentTransaction)  
  229.                  (Transaction.Current as DependentTransaction).Rollback();  
  230.              for (int i = 0; i < _tranStack.Count - 1; i++)  
  231.              {  
  232.                  //依賴事務  
  233.                  (_tranStack.Pop() as DependentTransaction).Rollback();  
  234.              }  
  235.              //提交事務,主事務。必須進行克隆主體的提交才能完成所有階段的操作。  
  236.              (_tranStack.Pop() as CommittableTransaction).Rollback();  
  237.          }  
  238.      }  
  239.  } 

3.示例

這里我使用了一個簡單的String Builder作為資源管理器需要管理的對象。

請看代碼:

  1. View Code   
  2.  /***  
  3.   * author:深度訓練  
  4.   * blog:http://wangqingpei557.blog.51cto.com/  
  5.   * **/ 
  6.  using System;  
  7.  using System.Collections.Generic;  
  8.  using System.Text;  
  9.  using System.Data;  
  10.  using System.Transactions;  
  11.  using ReversibleLib;  
  12.    
  13.  namespace ConsoleApplication1  
  14.  {  
  15.      class Program  
  16.      {  
  17.          static void Main(string[] args)  
  18.          {  
  19.              //構造數據  
  20.              StringBuilder strbuilder = new StringBuilder();  
  21.              strbuilder.Append("0");//初始數據為0  
  22.    
  23.              //資源管理器  
  24.              ReResourceManager<StringBuilder, StringBuilderCopy> strResource =  
  25.                  new ReResourceManager<StringBuilder, StringBuilderCopy>(strbuilder, new StringBuilderCopy());  
  26.              strResource.Name = "0資源管理器";  
  27.              //開始進入可逆框架處理環境  
  28.              using (ReversibleManagerScope reversible = new ReversibleManagerScope(strResource))  
  29.              {  
  30.                  try 
  31.                  {  
  32.                      ReversibleManager.Current.PreviousEvent += new ReversibleManager.PhaseHanlder(Current_PreviousEvent);  
  33.                      ReversibleManager.Current.NextEvent += new ReversibleManager.PhaseHanlder(Current_NextEvent);  
  34.                      strbuilder.Append("1");//首次修改數據為01  
  35.    
  36.                      //獲取下一步操作的數據  
  37.                      StringBuilder strbuilder2 = (strResource as IReversibleGetResourceData<StringBuilder>).GetNextData();  
  38.                      //構造下一步操作的自定義資源管理器  
  39.                      ReResourceManager<StringBuilder, StringBuilderCopy> strResource2 =  
  40.                          new ReResourceManager<StringBuilder, StringBuilderCopy>(strbuilder2, new StringBuilderCopy());  
  41.                      strResource2.Name = "2資源管理器";  
  42.                      ReversibleManager.Current.Next<ReResourceManager<StringBuilder, StringBuilderCopy>>(  
  43.                          System.Transactions.IsolationLevel.Serializable, strResource2);  
  44.                      strbuilder2.Append("2");//第二步修改數據為012  
  45.    
  46.                      //返回上一步,也就是回滾對數據進行“2”設置的前一個狀態  
  47.                      StringBuilder strbuilder3;  
  48.                      ReversibleManager.Current.Previous<StringBuilder>(out strbuilder3);//獲取上一步使用的數據,這里應該是01  
  49.    
  50.                      reversible.Completed();//提交所有操作  
  51.                      Console.WriteLine(strbuilder3);  
  52.                  }  
  53.                  catch (Exception err)  
  54.                  { Console.WriteLine(err.Message); ReversibleManager.Current.RollBack(); }  
  55.              }  
  56.              Console.ReadLine();  
  57.          }  
  58.    
  59.          static void Current_NextEvent(Transaction tran)  
  60.          {  
  61.              Console.WriteLine("下一步:" + tran.TransactionInformation.LocalIdentifier);  
  62.              Console.WriteLine("下一步:" + tran.TransactionInformation.DistributedIdentifier);  
  63.          }  
  64.          static void Current_PreviousEvent(Transaction tran)  
  65.          {  
  66.              Console.WriteLine("上一步:" + tran.TransactionInformation.LocalIdentifier);  
  67.              Console.WriteLine("上一步:" + tran.TransactionInformation.DistributedIdentifier);  
  68.          }  
  69.      }  
  70.  } 

這里我使用0作為資源的初始數據,然后進入到第一個環節,我將它附加了1,然后進入到第二個環節,我將它附加了2,這里應該是012了,但是下面我突然又返回到了上一步,所以最后的數據應該是01。如果我們需要使用復雜的數據對象,如常用的Data Table類型,我們一般都是用它來展現一組數據,然后對這組數據進行一系列的操作。

總結:

這篇文章主要是想介紹一下事務的另一種使用方式,對可逆框架的設計方向算是一個拋磚引玉吧,希望大家用的著。

源碼地址:http://files.cnblogs.com/wangiqngpei557/Reversible.zip

原文鏈接:http://www.cnblogs.com/wangiqngpei557/archive/2012/06/24/2560576.html

責任編輯:林師授 來源: 博客園
相關推薦

2012-06-25 09:28:42

.NET可逆框架

2016-03-23 11:05:58

Socket開發框架分析

2009-09-08 09:12:12

LINQ構建框架設計

2020-07-30 10:35:32

Java反射框架設計

2012-01-18 10:20:42

框架設計

2010-09-25 13:09:39

UISymbian

2012-01-10 10:04:43

Node.js

2022-09-15 18:32:13

SPI模型框架

2021-02-23 08:18:04

Java 反射機制

2011-04-22 09:26:57

MVC設計

2022-06-15 11:01:59

自定義SPIJava

2022-10-10 09:11:12

互聯網存儲系統云計算

2017-04-12 23:33:38

DevOps平衡計分卡框架

2022-09-25 21:45:54

日志平臺

2022-04-03 15:44:55

Vue.js框架設計設計與實現

2014-09-23 10:05:55

2013-09-03 09:35:48

無線客戶端框架設計iOS

2013-09-03 09:55:42

iOS無線客戶端框架設計

2013-09-09 10:48:24

iOS無線客戶端框架設計

2022-12-16 12:16:21

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品一区二区电影 | 午夜精品久久久 | 国产精品久久久久影院色老大 | 一片毛片| 日韩成人精品视频 | 欧美三区视频 | 亚洲一区在线免费观看 | 久久爱一区 | 国产线视频精品免费观看视频 | 91精品久久久久久久久久入口 | 成人在线中文字幕 | 国产aⅴ爽av久久久久久久 | 国产精品视频一区二区三区, | 午夜视频一区二区三区 | 国产在线一区观看 | 久久国产精品视频免费看 | 亚洲电影一区二区三区 | 激情网站 | www日本高清 | 中文字幕一区二区三 | 国产一区2区 | 国产精品视频97 | 日韩一区二区三区在线观看 | 欧美高清性xxxxhd | 日本欧美视频 | 日韩第一区 | 国产精品久久久久av | 在线视频h | 成人黄色在线观看 | 精品一二区 | 久久曰视频| 中文字幕在线播放第一页 | 国产一级在线视频 | 欧美精品v国产精品v日韩精品 | 国产精品高清在线 | 亚洲日本免费 | 欧美国产91 | 精品一区二区在线观看 | 欧美一级电影免费观看 | 亚洲一本 | 国产免费一级片 |