日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

iOS獲取設備唯一標識的實現步驟

瀏覽:220日期:2022-09-16 16:52:09
1. 常用的UUID

UDID是一個40位十六進制序列(越獄的設備通過某些工具可以改變設備的 UDID),移動網絡可以利用 UDID 來識別移動設備。

許多開發者把 UDID 跟用戶的個人信息關聯起來,網絡窺探者會從多個應用收集這些數據,然后順藤摸瓜得到這個人的許多隱私數據,同時大部分應用確實在頻繁傳輸 UDID 和私人信息。 為了避免集體訴訟,蘋果最終決定在 iOS 5 的時候,將這一慣例廢除。

獲取UUID的方法:

/** 卸載應用重新安裝后會不一致*/+ (NSString *)getUUID{ return [UIDevice currentDevice].identifierForVendor.UUIDString;;}2. MAC 地址

MAC地址,用來表示互聯網上每一個站點的標示符,是一個六個字節(48位)的十六進制序列。前三個字節是由 IEEE 的注冊管理機構 RA 負責給不同廠家分配的”編制上唯一的標示符,后三個字節由各廠家自行指派給生產的適配器接口。

MAC 地址在網絡上用來區分設備的唯一性,接入網絡的設備都有一個MAC地址,他們肯定都是唯一的。一部 iPhone 上可能有多個 MAC 地址,包括 WIFI 的、SIM 的等,但是 iTouch 和 iPad 上就有一個 WIFI 的,因此只需獲取 WIFI 的 MAC 地址就好了。一般會采取 MD5(MAC 地址 + bundleID)獲取唯一標識。

但是 MAC 地址和 UDID 一樣,存在隱私問題, iOS 7 之后,所有設備請求 MAC 地址會返回一個固定值,這個方法也不攻自破了。

獲取MAC在github找到一個挺好的方法:

2.1 首先導入下面幾個庫:

iOS獲取設備唯一標識的實現步驟

2.2 新建一個文件,繼承NSObject,在.m文件導入頭文件,以及定義一些宏

#import 'XWGetMAC.h'#import <ifaddrs.h>#import <resolv.h>#import <arpa/inet.h>#import <net/if.h>#import <netdb.h>#import <netinet/ip.h>#import <net/ethernet.h>#import <net/if_dl.h>#define MDNS_PORT 5353#define QUERY_NAME '_apple-mobdev2._tcp.local'#define DUMMY_MAC_ADDR @'02:00:00:00:00:00'#define IOS_CELLULAR @'pdp_ip0'#define IOS_WIFI@'en0'#define IOS_VPN @'utun0'#define IP_ADDR_IPv4 @'ipv4'#define IP_ADDR_IPv6 @'ipv6'

+ (NSString *)getMAC:(BOOL)preferIPv4 {return [[XWGetMAC alloc] getIPAddress:preferIPv4];}/* * 獲取設備當前網絡IP地址 */- (NSString *)getIPAddress:(BOOL)preferIPv4{ NSArray *searchArray = preferIPv4 ? @[ IOS_VPN @'/' IP_ADDR_IPv4, IOS_VPN @'/' IP_ADDR_IPv6, IOS_WIFI @'/' IP_ADDR_IPv4, IOS_WIFI @'/' IP_ADDR_IPv6, IOS_CELLULAR @'/' IP_ADDR_IPv4, IOS_CELLULAR @'/' IP_ADDR_IPv6 ] : @[ IOS_VPN @'/' IP_ADDR_IPv6, IOS_VPN @'/' IP_ADDR_IPv4, IOS_WIFI @'/' IP_ADDR_IPv6, IOS_WIFI @'/' IP_ADDR_IPv4, IOS_CELLULAR @'/' IP_ADDR_IPv6, IOS_CELLULAR @'/' IP_ADDR_IPv4 ] ;NSDictionary *addresses = [self getIPAddr];__block NSString *address; [searchArray enumerateObjectsUsingBlock:^(NSString *key, NSUInteger idx, BOOL * _Nonnull stop) {address = addresses[key];//篩選出IP地址格式if([self isValidatIP:address]) *stop = YES; }]; return address ? address : @'0.0.0.0';}- (BOOL)isValidatIP:(NSString *)ipAddress { if (ipAddress.length == 0) {return NO; } NSString *urlRegEx = @'^([01]?dd?|2[0-4]d|25[0-5]).' '([01]?dd?|2[0-4]d|25[0-5]).' '([01]?dd?|2[0-4]d|25[0-5]).' '([01]?dd?|2[0-4]d|25[0-5])$';NSError *error; NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:urlRegEx options:0 error:&error];if (regex != nil) {NSTextCheckingResult *firstMatch=[regex firstMatchInString:ipAddress options:0 range:NSMakeRange(0, [ipAddress length])];return firstMatch; } return NO;}- (NSDictionary *)getIPAddr{ NSMutableDictionary *addresses = [NSMutableDictionary dictionaryWithCapacity:8];// retrieve the current interfaces - returns 0 on success struct ifaddrs *interfaces; if(!getifaddrs(&interfaces)) {// Loop through linked list of interfacesstruct ifaddrs *interface;for(interface=interfaces; interface; interface=interface->ifa_next) { if(!(interface->ifa_flags & IFF_UP) /* || (interface->ifa_flags & IFF_LOOPBACK) */ ) {continue; // deeply nested code harder to read } const struct sockaddr_in *addr = (const struct sockaddr_in*)interface->ifa_addr; char addrBuf[ MAX(INET_ADDRSTRLEN, INET6_ADDRSTRLEN) ]; if(addr && (addr->sin_family==AF_INET || addr->sin_family==AF_INET6)) {NSString *name = [NSString stringWithUTF8String:interface->ifa_name];NSString *type;if(addr->sin_family == AF_INET) { if(inet_ntop(AF_INET, &addr->sin_addr, addrBuf, INET_ADDRSTRLEN)) {type = IP_ADDR_IPv4; }} else { const struct sockaddr_in6 *addr6 = (const struct sockaddr_in6*)interface->ifa_addr; if(inet_ntop(AF_INET6, &addr6->sin6_addr, addrBuf, INET6_ADDRSTRLEN)) {type = IP_ADDR_IPv6; }}if(type) { NSString *key = [NSString stringWithFormat:@'%@/%@', name, type]; addresses[key] = [NSString stringWithUTF8String:addrBuf];} }}// Free memoryfreeifaddrs(interfaces); } return [addresses count] ? addresses : nil;}

/* * 獲取設備物理地址 */- (nullable NSString *)getMacAddress { res_9_init(); int len; //get currnet ip address NSString *ip = [self currentIPAddressOf:IOS_WIFI]; if(ip == nil) {fprintf(stderr, 'could not get current IP address of en0n');return DUMMY_MAC_ADDR; }//end if//set port and destination _res.nsaddr_list[0].sin_family = AF_INET; _res.nsaddr_list[0].sin_port = htons(MDNS_PORT); _res.nsaddr_list[0].sin_addr.s_addr = [self IPv4Pton:ip]; _res.nscount = 1;unsigned char response[NS_PACKETSZ]; //send mdns query if((len = res_9_query(QUERY_NAME, ns_c_in, ns_t_ptr, response, sizeof(response))) < 0) {fprintf(stderr, 'res_search(): %sn', hstrerror(h_errno));return DUMMY_MAC_ADDR; }//end if//parse mdns message ns_msg handle; if(ns_initparse(response, len, &handle) < 0) {fprintf(stderr, 'ns_initparse(): %sn', hstrerror(h_errno));return DUMMY_MAC_ADDR; }//end if//get answer length len = ns_msg_count(handle, ns_s_an); if(len < 0) {fprintf(stderr, 'ns_msg_count return zeron');return DUMMY_MAC_ADDR; }//end if//try to get mac address from data NSString *macAddress = nil; for(int i = 0 ; i < len ; i++) {ns_rr rr;ns_parserr(&handle, ns_s_an, 0, &rr);if(ns_rr_class(rr) == ns_c_in && ns_rr_type(rr) == ns_t_ptr && !strcmp(ns_rr_name(rr), QUERY_NAME)) { char *ptr = (char *)(ns_rr_rdata(rr) + 1); int l = (int)strcspn(ptr, '@');char *tmp = calloc(l + 1, sizeof(char)); if(!tmp) {perror('calloc()');continue; }//end if memcpy(tmp, ptr, l); macAddress = [NSString stringWithUTF8String:tmp]; free(tmp);}//end if }//end for each macAddress = macAddress ? macAddress : DUMMY_MAC_ADDR; return macAddress;}//end getMacAddressFromMDNS- (nonnull NSString *)currentIPAddressOf: (nonnull NSString *)device { struct ifaddrs *addrs; NSString *ipAddress = nil;if(getifaddrs(&addrs) != 0) {return nil; }//end if//get ipv4 address for(struct ifaddrs *addr = addrs ; addr ; addr = addr->ifa_next) {if(!strcmp(addr->ifa_name, [device UTF8String])) { if(addr->ifa_addr) {struct sockaddr_in *in_addr = (struct sockaddr_in *)addr->ifa_addr;if(in_addr->sin_family == AF_INET) { ipAddress = [self IPv4Ntop:in_addr->sin_addr.s_addr]; break;}//end if }//end if}//end if }//end forfreeifaddrs(addrs); return ipAddress;}//end currentIPAddressOf:- (nullable NSString *)IPv4Ntop: (in_addr_t)addr { char buffer[INET_ADDRSTRLEN] = {0}; return inet_ntop(AF_INET, &addr, buffer, sizeof(buffer)) ? [NSString stringWithUTF8String:buffer] : nil;}//end IPv4Ntop:- (in_addr_t)IPv4Pton: (nonnull NSString *)IPAddr { in_addr_t network = INADDR_NONE; return inet_pton(AF_INET, [IPAddr UTF8String], &network) == 1 ? network : INADDR_NONE;}//end IPv4Pton:

如果出現 “_res_9_ninit', referenced from:”這種報錯,是因為沒有添加步驟1的幾個庫

3.UUID+自己存儲3.1 獲取UUID的兩個方法

/** 卸載應用重新安裝后會不一致*/+ (NSString *)getUUID{ CFUUIDRef uuid = CFUUIDCreate(NULL); NSString *UUID = (__bridge_transfer NSString *)CFUUIDCreateString(NULL, uuid); CFRelease(uuid); return UUID;} /** 卸載應用重新安裝后會不一致*/+ (NSString *)getUUID{ return [UIDevice currentDevice].identifierForVendor.UUIDString;;}

很明顯UUID已經不足以支持設備的唯一性了,目前很多App都有新用戶的優惠,但是又要保證每臺設備綁定一個賬戶,如果單純使用UUID的話已經滿足不了這個需求,所以,這里需要用keychain保存,這樣即使卸載app在安裝,獲取到的UUID也是唯一性的。

3.2 首先在項目中添加 KeyChain Sharing

iOS獲取設備唯一標識的實現步驟

3.3 導入第三方庫 Security.framework

iOS獲取設備唯一標識的實現步驟

3.4 核心代碼(代碼有點多)

在github搜索SSKeychain可以找到,只要 SSKeychain.h 和 SSKeychain.m 文件即可

#import <Foundation/Foundation.h>#import <Security/Security.h>/** Error codes that can be returned in NSError objects. */typedef enum {/** No error. */SSKeychainErrorNone = noErr,/** Some of the arguments were invalid. */SSKeychainErrorBadArguments = -1001,/** There was no password. */SSKeychainErrorNoPassword = -1002,/** One or more parameters passed internally were not valid. */SSKeychainErrorInvalidParameter = errSecParam,/** Failed to allocate memory. */SSKeychainErrorFailedToAllocated = errSecAllocate,/** No trust results are available. */SSKeychainErrorNotAvailable = errSecNotAvailable,/** Authorization/Authentication failed. */SSKeychainErrorAuthorizationFailed = errSecAuthFailed,/** The item already exists. */SSKeychainErrorDuplicatedItem = errSecDuplicateItem,/** The item cannot be found.*/SSKeychainErrorNotFound = errSecItemNotFound,/** Interaction with the Security Server is not allowed. */SSKeychainErrorInteractionNotAllowed = errSecInteractionNotAllowed,/** Unable to decode the provided data. */SSKeychainErrorFailedToDecode = errSecDecode} SSKeychainErrorCode;extern NSString *const kSSKeychainErrorDomain;/** Account name. */extern NSString *const kSSKeychainAccountKey;/** Time the item was created. The value will be a string. */extern NSString *const kSSKeychainCreatedAtKey;/** Item class. */extern NSString *const kSSKeychainClassKey;/** Item description. */extern NSString *const kSSKeychainDescriptionKey;/** Item label. */extern NSString *const kSSKeychainLabelKey;/** Time the item was last modified. The value will be a string. */extern NSString *const kSSKeychainLastModifiedKey;/** Where the item was created. */extern NSString *const kSSKeychainWhereKey;/** Simple wrapper for accessing accounts, getting passwords, setting passwords, and deleting passwords using the system Keychain on Mac OS X and iOS. This was originally inspired by EMKeychain and SDKeychain (both of which are now gone). Thanks to the authors. SSKeychain has since switched to a simpler implementation that was abstracted from [SSToolkit](http://sstoolk.it). */@interface SSKeychain : NSObject///-----------------------/// @name Getting Accounts///-----------------------/** Returns an array containing the Keychain’s accounts, or `nil` if the Keychain has no accounts. See the `NSString` constants declared in SSKeychain.h for a list of keys that can be used when accessing the dictionaries returned by this method. @return An array of dictionaries containing the Keychain’s accounts, or `nil` if the Keychain doesn’t have any accounts. The order of the objects in the array isn’t defined. @see allAccounts: */+ (NSArray *)allAccounts;/** Returns an array containing the Keychain’s accounts, or `nil` if the Keychain doesn’t have any accounts. See the `NSString` constants declared in SSKeychain.h for a list of keys that can be used when accessing the dictionaries returned by this method. @param error If accessing the accounts fails, upon return contains an error that describes the problem. @return An array of dictionaries containing the Keychain’s accounts, or `nil` if the Keychain doesn’t have any accounts. The order of the objects in the array isn’t defined. @see allAccounts */+ (NSArray *)allAccounts:(NSError **)error;/** Returns an array containing the Keychain’s accounts for a given service, or `nil` if the Keychain doesn’t have any accounts for the given service. See the `NSString` constants declared in SSKeychain.h for a list of keys that can be used when accessing the dictionaries returned by this method. @param serviceName The service for which to return the corresponding accounts. @return An array of dictionaries containing the Keychain’s accountsfor a given `serviceName`, or `nil` if the Keychain doesn’t have any accounts for the given `serviceName`. The order of the objects in the array isn’t defined. @see accountsForService:error: */+ (NSArray *)accountsForService:(NSString *)serviceName;/** Returns an array containing the Keychain’s accounts for a given service, or `nil` if the Keychain doesn’t have any accounts for the given service. @param serviceName The service for which to return the corresponding accounts. @param error If accessing the accounts fails, upon return contains an error that describes the problem. @return An array of dictionaries containing the Keychain’s accountsfor a given `serviceName`, or `nil` if the Keychain doesn’t have any accounts for the given `serviceName`. The order of the objects in the array isn’t defined. @see accountsForService: */+ (NSArray *)accountsForService:(NSString *)serviceName error:(NSError **)error;///------------------------/// @name Getting Passwords///------------------------/** Returns a string containing the password for a given account and service, or `nil` if the Keychain doesn’t have a password for the given parameters. @param serviceName The service for which to return the corresponding password. @param account The account for which to return the corresponding password. @return Returns a string containing the password for a given account and service, or `nil` if the Keychain doesn’t have a password for the given parameters. @see passwordForService:account:error: */+ (NSString *)passwordForService:(NSString *)serviceName account:(NSString *)account;/** Returns a string containing the password for a given account and service, or `nil` if the Keychain doesn’t have a password for the given parameters. @param serviceName The service for which to return the corresponding password. @param account The account for which to return the corresponding password. @param error If accessing the password fails, upon return contains an error that describes the problem. @return Returns a string containing the password for a given account and service, or `nil` if the Keychain doesn’t have a password for the given parameters. @see passwordForService:account: */+ (NSString *)passwordForService:(NSString *)serviceName account:(NSString *)account error:(NSError **)error;/** Returns the password data for a given account and service, or `nil` if the Keychain doesn’t have data for the given parameters. @param serviceName The service for which to return the corresponding password. @param account The account for which to return the corresponding password. @param error If accessing the password fails, upon return contains an error that describes the problem. @return Returns a the password data for the given account and service, or `nil` if the Keychain doesn’t have data for the given parameters. @see passwordDataForService:account:error: */+ (NSData *)passwordDataForService:(NSString *)serviceName account:(NSString *)account;/** Returns the password data for a given account and service, or `nil` if the Keychain doesn’t have data for the given parameters. @param serviceName The service for which to return the corresponding password. @param account The account for which to return the corresponding password. @param error If accessing the password fails, upon return contains an error that describes the problem. @return Returns a the password data for the given account and service, or `nil` if the Keychain doesn’t have a password for the given parameters. @see passwordDataForService:account: */+ (NSData *)passwordDataForService:(NSString *)serviceName account:(NSString *)account error:(NSError **)error;///-------------------------/// @name Deleting Passwords///-------------------------/** Deletes a password from the Keychain. @param serviceName The service for which to delete the corresponding password. @param account The account for which to delete the corresponding password. @return Returns `YES` on success, or `NO` on failure. @see deletePasswordForService:account:error: */+ (BOOL)deletePasswordForService:(NSString *)serviceName account:(NSString *)account;/** Deletes a password from the Keychain. @param serviceName The service for which to delete the corresponding password. @param account The account for which to delete the corresponding password. @param error If deleting the password fails, upon return contains an error that describes the problem. @return Returns `YES` on success, or `NO` on failure. @see deletePasswordForService:account: */+ (BOOL)deletePasswordForService:(NSString *)serviceName account:(NSString *)account error:(NSError **)error;///------------------------/// @name Setting Passwords///------------------------/** Sets a password in the Keychain. @param password The password to store in the Keychain. @param serviceName The service for which to set the corresponding password. @param account The account for which to set the corresponding password. @return Returns `YES` on success, or `NO` on failure. @see setPassword:forService:account:error: */+ (BOOL)setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account;/** Sets a password in the Keychain. @param password The password to store in the Keychain. @param serviceName The service for which to set the corresponding password. @param account The account for which to set the corresponding password. @param error If setting the password fails, upon return contains an error that describes the problem. @return Returns `YES` on success, or `NO` on failure. @see setPassword:forService:account: */+ (BOOL)setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account error:(NSError **)error;/** Sets arbirary data in the Keychain. @param password The data to store in the Keychain. @param serviceName The service for which to set the corresponding password. @param account The account for which to set the corresponding password. @param error If setting the password fails, upon return contains an error that describes the problem. @return Returns `YES` on success, or `NO` on failure. @see setPasswordData:forService:account:error: */+ (BOOL)setPasswordData:(NSData *)password forService:(NSString *)serviceName account:(NSString *)account;/** Sets arbirary data in the Keychain. @param password The data to store in the Keychain. @param serviceName The service for which to set the corresponding password. @param account The account for which to set the corresponding password. @param error If setting the password fails, upon return contains an error that describes the problem. @return Returns `YES` on success, or `NO` on failure. @see setPasswordData:forService:account: */+ (BOOL)setPasswordData:(NSData *)password forService:(NSString *)serviceName account:(NSString *)account error:(NSError **)error;///--------------------/// @name Configuration///--------------------#if __IPHONE_4_0 && TARGET_OS_IPHONE/** Returns the accessibility type for all future passwords saved to the Keychain. @return Returns the accessibility type. The return value will be `NULL` or one of the 'Keychain Item Accessibility Constants' used for determining when a keychain item should be readable. @see accessibilityType */+ (CFTypeRef)accessibilityType;/** Sets the accessibility type for all future passwords saved to the Keychain. @param accessibilityType One of the 'Keychain Item Accessibility Constants' used for determining when a keychain item should be readable. If the value is `NULL` (the default), the Keychain default will be used. @see accessibilityType */+ (void)setAccessibilityType:(CFTypeRef)accessibilityType;#endif@end3.4 創建新類,引用 SSKeychain 封裝

#import 'GetKeychain.h'#import 'SSKeychain.h'@implementation GetKeychain+ (NSString *)getDeviceUUID { NSString *currentDeviceUUIDStr = [SSKeychain passwordForService:@'項目boudle id' account:@'uuid']; if (currentDeviceUUIDStr == nil || [currentDeviceUUIDStr isEqualToString:@'']) {NSUUID *currentDeviceUUID = [UIDevice currentDevice].identifierForVendor;currentDeviceUUIDStr = currentDeviceUUID.UUIDString;currentDeviceUUIDStr = [currentDeviceUUIDStr stringByReplacingOccurrencesOfString:@'-' withString:@''];currentDeviceUUIDStr = [currentDeviceUUIDStr lowercaseString];[SSKeychain setPassword: currentDeviceUUIDStr forService:@'項目boudle id' account:@'uuid']; }return currentDeviceUUIDStr;}@end

以上就是iOS獲取設備唯一標識的實現步驟的詳細內容,更多關于iOS獲取設備唯一標識的資料請關注好吧啦網其它相關文章!

標簽: IOS
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩精品亚洲一区二区三区免费| 亚洲欧美网站| 亚洲精品四区| 蜜桃视频在线观看一区| 99在线精品免费视频九九视| 亚洲我射av| 国产一区二区三区天码| 婷婷国产精品| 手机精品视频在线观看| 亚洲免费资源| 日韩精品导航| 色爱av综合网| 国产伦乱精品| 成人精品天堂一区二区三区| 蜜桃久久久久久| 国产欧美亚洲精品a| 精品一区二区三区亚洲| 女人天堂亚洲aⅴ在线观看| 国内自拍视频一区二区三区| 麻豆高清免费国产一区| 亚洲精品第一| 免费日韩一区二区| 超碰在线99| 亚洲五月综合| 韩国精品主播一区二区在线观看| 亚洲一级黄色| 日韩精品午夜视频| 成人羞羞在线观看网站| 石原莉奈在线亚洲三区| 成人午夜网址| 青青草国产成人99久久| 天堂成人国产精品一区| 天堂网在线观看国产精品| 欧美激情一区| 欧美亚洲自偷自偷| 日韩黄色av| 亚洲一区二区小说| 麻豆久久精品| 久久大逼视频| 亚洲精品在线国产| 日韩毛片视频| 999久久久国产精品| 性欧美69xoxoxoxo| 影音先锋国产精品| 蜜桃av一区二区在线观看| 日韩国产欧美在线播放| 国产精品白浆| 国产拍在线视频| 久久精品影视| 91成人精品视频| 日本午夜免费一区二区 | 欧美成人国产| 亚洲资源网站| 欧美日韩一二三四| 亚洲精品动态| 国产色噜噜噜91在线精品| 嫩草伊人久久精品少妇av杨幂| 精品国产日韩欧美精品国产欧美日韩一区二区三区 | 久久精品伊人| 日韩国产欧美视频| 日韩高清电影免费| 国产一区二区三区亚洲| 日本欧美不卡| 国产精品毛片视频| 夜夜嗨一区二区| 麻豆精品久久久| 视频一区中文字幕| 日本蜜桃在线观看视频| 日韩av一区二区在线影视| 精品在线播放| 午夜精品成人av| 免费在线日韩av| 在线看片日韩| 尤物网精品视频| 精品国产一区二| 日韩国产精品久久久久久亚洲| 亚洲天堂黄色| 欧美成人基地| 成人久久一区| 免费污视频在线一区| 国产传媒av在线| 久久精品午夜| 国产精品久久久久久久久久白浆| 男人天堂欧美日韩| 久久都是精品| 三级亚洲高清视频| 免费精品视频最新在线| 亚洲精品1区2区| 亚洲激情五月| 日本不卡的三区四区五区| 丝袜诱惑制服诱惑色一区在线观看 | 日本不卡不码高清免费观看| 91九色精品国产一区二区| 国精品一区二区三区| 在线日韩欧美| 日韩亚洲精品在线| 日韩视频不卡| 国产精品97| 天堂精品久久久久| 免费看久久久| 国产欧美自拍| 9999国产精品| 九色精品91| 日韩黄色在线观看| 国产一区三区在线播放| 日韩一区自拍| 亚洲人成亚洲精品| 日本久久精品| 日韩午夜精品| 精品国产一级| 久久xxxx精品视频| 四虎国产精品免费观看| 狠狠爱成人网| 久久精品一区二区国产| 国产一区二区高清| 欧美精品国产白浆久久久久| 日本欧美不卡| 精品一区二区三区四区五区| 久久国产精品99国产| 日韩国产一区二区三区| 69精品国产久热在线观看| 香蕉精品视频在线观看| 久久成人福利| 欧美一级久久| 一区二区精品| 午夜一级久久| 视频一区欧美精品| 亚洲不卡av不卡一区二区| 精品深夜福利视频| 久久xxx视频| 欧美va天堂在线| 日韩欧美三级| 日本少妇一区| 亚洲播播91| 三级在线看中文字幕完整版| 麻豆视频久久| 成人午夜亚洲| 91亚洲国产| 高清一区二区| 久久精品天堂| 国产成人精品一区二区三区在线| 国产精品一区二区三区av| 日韩1区2区日韩1区2区| 亚洲3区在线| 欧美激情网址| 九九99久久精品在免费线bt| 国内在线观看一区二区三区| 国产精品亚洲欧美一级在线| 国产日产精品_国产精品毛片 | 自拍日韩欧美| 日韩精品一区二区三区免费观看| 成人在线黄色| 精品91久久久久| 合欧美一区二区三区| 亚洲乱码视频| 国产一区二区三区不卡视频网站| 国产福利亚洲| 日韩久久精品网| 久久不射中文字幕| 国产欧美日韩一区二区三区在线| 国产日产一区| 裤袜国产欧美精品一区| 美女网站一区| 国产激情精品一区二区三区| 亚洲v在线看| 奇米色欧美一区二区三区| 欧美成人aaa| 久久国产精品成人免费观看的软件| 亚洲一区二区动漫| 精品国产欧美日韩一区二区三区| 99精品视频在线| 亚洲成人精品| 国产精品一区二区三区av| 91欧美在线| 国产精品亲子伦av一区二区三区| 欧美1级日本1级| 国产一区二区三区探花| 中文字幕av一区二区三区人| 婷婷综合六月| 9999国产精品| 欧美日韩午夜| 免费久久99精品国产| 国产韩日影视精品| 91看片一区| 精品国产一区二区三区av片| 欧美在线精品一区| 日韩精品水蜜桃| 国产精品二区不卡| 麻豆国产精品一区二区三区| 日韩黄色在线观看| 日韩综合一区二区三区| 久久高清一区| 亚洲专区欧美专区| 亚洲综合另类| 亚洲资源av| 日韩精品第二页| 国产欧美日韩一区二区三区四区 | 国产精品99免费看| 日韩精品久久久久久久软件91| 免费看黄色91|