IOSはWIFI転送帳を実現します

  • 問題

    ビジネスシナリオでは、携帯電話のファイルや写真を他のデバイスに転送する必要があります。

    書籍の譲渡だけでなく、資料の譲渡も行います

  • プログラム

    最も基本的な方法は、デバイス間にクラウドを追加し、クラウドにデータをアップロードし、各デバイスにデータを同期することです。これは複数のデバイスに適しています。これは必須ではありません。

    2つのデバイスの間にあり、サーバーを無視する場合はどうすればよいですか?Books AppのWiFi転送を彷彿とさせる、クラウドの概念がないようですが、どうすればよいでしょうか。

  • サーブ

    • CocoaPodsを通じて導入されたフレームワークGCDWebServerを使用する
    pod "GCDWebServer", "~> 3.0"
    • ローカル受信ディレクトリを設定し、サーバーを初期化して起動します
    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
    
        if let filepath = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first as NSString? {
            let path = filepath.appendingPathComponent("transfer")
            if !FileManager.default.fileExists(atPath: path) {
                do {
                    try FileManager.default.createDirectory(atPath: path, withIntermediateDirectories: false, attributes: nil)
                } catch {
                    print(error)
                }
            }
    
            webServer = GCDWebUploader(uploadDirectory: path)
            webServer?.delegate = self
            webServer?.allowHiddenItems = true
            webServer?.allowedFileExtensions = ["doc", "docx", "xls", "xlsx", "txt", "pdf", "jpeg", "jpg"]
            webServer?.title = "善斋工具"
            webServer?.prologue = "欢饮使用善斋工具的WIFI管理平台"
            webServer?.epilogue = "善斋书屋制作"
    
            if webServer?.start() == true, let address = IPHelper.deviceIPAdress(), address.count > 0, let port = webServer?.port {
                ipLb.text = "1.确保设备在同一局域网 \n2.上传时勿关闭该页面 \n3.请网页中输入该地址 \nhttp://\(address):\(port)/"
            } else {
                ipLb.text = "GCDWebServer not running!"
            }
        }
    }
    • ローカルエリアネットワーク内のマシンのIPアドレスを取得し、他のデバイスのアクセスリンクを設定します
    #import <ifaddrs.h>
    #import <arpa/inet.h>
    #import <net/if.h>
    
    @implementation IPHelper
    
    + (NSString *)deviceIPAdress {
        NSString *address = @"";
        struct ifaddrs *interfaces = NULL;
        struct ifaddrs *temp_addr = NULL;
        int success = 0;
        success = getifaddrs(&interfaces);
        if (success == 0) { // 0 表示获取成功
            temp_addr = interfaces;
            while (temp_addr != NULL) {
                if( temp_addr->ifa_addr->sa_family == AF_INET) {
                    // Check if interface is en0 which is the wifi connection on the iPhone
                    if ([[NSString stringWithUTF8String:temp_addr->ifa_name] isEqualToString:@"en0"]) {
                        // Get NSString from C String
                        address = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)];
                    }
                }
                temp_addr = temp_addr->ifa_next;
            }
        }
        freeifaddrs(interfaces);
        return address;
    
    }
    
    #define IOS_CELLULAR    @"pdp_ip0"
    #define IOS_WIFI        @"en0"
    #define IOS_***         @"utun0"
    #define IP_ADDR_IPv4    @"ipv4"
    #define IP_ADDR_IPv6    @"ipv6"
    
    #pragma mark - 获取设备当前网络IP地址
    + (NSString *)getIPAddress:(BOOL)preferIPv4 {
        NSArray *searchArray = preferIPv4 ?
        @[ IOS_*** @"/" IP_ADDR_IPv4, IOS_*** @"/" IP_ADDR_IPv6, IOS_WIFI @"/" IP_ADDR_IPv4, IOS_WIFI @"/" IP_ADDR_IPv6, IOS_CELLULAR @"/" IP_ADDR_IPv4, IOS_CELLULAR @"/" IP_ADDR_IPv6 ] :
        @[ IOS_*** @"/" IP_ADDR_IPv6, IOS_*** @"/" 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 getIPAddresses];
        NSLog(@"addresses: %@", addresses);
    
        __block NSString *address;
        [searchArray enumerateObjectsUsingBlock:^(NSString *key, NSUInteger idx, BOOL *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]?\\d\\d?|2[0-4]\\d|25[0-5])\\."
        "([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\."
        "([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\."
        "([01]?\\d\\d?|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])];
    
            if (firstMatch) {
                NSRange resultRange = [firstMatch rangeAtIndex:0];
                NSString *result=[ipAddress substringWithRange:resultRange];
                //输出结果
                NSLog(@"%@",result);
                return YES;
            }
        }
        return NO;
    }
    
    + (NSDictionary *)getIPAddresses
    {
        NSMutableDictionary *addresses = [NSMutableDictionary dictionaryWithCapacity:8];
    
        // retrieve the current interfaces - returns 0 on success
        struct ifaddrs *interfaces;
        if(!getifaddrs(&interfaces)) {
            // Loop through linked list of interfaces
            struct 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 memory
            freeifaddrs(interfaces);
        }
        return [addresses count] ? addresses : nil;
    }
    
    • 他のデバイスのアドレスにアクセスするだけです
    let address = IPHelper.deviceIPAdress()
    let port = webServer?.port
    http://\(address):\(port)/
    • 備考:
      • デバイスが同じローカルエリアネットワークにあることを確認してください
      • アップロード中はこのページを閉じないでください
  • ゲームオーバー

    ローカルエリアネットワークでは、デバイスがサーバーとして機能し、他のデバイスがクライアントとして機能します。HTTP経由でファイルをサーバーにアップロードするだけです。初期構成パスは、サーバーがファイルを受信した後にファイルが保存されるパスです。

おすすめ

転載: blog.51cto.com/15070994/2640585