色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

在ns2中實現網絡協議的方案詳解

454398 ? 來源:博客園 ? 作者: 原來... ? 2020-10-27 14:24 ? 次閱讀

用IE不會有顯示的問題Firefox有的代碼顯示不出來;

這篇文章適合初學者,關于初學者應該參考的文檔:NS by Example、NS2 Beginners Page都有很多實例可以參考。

本文通過實現一個簡單的傳輸協議來說明如何在 ns2 中實現網絡協議,當然,這個協議非常簡單,但是在ns2 中實現協議(不是修改)的流程大體就是這個樣子的了。我們稱這個簡單的協議做: simple_trans 協議,我們一步一步來,把 simple_trans 這個協議慢慢做的復雜。首先我想要明確一個概念:什么是在 ns2 中實現網絡協議,不把這個問題搞明白我們都不知道自己在做什么。網路協議顧名思義網絡上運行的協議,網絡是由關系(無論什么關系)組成的,在這個網絡上運行的規則(無論是優化網絡數據傳輸還是共享網絡信息)就叫做協議,所以我覺得把協議理解為強邏輯的規則是沒有問題的。我們實現一個網絡協議的前提是這個協議被設計出來,所以我們先要想好我們所要實現的協議是要用來做什么事情的;回到 ns2 , ns2 幫我們實現好了一個框架,這個框架給我們提供了數據包初始化,鏈路連接,數據包傳遞路由等功能,也就是說我們只要搭建好我們的邏輯就可以完成協議的模擬了,在 ns2 中我們通過對數據包類型、發送數據包邏輯等等進行控制。這就好比于 ns2 給我們提供了一個鐵路網,火車需要的電也有了,火車不夠了還可以生產,我們在 ns2 中實現協議就是要對火車進行調度,何時到站,到站后如何運行等等就是協議的內容。

下面就從我們的 simple_trans開始 說起,在這個協議里,首先我們要實現的任務非常簡單,簡單到什么程度了呢,簡單到這個協議就是 a 節點對 b 節點說一句話:“ hi!, I’m a ”。不要笑,這也是一個協議。要在 ns2 上完成這個任務,我們首先要給 simple_trans 這個協議起個名字使得 ns2 可以發出這個協議的數據包并且認得這個協議發出的數據包,現在開始就是第一步了。

1, 在 NS_HOME/common/packet.h 的 enum packet中 加入協議數據包名稱 PT_SIMPLE_TRANS_PACKET(必須的,注意不要加錯地方,最好加在倒數第二的地方),在 class p_info 中加入name_[PT_SIMPLE_TRANS_PACKET] = “simple_trans_packet” (非必須的)。 Packet.cc 就不要動了。

2, 為了我們協議的獨立性、好看性,我們在 NS_HOME 根目錄下創建一個文件夾,我就叫他 kgn ,好在 kgn目錄(也就是 NS_HOME/kgn )目錄下給協議的主角: simple_trans.h&simple_trans.cc 。兩個空文件沒什么用,下面我們添加協議內容。

3, simple_trans.h 內容:

view plain

#ifndef ns_simple_trans_h

#define ns_simple_trans_h

#include “agent.h”

#include “tclcl.h”

#include “packet.h”

#include “address.h”

#include “ip.h”

#define PROTOCOL_DEFAULT_PORT 1023

#define PROTOCOL_INIT_SYN 1

首先我們引用一些需要用到的頭文件,然后我們定義了兩個宏,第一個是我們 simple_trans 協議默認傳輸的端口(這方面如果有所疑問請參考這里 ),第二個是我們僅有的一條指令:同步指令(類似于 TCP 協議中三次握手的第一步,事實上我們的這個協議最終就是要實現一個簡化版的三步握手)。繼續看:

view plain

struct hdr_simple_trans {

int type;

static int offset_;

inline static int& offset() {

return offset_;

}

inline static hdr_simple_trans * access(const Packet * p) {

return (hdr_simple_trans*) p-》access(offset_);

}

};

這些可以當做領導講話的開頭部分內容。就是定義一個我們協議的頭所包括的內容,只有 type 這個是我定義的,其他的內容是 ns2 系統需要的。再繼續:

view plain

class simple_trans_agent : public Agent {

public :

simple_trans_agent();

virtual void recv(Packet *, Handler *);

void send_simple_msg(int type, int target);

int get_target(){ return simple_target; }

protected:

int simple_target;

int simple_port;

int command(int argc, const char*const*argv);

};

這里就是我們定義的負責“調度火車”的功能的類了。繼承的是 agent 類,在 ns2 中,這個 agent 不可小覷,他是我們可以產生數據包、發送數據包、接收數據包的地方,包括的 target 變量就是數據包發送給的下一個目標。recv 函數會在仿真的過程中“自動”的收到網絡上傳輸的數據包(更深層次的是經過了地址和端口過濾器);send_simple_msg 函數用來執行創建并發送數據的功能; get_target 就不用解釋了(接口保護)。接下來是我們在協議制定過程中經常會用到的 timer 的定義, timer 顧名思義是一個定時器(鬧鐘)在到時時候會調用一個expire (超時)函數,這個被執行的超時函數的內容就是我們所感興趣的,因為通過 timer 我們可以實現很多邏輯。

view plain

class SYNTimer : public TimerHandler {

public:

SYNTimer(simple_trans_agent* t) : TimerHandler(), t_(t) {

}

inline virtual void expire(Event *);

protected:

simple_trans_agent* t_;

};

我們只要實現 expire 函數即可, timer 的初始和使用見 simpe_trans.cc 文件:

view plain

SYNTimer *syn_timer = new SYNTimer(this);

syn_timer-》resched(1.00);

resched 用來給“鬧鐘上弦”。

view plain

void SYNTimer::expire(Event *){

t_-》send_simple_msg(PROTOCOL_INIT_SYN, t_-》get_target());

this-》resched(1.00);

}

expire 可以實現我們的“理想”了,譬如,我們到時了就發送我們的 SYN 信息給我們的目標節點(目標節點通過tcl 文件定義,下文中我們會見到)。

4, simple_trans.cc 內容:

view plain

int hdr_simple_trans::offset_;

static class simple_transHeaderClass : public PacketHeaderClass {

public:

simple_transHeaderClass() : PacketHeaderClass(“PacketHeader/simple_trans”,sizeof(hdr_simple_trans)) {

bind_offset(&hdr_simple_trans::offset_);

}

} class_simple_transhdr;

static class simple_transClass : public TclClass {

public:

simple_transClass() : TclClass(“Agent/simple_trans”) {}

TclObject* create(int, const char*const*) {

return (new simple_trans_agent());

}

} class_simple_trans;

simple_trans_agent::simple_trans_agent() : Agent(PT_SIMPLE_TRANS_PACKET),

simple_target(-1), simple_port(PROTOCOL_DEFAULT_PORT) {

bind(“simple_target_”, &simple_target);

bind(“simple_port_”, &simple_port);

}

這個又是八股文,前面幾個類照葫蘆畫瓢即可,如果想要理解是什么意思可以參考我的文章 ,最后一個我們bind 了幾個變量,這幾個變量通過綁定就可意思讓我們通過 tcl 腳本方便的改變他們的值了(不需要重新編譯c++ 文件)。

5, 在這個文件中我們主要注意這么幾點:

a) 在 send_simple_msg 中數據包的生成 Packet* pkt = allocpkt() ;

b) 數據包的訪問: hdr_ip *iph = hdr_ip::access(pkt) ;

c) 數據包 ip 地址和端口號的設定(從這里我們可以看出實現的是一個應用層協議);

d) 發送數據包 send( pkt, 0 ) ,我們可以不用去管 0 是什么意思;

e) Command 命令中不要忘記 return (TCL_OK) 這句話,否則會出錯的。

f) 在 recv 函數中實現我們的簡單邏輯:顯示出我們收到了來自對方的一個 simple_trans 的數據包。

6, 看我們這兩個宏命令:

view plain

#define NOW Scheduler::instance().clock()

#define MYNODE Address::instance().get_nodeaddr(addr())

這兩個命令給我們編程提供幫助,分別顯示系統時間和得到當前節點的地址,也許以后我們會用得著。

7, 在 tcl 腳本中我們需要使用我們的 simple_trans 協議:

view plain

set sT1 [new Agent/simple_trans]

$sT1 set-target [AddrParams addr2id [$n1 node-addr]]

$n0 attach $sT1 1023

set sT2 [new Agent/simple_trans]

$n1 attach $sT2 1023

。.. 。..

$ns at 1.0 “$sT1 begin”

在 tcl 中 new 一個對象,比如 sT1 之后我們要將其 attach 到所屬的節點上,注意最后一個 1023 ,這是我們attach 到節點上的給我們 simple_trans 協議分配的端口(深層次的意思是端口分類器會把目的端口是 1023 的數據包分給 sT1 )。 begin 方法是在 command 中實現的,回過頭到 simple_trans.cc 中可以看到他的意思,我們可以好好理解一下 command 中函數和 tcl 中的使用關系。

8, 最后一步,就是編譯我們整個協議將其鍵入到 ns 中了,編譯前我們要修改 makefile 文件,由于我們是在NS_HOME/kgn 目錄中所以, makefile 需要修改的有兩個地方:在 INCLUDES = 中加入 -I./kgn ,加入這個的好處就是我們在其他目錄使用 simple_trans.h 的時候不用將 kgn 次級目錄包含進去;在 OBJ_CC = 中加入 kgn/simple_trans.o / 。好了大功告成,下面回到 NS_HOME 目錄下 make 一下,如果成功,我們執行一下我們的 tcl 腳本,看看是不是真的可以運行了呢。

小結:到了這里我們已經添加了一個簡單的協議了,好了,有的人會說了,這么簡單的協議有什么用呢?那好,我們想一想我們有什么可以改進的嗎?以上的協議我們叫做 simpe_trans 協議 0.1 版,那么我們看看 0.2 版給我帶來了什么新的變化。

ACK timer

首先要做的就是協議的復雜化,我們將協議改為三次握手過程如圖所示:

這個過程對應以下代碼(修改simple_trans.h):

view plain

#define PROTOCOL_INIT_SYN 1

#define PROTOCOL_INIT_SYN_ACK 2

#define PROTOCOL_INIT_ACK 3

#define INTERVAL 0.3

class simple_trans_agent;

enum simple_state{

CLOSED,

SYN_SENT,

SYN_RCVD,

ESTABLISHED

};

其中 C-》CLOSED , SS-》SYN_SENT , SR-》SYN_RCVD , E-》ESTABLISHED 為節點可能處于的狀態在發送或接受 SYN 和發送 SYN-ACK 接受 ACK 后的變化,而兩個 timer 的作用就是使得沒有正確到底目的地的數據包可以被重新發送,當然這些 timer 需要在適當的時機取消比如: ack_timer-》cancel() ,取消 timer 使用 cancel 函數即可。具體代碼實現參考 0.2 版本的代碼。那么現在我們重新 make 編譯我們的程序,我們會發現兩個節點可以通過三次握手建立起來一個簡單的鏈接了,可以說我們在有這個簡單的可以建立連接的程序之后我們馬上想到是不是還可以發送數據呢,在 ns2 中,數據的發送,我們常見的如 CBR 或者 FTP ,都可以發送數據但是他們之間有很大的不同 CBR 使用的是 trafficgenerater ,而 FTP 可以看成是一個帶發送數據包的 agent ,現在為了讓我們的 simple_trans 協議可以在建立起連接以后發送數據,我們就有了兩種選擇,是繼承 trafficgenerater 成為數據發送源呢,還是類似 FTP 使用 agent 發送數據,考慮到我們協議的簡潔易懂性,我們直接使用一個 timer ,在每次 timer 到時的時候都利用 simple_trans 的 send 函數發送一個具有 PROTOCOL_DATA 類型(標識是一個數據)的包給通信對端( CN )。在 sendmsg 函數中的實現如下:

view plain

hdr_rtp* rh = hdr_rtp::access(p);

hdr_simple_trans *shdr = hdr_simple_trans::access(p);

hdr_ip* ih = hdr_ip::access(p);

double local_time = NOW;

hdr_cmn::access(p)-》size() = size;

hdr_cmn::access(p)-》timestamp() =

(u_int32_t) (SAMPLERATE * local_time);

rh-》seqno() = seqno++;

ih-》daddr() = simple_target;

ih-》dport() = simple_port;

ih-》saddr() = MYNODE;

shdr-》type = PROTOCOL_DATA;

target_-》recv(p);

這里面我們還可以通過 RTP 協議給每一個包設置序列號,當然也可以在 hdr_simple_trans 中添加一個 seq 的屬性。當然我們的協議升級到 0.3 版本后的變化并不只是有這些而已。我們還將 simple_trans 協議的數據包的大小以及發送頻率設置成可變的等,具體可以參考 0.3 的代碼。

小結:通過以上的設計,我們初步有了一個可以建立連接并發送數據的協議,什么?像是 SIP 協議,沒錯我們也可以將我們的程序叫做一個簡單的會話發起協議,當然你可以實現的更加復雜。至此,我們在 ns2 中添加一個基本網絡協議的事情已經完成了,我們注意到:不同的協議使用節點上的不同的端口,這樣的協議是不能夠影響到諸如路由、無線鏈路等協議的結果的,所以并不是所有的 ns2 中的協議都可以這么添加,我們還可以修改節點數據結構等方法添加我們自己的一些修改進 ns2 達到仿真的目的,所以這篇文章的目的還是介紹如何在 ns2 中實現協議的基礎,我們要根據我們自己的仿真需要來設計我們的程序。通過以上的介紹我們應該掌握的是在 ns2 中發送數據的方法、 ns2 中 timer 的使用方法等等技巧。下面我介紹一個比較有意思的利用我們的 simple_trans 做的協議修改實驗:添加無線節點丟包模型,在這里主要參考的是柯志亨老師的實現方法,但是在丟包方面我這里做的對原有協議破壞性更多(更不合理吧),我們將演示當兩個無線節點距離增大的時候會丟失數據包并且我們的ACKTimer 以及 SYNTimer 的作用。好,下面就是如何修改的過程了:

在 NS_HOME/mac 目錄下的 wireless-phy.cc 的 380 行左右,我們添加如下代碼:

view plain

//error model.

hdr_cmn *hdr_err = HDR_CMN(p);

hdr_simple_trans *sh = hdr_simple_trans::access(p);

double ratio = Pr/RXThresh_;

double std = error_modle_lf(ratio);

//printf(“wireless-phy model receive packet ratio=%lf std=%lf/n”,ratio,std);

if (hdr_err-》ptype() == PT_SIMPLE_TRANS_PACKET){

if (!sh-》error){

double tmp=((double)rand())/RAND_MAX;

if (tmp》std){

sh-》error = false;

}else{

sh-》error = true;

//printf(“wireless-phy error model set the packet error/n”);

}

}

}

//end of error model.

我們修改的是 WirelessPhy::sendUp(Packet *p) 函數,在發送數據包之前我們檢查數據包中 simple_trans 協議的數據包,并將該數據包中在 hdr_simple_trans 中定義好的 error 屬性置為 true (說明這個數據包出錯),實現數據包出錯分布的函數 error_modle_lf ,這是一個拉格朗日差值函數的實現:

view plain

double error_modle_lf(double ratio){

if(ratio 》1.5)return 0;

double x[6] = {1,1.1,1.2,1.3,1.4,1.5};

double y[6] = {1,0.5,0.3,0.1,0.02,0};

double res = 0;

for(int i = 0; i 《 6; i++){

double temp = 1;

double temp1 = 1;

for(int j = 0; j 《 6; j++){

if(i == j)continue;

temp *= (x[i] - x[j]);

temp1 *= (ratio - x[j]);

}

res += (temp1 / temp) * y[i];

}

return res;

}

顯然,我們設計的是無線節點離基站越遠對包個數越多。

1, 在 simple_trans.cc 中添加 if( shdr-》error )return ,這樣錯誤的包我們就“裝作”收不到了

2, 這里補充說明,柯志亨老師的錯誤模型實現是基于無線層的,出錯了就真的不發或者重發,而我的實現可以說是假的,還會造成無線網絡的吞吐,但是還是可以演示無線丟包情況的,具體結果可以編譯我稱作 0.4 版本的程序運行。可以將包的序列號畫出來,這樣會更形象的展現丟包情況。

3, 我們將包序列號、包收到時間等等信息都通過 printf 函數打印出來,這樣我們就可以不用去考慮如何通過trace 文件來分析得到數據,這種方法有的時候更加有效,我們不必去了解 trace 機制,這也算是一個捷徑了。

總結:

Ns2 作為一個在科研領域應用廣泛的仿真器有著其內在的很多優勢的:開源協議修改自如、分裂設計可設計不同的仿真場景而不需要修改協議代碼,但是,我們在做網絡協議的研究的時候往往會發現 ns2 現有的協議不足以完成我們的仿真,這是就需要自己設計協議或者修改現有的協議,所以通過對這個簡單的 simple_trans 協議的實現我們可以更加的有的放矢,知道如何在那里修改 ns2 的協議,雖然 simple_trans 只是一個超級笨的協議,但是它已經展現了基本的協議設計技巧:集成 agent 、 timer 的使用、協議包頭設計等等。如果我們能夠再將 ns2有線無線節點結構、路由模塊、無線 mac 等等這些代碼仔細研讀,那么到時你就會發現在 ns2 上面實現一個協議倒不是難事,反而是在協議自身的設計上,這就和我們高級程序語言一樣,語言的學習不是難事,而真正熟練的利用語言解決問題才是我們的學習目標。
編輯:hfy

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 仿真器
    +關注

    關注

    14

    文章

    1017

    瀏覽量

    83722
  • 網絡協議
    +關注

    關注

    3

    文章

    267

    瀏覽量

    21534
  • NS2
    NS2
    +關注

    關注

    4

    文章

    10

    瀏覽量

    12179
收藏 人收藏

    評論

    相關推薦

    LoRaWAN網絡智慧水務的創新解決方案

    隨著城市化的不斷發展,對水資源的高效管理變得愈發重要。LoRaWAN(低功耗廣域網)網絡作為一種適用于長距離、低功耗的通信解決方案,正日益智慧水務領域展現其獨特的創新應用。本文將探討LoRaWAN
    的頭像 發表于 12-20 16:17 ?85次閱讀
    LoRaWAN<b class='flag-5'>網絡</b><b class='flag-5'>在</b>智慧水務<b class='flag-5'>中</b>的創新解決<b class='flag-5'>方案</b>

    mtu不同網絡協議的應用

    現代網絡通信中,數據包的傳輸效率和可靠性是衡量網絡性能的關鍵指標。MTU作為網絡協議的一個重
    的頭像 發表于 12-16 14:22 ?288次閱讀

    dap協議跨鏈技術的應用

    和通信協議實現不同區塊鏈網絡之間的無縫連接。 一、DAP協議概述 DAP協議是一種去中心化應用協議
    的頭像 發表于 11-22 15:45 ?215次閱讀

    dap協議的基本概念 dap協議區塊鏈的應用

    DAP協議,即分布式應用協議(Distributed Application Protocol),是一種旨在促進去中心化應用(DApps)區塊鏈網絡上的構建和運行的框架。DAP
    的頭像 發表于 11-22 15:39 ?237次閱讀

    ipc協議物聯網的應用

    聯網概述 物聯網是一個由互聯網、傳統電信網、傳感器網絡等多種網絡組成的網絡,它允許物體與物體、物體與人、人與人之間的智能互聯。物聯網的核心在于數據的收集、傳輸、處理和應用,而通信協議
    的頭像 發表于 11-15 14:19 ?297次閱讀

    受電端Type-C設計,PD協議的純硬件實現詳解

    受電端Type-C設計,PD協議的純硬件實現詳解 USB Type-C口電源主要有以下三個角色定義,分別為: ·SOURCE 純供電方
    的頭像 發表于 11-13 11:01 ?315次閱讀

    打破網絡邊界:P2Link助力實現高效遠程訪問與內網穿透

    ,解決內網穿透難題,讓用戶輕松實現遠程訪問和管理。以下是一些典型的應用場景: 遠程辦公與文件訪問: 遠程辦公場景,員工常常需要從外部網絡訪問公司內網
    發表于 10-31 11:54

    華納云:探討可用于降低服務器網絡延遲的先進的網絡協議

    網絡延遲是影響在線服務性能的重要因素之一,尤其是實時應用和高交互性網站。通過采用更高級別的網絡協議,可以有效降低
    的頭像 發表于 09-30 15:14 ?215次閱讀

    NDP協議是怎樣幫助IPv6實現網絡安全運行的?

    復雜多變的網絡環境,確保各節點之間能夠高效、準確地發現與通信,是構建穩定、可擴展網絡架構的基石。那么IPv6是依靠什么實現
    的頭像 發表于 09-10 10:19 ?379次閱讀
    NDP<b class='flag-5'>協議</b>是怎樣幫助IPv6<b class='flag-5'>實現</b><b class='flag-5'>網絡</b>安全運行的?

    Linux網絡協議棧的實現

    網絡協議棧是操作系統核心的一個重要組成部分,負責管理網絡通信中的數據包處理。 Linux 操作系統
    的頭像 發表于 09-10 09:51 ?301次閱讀
    Linux<b class='flag-5'>網絡</b><b class='flag-5'>協議</b>棧的<b class='flag-5'>實現</b>

    LwIP協議棧源碼詳解—TCP/IP協議實現

    電子發燒友網站提供《LwIP協議棧源碼詳解—TCP/IP協議實現.pdf》資料免費下載
    發表于 07-03 11:22 ?3次下載

    RA MCU CANFDFSP的配置詳解

    RA MCU CANFDFSP的配置詳解
    的頭像 發表于 06-19 08:06 ?518次閱讀
    RA MCU CANFD<b class='flag-5'>在</b>FSP<b class='flag-5'>中</b>的配置<b class='flag-5'>詳解</b>

    工業網絡通訊協議有哪些

    隨著工業自動化的不斷發展,設備之間的互聯互通和數據交換變得越來越重要。工業網絡通訊協議作為實現這一功能的關鍵技術,其工業自動化系統扮演著
    的頭像 發表于 06-06 18:02 ?1643次閱讀

    網絡傳輸協議有幾種?

    協議)、TCP(傳輸控制協議)、UDP(用戶數據報協議)、ICMP(互聯網控制報文協議)等。這些協議負責數據傳輸、路由選擇、數據包封裝和拆封
    的頭像 發表于 04-02 16:04 ?1428次閱讀

    消防預警系統Modbus協議和EthernetIP協議都發揮著重要的作用

    技術,可以用于傳輸IP數據包。消防預警系統,Ethernet/IP協議也發揮著重要的作用。它可以實現設備之間的實時通信和數據交換,將各個傳感器、控制器、報警器等設備連接到
    發表于 01-02 19:34
    主站蜘蛛池模板: 免费果冻传媒2021在线观看| 福利一区福利二区| bbwvideos欧美老妇| 国模大胆一区二区三区| 日本一二三区视频在线| 99久久久无码国产精品免费人妻 | 亚洲精品国偷拍电影自产在线| 成人性视频全过程| 日日碰狠狠躁久久躁综合网| 吃奶吸咪咪动态图| 少妇高潮A视频| 国产精品AV色欲蜜臀在线| 少妇大荫蒂毛多毛大| 国产互换后人妻的疯狂VIDEO| 天天躁躁水汪汪人碰人| 国产欧美日韩网站| 亚洲欧洲日韩视频在钱| 久久精品亚洲热综合一本奇米| 伊在香蕉国产在线视频| 麻豆啊传媒app黄版破解免费| 99国产在线视频| 日韩毛片大全| 国产在线精品亚洲第一区| 一本大道熟女人妻中文字幕在线| 久久成人免费大片| 92午夜免费福利757| 青青伊人精品| 国产午夜精品理论片影院| 一本道中文无码亚洲| 免费看www视频| 大桥未久与黑人中出视频| 先锋影音av无码第1页| 久久99re6国产在线播放| 777久久人妻少妇嫩草AV| 日韩午夜影院| 护士WC女子撒尿| 99re在线播放| 香蕉久久av一区二区三区| 久久热最新网站获取3| yellow在线中文| 亚洲精品免费视频|