亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > PHP > 正文

細看PEAR的錯誤處理

2019-09-08 23:11:28
字體:
來源:轉載
供稿:網友
PEAR提供了強大的錯誤處理機制。這篇文章向你展示如何從這個系統中獲益。


許多程序已經使用了PEAR的包。許多PHP程序員或多或少的熟悉了PEAR中的錯誤處理。但是這個機制并不局限于PEAR的包――所有人都能在他們的類和程序中使用這些方法。

這篇文章被分為兩個部分:首先我們將看看類中用于錯誤處理的函數,然后我們將看看如何基于PEAR錯誤處理機制來處理錯誤。

我們的例子類稱為cvs2db,它把數據從CSV文件插入到數據庫的表中。因為數據可能是手寫的,他們的數據應該在插入之前先得到驗證――落實postcode。函數import()完成讀入,檢查和插入的工作;它返回損壞的記錄數目。如果返回的值大于0,出錯的記錄集能夠使用exportUnvalid()寫入到新的CSV文件中。典型的用法是這樣的:

<?php
$cd = new csv2db();
$dsn = 'mysql://root@localhost/csv2db';
if( 0 < $cd->import("./dat.csv", $dsn, 'address')) {
$cd->exportUnvalid("./dat2.csv");
}
?>



可能的錯誤包括:

要導入的CSV文件不存在,
連接到數據庫失敗,
記錄集損壞,以及CSV導出文件無法創建。

在提供錯誤信息的經典解決方案中你可能寫這樣的代碼:

<?php
$cd = new csv2db();
$dsn = 'mysql://root@localhost/csv2db';
$result = $cd->import("./dat.csv", $dsn, 'address')
switch($result) {
case FILE_NOT_OPENED:
...
break;
case DATABASE_ERROR:
...
break;
default:
if(0 < $result) {
$cd->exportUnvalid("./dat2.csv");
} else {
echo 'every thing ok!'
}
}
?>



這對于短的腳本來說是可接受的也是常用的辦法――但是對于錯誤處理經常受到關注的大程序來說不是這樣。傳統的可能性強迫類的作者做最終的決定!在大部分情況下,這個決定根據的是那時對類的調用而不是基于長期的使用和可重用代碼的思想。一個靈活的錯誤處理機制是可重用代碼的重要部分,PEAR Error API 就是這樣的一種受到良好測試的機制。


用戶眼中的類

除了那兩個函數之外,類提供了一套錯誤處理函數和一個自己的錯誤對象稱為DB2CVS_Error,它有一個特殊的本地化的錯誤信息的特性功能。

現在我將向你展示如何在錯誤發生時控制類的行為。

局部和全局錯誤處理

你用setErrorHandling()管理錯誤處理;這個函數需要兩個參數:第一個是錯誤模式,而第二個(可選的)參數是錯誤模式特定的選項。例如 setErrorHandling(PEAR_ERROR_PRINT, 'This error occurred %s') 還有 setErrorHandling(PEAR_ERROR_TRIGGER, E_USER_WARNING)。


這個函數的調用方式是一般行為中最重要的:靜態還是實體。在類cvs2db中,我們能兩者都用來設置錯誤處理,所有這些調用有相同的結構――為類設置錯誤模式:

// per instance
$cd = new csv2db();
$cd->setErrorHandling(PEAR_ERROR_DIE):
// static
CVS2DB::setErrorHandling(PEAR_ERROR_DIE);
PEAR::setErrorHandling(PEAR_ERROR_DIE);



如果兩者給出同樣的結果,區別在哪?實體調用僅僅為那個類設置而靜態調用對于所有使用PEAR_Error或者從那個類派生的所有類起作用。這個也作用于第一個靜態命令CVS2DB::setErrorHandling(PEAR_ERROR_DIE)――雖然它看上去僅僅影響了cvs2db類。


總結:作為一個實體函數使用命令意味著僅僅為這個實體(局部)設置錯誤模式,而作為靜態函數來調用就是為整個腳本設置錯誤模式(全局)。


setErrorHandling() 和 raiseError()


兩個函數都能夠被靜態調用和作為實體的函數調用。記住怎樣的一個組合使得他們如何互相影響的很重要。

基本上是:setErrorHandling()的靜態調用僅僅影響raiseError()的靜態調用――setErrorHandling()作為實體函數僅僅影響raiseError()作為靜態函數調用。在類csv2db中,使用csv2db::setErrorHandling()來設置錯誤模式是不可行的,因為我們使用$this->raiseError(...)。解決這個聞天有一點小技巧――改寫raiseError():

function raiseError(...,$mode=null, $options=null,...) {
if($mode==null && $this->_default_error_mode!=null) {
$mode = $this->_default_error_mode;
$options = $this->_default_error_options;
}
return PEAR::raiseError(...,$mode, $options,...);
}



這樣,我們映射實體調用到靜態上,如果你用錯誤模式調用raiseError(),然后這個模式將會覆蓋這些設置――這里是指的是全局的設置。

你應當當心錯誤是如何被類拋出的,如果你不小心,這可能導致不可預期的副作用。


錯誤的模式

對錯誤模式的了解對于使用PEAR的錯誤處理來說是重要的。PEAR錯誤處理讓用戶能夠決定怎么去做――注意:下文中術語用戶指的的是實際使用PEAR_Error程序的開發者而不是瀏覽腳本結果或者網頁的用戶。我將詳細展示可能的錯誤模式。

PEAR_ERROR_DIE――將這個模式開啟,程序將終結并且將打印錯誤信息??蛇x的,你能定義一個printf()式的字符串,它能夠用于產生信息;首先'%s'在字符串中將替代儲存在錯誤對象中的錯誤信息。

PEAR_ERROR_PRINT――僅僅打印錯誤信息,包括用于PEAR_ERROR_DIE的同樣的可選用的字符串。

PEAR_ERROR_RETURN――當錯誤發生時的一般行為;你能用類提供isError()函數或者PEAR::isError()檢查錯誤。

$db->setErrorhandling(PEAR_ERROR_RETURN)
if(!csv2db::isError(0 < $d = $cd->import("./dat.csv", $dsn, 'address'))) {
if(!csv2db::isError($cd->exportUnvalid("./dat2.csv")) {
} else {
// handle error
}
} else {
// handle error
}



PEAR_ERROR_TRIGGER――這兒函數向PHP運行時錯誤行為一樣。你必須定義哪種錯誤應該發生:E_USER_NOTICE,E_USER_WARNING或者E_USER_ERROR。他們直接和PHP本身產生的信息相對應。請注意,在錯誤信息中錯誤發生的那行(xxx on line yy)指的是在PEAR.php中調用trigger_error的那行――而不是錯誤直接發生的那行。

PEAR_ERROR_CALLBACK――這是只在一個地方處理錯誤并且讓你得代碼不用考慮錯誤處理的最佳方式。它需要一個函數或者類函數來捕獲錯誤,你能寫一個listing 2中展示的那樣的腳本,其中可以看到類相關錯誤對象的好處:import()函數拋出一個CSV2DB_Error給基于CSV的錯誤和一個DB_Error對象給相關于數據庫訪問的錯誤。

Listing 2

$cd = new csv2db();
$cd->setErrorHandling(PEAR_ERROR_CALLBACK, 'handleError');
$dsn = 'mysql://root@localhost/csv2db';
if( 0 < $d = $cd->import("./dat.csv", $dsn, 'address')) {
$cd->exportUnvalid("./dat2.csv");
}

function handleError($error) {
if(DB::isError($error) {
// handle database error
}
if(csv2db::isError($error) {
switch($error->getCode()) {
case FILE_NOT_OPENED :
...
break;
case CORRUPTED_RECORD :
...
break;
}
}
}



單個的錯誤處理

我們有兩種可能的錯誤:我們能夠忽略的錯誤(損壞的記錄),以及使得程序無法運行的錯誤(找不到文件或者打不開數據庫)。如果你在shell腳本中使用類,你可以讓腳本終止于第二類錯誤。

自然的,你可以寫 $cd->setErrorHandling(PEAR_ERROR_DIE)――但是這可能在如果損壞的記錄錯誤發生時導致問題。在這樣的情況下你需要對某個錯誤停用或者替換錯誤處理辦法的可能。解決辦法時expectError(),如果你傳遞一個錯誤代碼給這個函數,指定錯誤的錯誤模式將被單獨于缺省錯誤模式地設置為PEAR_ERROR_RETURN。

expectError()函數把傳遞來的錯誤代碼儲存在棧中,使用popExpected()移出最后傳遞的錯誤代碼。自從PHP 4.3之后你還能使用delExpect()了;這個函數從棧中刪除了指定錯誤代碼的匹配,你不需要關心位置了。

在實際使用中,是這樣的:

$cd->setErrorHandling(PEAR_ERROR_DIE);
...
$cd->expectError(CORRUPTED_RECORD);
$cd->import(...);
$cd->popExpect();



pushErrorHandling() 和 popErrorHandling() 用起來差不多;他們能夠暫時的控制錯誤處理。例如:如果在 exportUnvalid() 中的文件不能打開,你想要忽略錯誤:

PEAR::pushErrorHandling(PEAR_ERROR_RETURN);
$cd->exportUnvalid("./dat2.csv");
PEAR::popErrorHandling();



注意調用方法的區別!expectError()/popExpect()必須作為實體函數來調用――pushErrorHandling和popErrorHandling可以靜態調用。如果作為實體函數,那么他們僅僅影響那個實體。

用戶有很多可能性,這是否又意味著程序員要做很多的工作呢?是,是因為你要比return false做更多的事情;否,是因為PEAR Error API給你完成了很多工作。


一些關于錯誤處理的思考

作為好的程序員,你不應該從你的類的用戶眼中遮掩起確切的錯誤原因。這阻止了簡單的return false的用法;還要注意也可能被PHP自動型別轉換為0――這對于import()函數來說意味著所有的記錄都已經被正確插入了!簡單地終止腳本?,可能對于簡單地PHP shell腳本來說這是可以接受地,但是對于一個web程序來說是一個壞的選擇!而且,在記錄損壞的情況,錯誤是能夠被忽略的。什么不trigger_error()?這是一個可能的選擇,但是有兩個缺點:類的行為取決于php.ini的設置,而且對于類來說這種行為不常見??赡苄阅軌蛐枰妙~外的函數找出錯誤狀態。即使是所有類都提供了非標準的函數名,這還是有問題的,而且類的用戶看上去會忘記那樣的函數調用――正如郵件列表和新聞組中顯示的那樣。怎么辦?讓用戶決定PEAR錯誤處理API。PEAR錯誤系統被廣泛知道并且許多類已經使用了PEAR類;因而我們無論如何必須用PEAR錯誤處理機制――為什么不建立在其上呢?這避免了前面提到的問題并且給用戶提供了很大的可能性。看看Listing 1,它展示了csv2db類和它的錯誤對象的實現。它可能是有些讓人嚇到,但是我們將一行行地瀏覽源代碼。

Listing 1

<?php
require_once 'PEAR.php';
require_once 'DB.php';

define("FILE_NOT_OPENED", 10);
define("CORRUPTED_RECORD", 20);

class csv2db extends PEAR{

var $records=array();
var $unvalid=array();

function csv2db() {
$this->PEAR("CSV2DB_Error");
}

function import($file, $dsn, $table) {
$this->PEAR("CSV2DB_Error");
if($fp=@fopen($file, 'r')) {
while($data=fgetcsv($fp, 1024,';')) {
$this->records[]=$data;
}
fclose($fp);
} else {
return $this->raiseError(null, FILE_NOT_OPENED);
}

$unvalidCount=0;

$storeMode = $GLOBALS['_PEAR_default_error_mode'];
$storeOpts = $GLOBALS['_PEAR_default_error_options'];
$GLOBALS['_PEAR_default_error_mode'] = $this->_default_error_mode;
$GLOBALS['_PEAR_default_error_options'] = $this->_default_error_options;

$db = DB::connect($dsn);

$GLOBALS['_PEAR_default_error_mode']= $storeMode;
$GLOBALS['_PEAR_default_error_options'] = $storeOpts;

if(!DB::isError($db)) {
$db->setErrorHandling($this->_default_error_mode,
$this->_default_error_options);
$qp = $db->prepare("INSERT INTO $table VALUES (?, ?, ?, ?)");
foreach( $this->records as $record) {
if(preg_match('/d{5}/',$record[2])) {
$db->execute($qp, $record);
} else {
$unvalidCount++;
$this->unvalid[]=$record;
$this->raiseError(corrupted record, CORRUPTED_RECORD);
}
}
$db->disconnect();
} else {
return $db;
}
return $unvalidCount;
}

function exportUnvalid($file) {
if($fp=@fopen($file, "w")) {
foreach($this->unvalid as $data) {
fwrite($fp, implode(';', $data)."n", 1024);
}
fclose($fp);
} else {
return $this->raiseError(null,FILE_NOT_OPENED);
}
}

function isError($data) {
return (bool)(is_object($data) &&
(get_class($data) == 'CSV2DB_Error' ||
is_subclass_of($data, 'CSV2DB_Error')));
}
}

class CSV2DB_Error extends PEAR_Error {
var $msgs = array(
FILE_NOT_OPENED =>
array( 'de' =>"Datei konnte nicht ge?ffnet werden",
'en' => "File couldn't be opened"),
CORRUPTED_RECORD =>
array( 'de' =>"fehlerhafter Datensatz",
'en' => "corrupted record")
);

function CSV2DB_Error($message=null, $code = null, $mode = null,
$level = null, $debuginfo = null) {
$this->PEAR_Error(null, $code, $mode, $level, $debuginfo);
}

function getMessage($lang = "en") {
return $this->msgs[$this->code][$lang];
}
}
?>



自己的錯誤對象

有一個自己的錯誤類總是好的,雖然它可能對于這么一個小的類來說是太大的額外負擔――但是這個類僅僅是一個例子并且你從如果沒有錯誤對象需要很多代碼來實現的特性那兒獲益良多。好處是:首先錯誤是直接賦給類的;以及本地化變得更加容易。

類必須從PEAR_Error繼承而來,為的是保持我們的實現簡單,否則PEAR::isError()將不能正常工作。

實現包含了構造函數,其中沒有改變地把參數傳遞給了PEAR_Error地構造函數。

改寫getMessage()函數是提供本地化錯誤信息地關鍵。錯誤定西被定義為類的變量并且將取決于語言動態的賦值。這也將幫助消息聚集于一處――而不是把他們分散于整個主要類的源代碼中。

實現PEAR錯誤處理

你在文章的第一部分看到了我們的類提供了一堆函數――但是他們中的僅僅有四個是直接實現的。所有的相關函數的錯誤處理是由PEAR基類提供的。為了從所有那些錯誤處理特性中獲益,我們必須讓cvs2db類從PEAR基類繼承,也就是:class csv2db extends PEAR。

在前面的錯誤對象段落中,我從對isError()的解釋開始。覆蓋這個方法不是必要的,雖然它確實使得我們能夠直接檢查我們的錯誤類,并且使得錯誤跟蹤更加精確并且可能節約了幾毫秒。

類的構造函數僅僅只是用錯誤類名稱最為參數調用了父類的構造函數。這個調用注冊了我們的錯誤對象并且確保了我們的錯誤類在每次觸發錯誤的時候被使用。


raiseError

在import()和exportUnvalid()的函數體中對raiseError()的使用是值得注意的。這是創建錯誤的關鍵函數;PEAR提供兩個函數用于這個目的:raiseError() 和 throwError()。后一個自從PHP 4.3開始存活在并且是raiseError()的一個簡化變體,兩者行為是一致的;它們的參數在段落 'raiseError 和 throwError' 中描述。


raiseError 和 throwError

原型:

&raiseError( $message, $code, $mode, $options, $userinfo, $errorclass, $skipmessage)
&throwError( $message, $code, $userinfo)

Parameter Description
$message (string) The error message
$code (int) The error number
$mode (constant) Error mode
$options (mixed) Error mode specific parameters
$userinfo (mixed) additional data (ie. Debug information)
$errorclass (string) A class name



可選的你能夠把已經存在的錯誤對象傳遞給這些函數:

&raiseError($error_object)
&throwError($error_object)



如果你從源代碼比較這兩個函數的參數表你將看到類并沒有設置message參數――這是不必須的因為我們在錯誤類中用 getMessage() 函數賦給錯誤信息。而且,調用PEAR構造函數來引入你的錯誤類也是不必要的,你可以在對 raiseError() 調用中指定錯誤類。在腦子中記住這個選項!例如,如果你的類提供了靜態函數或者多于一個錯誤對象,你不能給你的類像我們在csv2db中做得那樣全局地設置它們。

raiseError() 和 throwError() 能夠被靜態地調用以及像 setErrorHandling() 那樣作為實體函數來調用。當你作不作靜態調用地時候做正確地決定是重要的――它直接影響了用戶如何用setErrorHandling()來錯作我們的類。留意 setErrorHandling() 和 raiseError(),這將避免你和你的用戶的頭疼。

從類的這個部分能夠看到全局和局部的錯誤設置和觸發的負面影響。

$storeMode = $GLOBALS['_PEAR_default_error_mode'];
$storeOpts = $GLOBALS['_PEAR_default_error_options'];
$GLOBALS['_PEAR_default_error_mode'] = $this->_default_error_mode;
$GLOBALS['_PEAR_default_error_options'] = $this->_default_error_options;

$db = DB::connect($dsn);

$GLOBALS['_PEAR_default_error_mode'] = $storeMode;
$GLOBALS['_PEAR_default_error_options'] = $storeOpts;



首先,全局的錯誤模式被保存了,然后全局的錯誤模式設置給了局部的錯誤模式并且最后幾行,原來的錯誤模式被還原了。為什么?Connect()是一個靜態函數!它必須使用PEAR::raiseError()。因而假如我們不保存并且還原設置,我們會遇到問題:看看listing 3――如果類在import()函數不能連接到數據庫的時候會發生什么?因為對raiseError()的靜態調用受到全局錯誤模式的影響,而不是局部的$cd->setErrorHandling(...)的影響,腳本終止執行 。實際上push和popErrorHandling()就是設計來用于這樣的任務的――但是PHP中一個現下的bug看上去不幸的組織了它很好的工作。

強制$db對象使用我們的錯誤模式是更舒服的方式,它支持完整的PEAR Error API,這使得代碼能這樣寫:$db->setErrorHandling($this->_default_error_mode, $this->_default_error_options)。兩個實體變量都是由PEAR_Error類提供的。

那行$this->raiseError(corrupted record, CORRUPTED_RECORD)看上去值得注意――而且缺失的返回看上去不順眼。原因是:我們不想在發現損壞的記錄時中止函數執行。你能把這個和觸發一個警告進行比照。唯一的限制時模式PEAR_ERROR_RETURN沒有工作。

Listing 3

<?php
...
PEAR::setErrorHandling(PEAR_ERROR_DIE)

$cd = new csv2db();
$cd->setErrorHandling(PEAR_ERROR_CALLBACK, 'handleError');
$dsn = 'mysql://root@localhost/csv2db';
if( 0 < $d = $cd->import("./dat.csv", $dsn, 'address')) {
$cd->exportUnvalid("./dat2.csv");
}

$db = DB::connect($dsn);
$db->query(...);
...

function handleError($error) {
if(DB::isError($error) {
// handle database error
}
if(csv2db::isError($error) {
switch($error->getCode()) {
case FILE_NOT_OPENED :
...
break;
case CORRUPTED_RECORD :
...
break;
}
}
}
?>




PEAR錯誤處理和PHP 5

因為我們使用函數來創建錯誤,我們沒有考慮在PHP 5中的try/catch/throw機制;raiseMethod和throwError將為你完成這些!對于PHP 5,函數能夠為你的類透明地調用拋出PEAR_Error()――錯誤模式PEAR_ERROR_EXCEPTION能夠用于這個目的。一下的代碼應該能夠在不改變類的情況下用于PHP5中:

<?php
$i = new csv2db();
$dsn = 'mysql://root@localhost/csv2db';
try {
if( 0 < $d = $i->import("./dat.csv", $dsn, 'address')) {
$i->exportUnvalid("./dat2.csv");
}
}
catch CSV2DB_Error {
// fetch the error
}
?>



結論

我希望你大概了解了PEAR錯誤處理,它提供了排除和處理錯誤的強大機制??纯碢EAR手冊[1]的代碼部分并且找出這些函數提供的好處。

Alexander Merz (alexmerz at php dot net) 是PEAR手冊的編輯并且以自由創作者和作家為職業。

鏈接

[1] pear.php.net/manual/en/core.pear.html
上一篇:PHP4之真OO

下一篇:超越模板引擎

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产成人精品一区二区在线| 欧美电影《睫毛膏》| 久久精品国产一区二区三区| 久久综合88中文色鬼| 国产精品美女久久久久久免费| 精品久久久91| 久久中文久久字幕| 欧美片一区二区三区| 成人av在线网址| 亚洲欧美日韩爽爽影院| 久热精品视频在线观看| 国内精品久久影院| 国产精品∨欧美精品v日韩精品| 国模gogo一区二区大胆私拍| 亚洲一区国产精品| 精品国产精品三级精品av网址| 日本一区二三区好的精华液| 欧美日韩xxx| 亚洲第一免费播放区| 91在线免费网站| 国产狼人综合免费视频| 久久久久久久影院| 国产一区二区三区免费视频| 中文字幕视频一区二区在线有码| 亚洲国产精品va在线看黑人| 国产日韩在线视频| www.国产精品一二区| 91国在线精品国内播放| 久久综合久中文字幕青草| 欧美多人爱爱视频网站| 久久视频在线观看免费| 国产福利成人在线| 欧美不卡视频一区发布| 色偷偷av亚洲男人的天堂| 中文日韩在线观看| 奇米影视亚洲狠狠色| 国产日本欧美一区| 久久精品99久久香蕉国产色戒| 日韩欧美中文字幕在线播放| 69久久夜色精品国产7777| 亚洲色图35p| 亚洲国产日韩欧美综合久久| 国产91在线播放九色快色| 国产视频精品久久久| 国产午夜精品全部视频在线播放| 91午夜在线播放| 亚洲sss综合天堂久久| 成人性生交大片免费看视频直播| 日韩女优在线播放| 2019亚洲日韩新视频| 国产精品美女主播在线观看纯欲| 日韩美女视频在线观看| 久久久成人av| 尤物yw午夜国产精品视频| 亚洲精品99久久久久中文字幕| 在线观看中文字幕亚洲| 欧美影院在线播放| 久久久久国产精品一区| 国产99视频精品免视看7| 91精品啪在线观看麻豆免费| 日韩欧美第一页| 日本成人激情视频| 2019精品视频| 国产精品高潮呻吟视频| 91久久嫩草影院一区二区| 97免费中文视频在线观看| 日韩一区av在线| 国产精自产拍久久久久久| 九九热这里只有精品6| 日本在线精品视频| 国产欧美一区二区三区在线| 日韩欧美中文第一页| 8090理伦午夜在线电影| 综合网中文字幕| 在线观看免费高清视频97| 久久久视频免费观看| 久久99精品久久久久久青青91| 国产精品91在线观看| 精品国产精品自拍| 九九久久国产精品| 亚洲精品xxxx| 久久久久女教师免费一区| 国产精品电影久久久久电影网| 国产精品福利在线| 日韩69视频在线观看| 在线看福利67194| 国产91热爆ts人妖在线| 少妇精69xxtheporn| 国产精品视频区1| 日韩在线观看免费全集电视剧网站| 91tv亚洲精品香蕉国产一区7ujn| 国产成人精品999| 午夜精品在线视频| 欧美在线日韩在线| 羞羞色国产精品| 亚洲欧美自拍一区| 欧美专区第一页| 疯狂蹂躏欧美一区二区精品| 亚洲自拍偷拍在线| 美女久久久久久久久久久| 懂色av中文一区二区三区天美| 国产精品成熟老女人| 亚洲一区二区中文字幕| 91精品久久久久久久久不口人| xxxx欧美18另类的高清| 久久久国产精彩视频美女艺术照福利| 在线观看精品国产视频| 美女啪啪无遮挡免费久久网站| 日韩久久免费电影| 日本一本a高清免费不卡| 国产一区二区免费| 欧美午夜女人视频在线| 亚洲精品www久久久| 精品亚洲一区二区| 动漫精品一区二区| 国产精品久久久久一区二区| 亚洲片av在线| 成人精品视频久久久久| 日韩小视频在线| 亚洲成年人影院在线| 777国产偷窥盗摄精品视频| 午夜精品久久久久久久99热浪潮| 国产精品久久久久久久久久三级| 国产精品午夜国产小视频| 欧美巨乳美女视频| 久久五月情影视| 欧美电影在线观看网站| 97精品国产97久久久久久免费| 最近2019中文字幕在线高清| 国产精品久久久久久久久久ktv| 久久人91精品久久久久久不卡| www.欧美精品一二三区| 亚洲伊人久久大香线蕉av| 亚洲国产成人精品电影| 成人写真视频福利网| 欧美小视频在线| 久久精品99久久久久久久久| 欧美—级高清免费播放| 欧美中文字幕第一页| 日韩不卡在线观看| 欧美成人性生活| 欧美激情中文字幕乱码免费| 4438全国成人免费| 亚洲性av在线| 91久久夜色精品国产网站| 国产成人综合一区二区三区| 欧美性猛交xxxxx免费看| 久久久久久91香蕉国产| 中文字幕日韩免费视频| 57pao国产成人免费| 美女999久久久精品视频| 97在线观看免费高清| 日韩欧美主播在线| 久久亚洲精品国产亚洲老地址| 亚洲另类欧美自拍| 亚洲无线码在线一区观看| 国产91在线播放| 国产精品香蕉av| 热久久免费国产视频| 久久久久国产视频| 国产美女91呻吟求| 国外成人性视频| 国产欧美日韩高清| 日韩电影中文字幕|