如何在iOS或macOS上检查网络连接是否正常?

1406

我想通过使用 Cocoa Touch 库来检查 iOS 上是否有互联网连接,或者使用 Cocoa 库在 macOS 上进行检查。

我用 NSURL 想到了一种方法来实现这一点。但是我觉得这种方法有点不可靠(因为即使谷歌某天挂了,依赖于第三方也是不好的),而且如果谷歌不能响应,我可以检查其他网站的响应,但这似乎是浪费并增加了额外负担。

- (BOOL)connectedToInternet {
    NSString *URLString = [NSString stringWithContentsOfURL:[NSURL URLWithString:@"http://www.google.com"]];
    return ( URLString != NULL ) ? YES : NO;
}

我所做的是否不好,(更不用提 stringWithContentsOfURL 在 iOS 3.0 和 macOS 10.4 中已被弃用),如果不好,有什么更好的方法可以实现这个功能吗?


12
建议使用 return !!URLString 或者 return URLString != nil,甚至比 return (BOOL)URLString; 更好。这样做可以使代码更加简洁易懂,而且表达的意思与原文一致。 - user529758
3
我不知道你的使用情况,但如果可能的话,最好尝试发送请求并处理任何出现的错误,比如缺少连接。如果你无法这样做,那么在这种情况下有很多好的建议。 - Ken
2
你的解决方案很聪明,我更喜欢它。你也可以使用 NSString *URLString = [NSString stringWithContentsOfURL:[NSURL URLWithString:@"https://twitter.com/getibox"] encoding:NSUTF8StringEncoding error:nil]; 来消除烦人的警告。 - Abdelrahman Eid
4
请尝试使用下面链接中的Reachability类,它适用于您的情况。https://github.com/tonymillion/Reachability - Dhaval H. Nena
4
针对最近看到这个回答的人:https://dev59.com/NGoy5IYBdhLWcg3wCpsz#8813279 - afollestad
显示剩余4条评论
46个回答

1319

重要提示:此检查应始终异步执行。以下大部分答案是同步的,因此请小心,否则您的应用程序将会冻结。


Swift

  1. 通过 CocoaPods 或 Carthage 安装:https://github.com/ashleymills/Reachability.swift

  2. 通过闭包测试可达性

    let reachability = Reachability()!
    
    reachability.whenReachable = { reachability in
        if reachability.connection == .wifi {
            print("Reachable via WiFi")
        } else {
            print("Reachable via Cellular")
        }
    }
    
    reachability.whenUnreachable = { _ in
        print("Not reachable")
    }
    
    do {
        try reachability.startNotifier()
    } catch {
        print("Unable to start notifier")
    }
    

Objective-C

  1. SystemConfiguration框架添加到项目中,但无需担心在任何地方包含它

  2. 将Tony Million的Reachability.hReachability.m版本添加到项目中(在此处找到:https://github.com/tonymillion/Reachability

  3. 更新接口部分

#import "Reachability.h"

// Add this to the interface in the .m file of your view controller
@interface MyViewController ()
{
    Reachability *internetReachableFoo;
}
@end
然后在您的视图控制器的.m文件中实现此方法,您可以调用它。
// Checks if we have an internet connection or not
- (void)testInternetConnection
{
    internetReachableFoo = [Reachability reachabilityWithHostname:@"www.google.com"];

    // Internet is reachable
    internetReachableFoo.reachableBlock = ^(Reachability*reach)
    {
        // Update the UI on the main thread
        dispatch_async(dispatch_get_main_queue(), ^{
            NSLog(@"Yayyy, we have the interwebs!");
        });
    };

    // Internet is not reachable
    internetReachableFoo.unreachableBlock = ^(Reachability*reach)
    {
        // Update the UI on the main thread
        dispatch_async(dispatch_get_main_queue(), ^{
            NSLog(@"Someone broke the internet :(");
        });
    };

    [internetReachableFoo startNotifier];
}

重要提示: Reachability 类是项目中使用最广泛的类之一,因此您可能会遇到与其他项目的命名冲突。如果发生这种情况,您需要将一对 Reachability.hReachability.m 文件重命名为其他名称以解决该问题。

注意:您使用的域名并不重要,它只是测试任何域名的网关。


4
@gonzobrains:您使用的域名并不重要。它只是测试到任何域名的网关。 - iwasrobbed
3
这只是一个例子,您可以使用任何领域。它只是检查是否有通往互联网的网关,而不是该域名是否真正可用。 - iwasrobbed
1
哦,顺便提一下,你还需要将 SystemConfiguration.framework 添加到项目中(对于方法1)。 - wiseindy
2
现在(2018年)使用www.appleiphonecell.com已经不是一个好选择了。它现在重定向到Apple.com,这是一个超过40kB的HTML文件。回到google.com - 它只有11kB。顺便说一下,google.com/m的大小相同,但据报道比较慢,慢120毫秒。 - BlueskyMed
显示剩余32条评论

316

我喜欢保持简单。我做到的方法是:

//Class.h
#import "Reachability.h"
#import <SystemConfiguration/SystemConfiguration.h>

- (BOOL)connected;

//Class.m
- (BOOL)connected
{
    Reachability *reachability = [Reachability reachabilityForInternetConnection];
    NetworkStatus networkStatus = [reachability currentReachabilityStatus];
    return networkStatus != NotReachable;
}

然后,每当我想查看是否有连接时,我使用这个:

if (![self connected]) {
    // Not connected
} else {
    // Connected. Do some Internet stuff
}

此方法不等待网络状态改变以执行任务,而是在您要求时测试当前状态。


嗨@msec,您可以在此页面尝试Andrew Zimmer的解决方案,它在断开adsl连接(并连接wifi)时运行良好。 - Williew
2
对于那些像我一样只是复制粘贴上面代码的人,请手动添加SystemConfiguration.framework,否则会出现链接错误。 - Iftikhar Ali Ansari
无论何时我连接WiFi,它总是可达的。但WiFi并不意味着它具有互联网连接。我想验证互联网连接,即使它已经连接了WiFi。你能帮我吗? - wesley

147

使用苹果的 Reachability 代码,我创建了一个函数,它可以正确地检查这个问题,而不需要你包含任何类。

在你的项目中包含 SystemConfiguration.framework。

进行一些导入:

#import <sys/socket.h>
#import <netinet/in.h>
#import <SystemConfiguration/SystemConfiguration.h>
现在只需调用此函数:
/*
Connectivity testing code pulled from Apple's Reachability Example: https://developer.apple.com/library/content/samplecode/Reachability
 */
+(BOOL)hasConnectivity {
    struct sockaddr_in zeroAddress;
    bzero(&zeroAddress, sizeof(zeroAddress));
    zeroAddress.sin_len = sizeof(zeroAddress);
    zeroAddress.sin_family = AF_INET;

    SCNetworkReachabilityRef reachability = SCNetworkReachabilityCreateWithAddress(kCFAllocatorDefault, (const struct sockaddr*)&zeroAddress);
    if (reachability != NULL) {
        //NetworkStatus retVal = NotReachable;
        SCNetworkReachabilityFlags flags;
        if (SCNetworkReachabilityGetFlags(reachability, &flags)) {
            if ((flags & kSCNetworkReachabilityFlagsReachable) == 0)
            {
                // If target host is not reachable
                return NO;
            }

            if ((flags & kSCNetworkReachabilityFlagsConnectionRequired) == 0)
            {
                // If target host is reachable and no connection is required
                //  then we'll assume (for now) that your on Wi-Fi
                return YES;
            }


            if ((((flags & kSCNetworkReachabilityFlagsConnectionOnDemand ) != 0) ||
                 (flags & kSCNetworkReachabilityFlagsConnectionOnTraffic) != 0))
            {
                // ... and the connection is on-demand (or on-traffic) if the
                //     calling application is using the CFSocketStream or higher APIs.

                if ((flags & kSCNetworkReachabilityFlagsInterventionRequired) == 0)
                {
                    // ... and no [user] intervention is needed
                    return YES;
                }
            }

            if ((flags & kSCNetworkReachabilityFlagsIsWWAN) == kSCNetworkReachabilityFlagsIsWWAN)
            {
                // ... but WWAN connections are OK if the calling application
                //     is using the CFNetwork (CFSocketStream?) APIs.
                return YES;
            }
        }
    }

    return NO;
}

这是为您测试过的iOS 5


@JezenThomas 这个代码没有异步执行网络检查,这就是为什么它“更加轻便”... 你应该总是通过订阅通知来异步执行此过程,以避免在此过程中挂起应用程序。 - iwasrobbed
谢谢,这个工作即使你正在使用WiFi ADSL而ADSL没有连接也是我所需要的。 - Williew
6
这会导致内存泄漏 - 需要使用CFRelease()释放“readability”结构(对象、事物)。 - Russell Mull
很奇怪,这个回答比我的(在一个标记为重复的问题上)早了2年,与我的完全相同,但我直到今天才看到它。 - ArtOfWarfare

124

这曾经是正确的答案,但现在已过时,因为你应该订阅可达性通知。这种方法进行同步检查:


你可以使用苹果的可达性类。它还允许您检查 Wi-Fi 是否已启用:

Reachability* reachability = [Reachability sharedReachability];
[reachability setHostName:@"www.example.com"];    // Set your host name here
NetworkStatus remoteHostStatus = [reachability remoteHostStatus];

if (remoteHostStatus == NotReachable) { }
else if (remoteHostStatus == ReachableViaWiFiNetwork) { }
else if (remoteHostStatus == ReachableViaCarrierDataNetwork) { }

Reachability类并不随SDK一起提供,而是作为苹果示例应用程序的一部分。只需下载它,并将Reachability.h/m复制到您的项目中。此外,您还需要将SystemConfiguration框架添加到您的项目中。


7
请参考我上面的评论,不要像那样使用Reachability。以异步模式使用它并订阅它发送的通知-不要这样做。 - Kendall Helmstetter Gelner
这段代码是一个很好的起点,用于在使用可达性类的委托方法之前设置必要的事项。 - Brock Woolf

83

以下是一个非常简单的答案:

NSURL *scriptUrl = [NSURL URLWithString:@"http://www.google.com/m"];
NSData *data = [NSData dataWithContentsOfURL:scriptUrl];
if (data)
    NSLog(@"Device is connected to the Internet");
else
    NSLog(@"Device is not connected to the Internet");

URL应该指向一个非常小的网站。我在这里使用Google的移动网站,但如果我有可靠的Web服务器,我将上传只有一个字符的小文件以获得最大速度。

如果您只想检查设备是否以某种方式连接到互联网,我强烈建议使用这个简单的解决方案。如果您需要知道用户是如何连接的,则使用Reachability是正确的方法。

注意:这将在加载网站时短暂地阻止您的线程。在我的情况下,这不是问题,但您应该考虑到这一点(感谢Brad指出这一点)。


10
我很喜欢这个想法,但我建议为了保持99.999%的可靠性并保持较小的响应大小,选择www.google.com/m,它是Google的移动版网页。 - rwyland
1
很棒的解决方案@Erik。我还建议您使用www.google.com,而不是rwyland所说的www.google.com/m。 很奇怪,但从我的测试中,移动版本始终比www.google.com慢大约120毫秒。 - Sebyddd
5
苹果文档建议不要这样做,因为它可能会在慢速网络上阻塞线程,导致应用程序在iOS中被终止。 - Bradley Thomas
不要将异步代码“伪装”成异步。如果这样做,如果例如DNS未设置,则您可能会永远等待(应用程序卡住),调用将无限期地阻塞。 - ingconti
5
LOL,我相信谷歌会感激你建议大家将他们作为互联网查询资源。 - mxcl
显示剩余4条评论

74

这是我在我的应用程序中的做法:虽然200状态响应代码并不能保证什么,但对我来说足够稳定。与此处发布的NSData答案相比,我的方法不需要加载那么多数据,因为它只检查头部响应。

Swift 代码

func checkInternet(flag:Bool, completionHandler:(internet:Bool) -> Void)
{
    UIApplication.sharedApplication().networkActivityIndicatorVisible = true

    let url = NSURL(string: "http://www.google.com/")
    let request = NSMutableURLRequest(URL: url!)

    request.HTTPMethod = "HEAD"
    request.cachePolicy = NSURLRequestCachePolicy.ReloadIgnoringLocalAndRemoteCacheData
    request.timeoutInterval = 10.0

    NSURLConnection.sendAsynchronousRequest(request, queue:NSOperationQueue.mainQueue(), completionHandler:
    {(response: NSURLResponse!, data: NSData!, error: NSError!) -> Void in

        UIApplication.sharedApplication().networkActivityIndicatorVisible = false

        let rsp = response as! NSHTTPURLResponse?

        completionHandler(internet:rsp?.statusCode == 200)
    })
}

func yourMethod()
{
    self.checkInternet(false, completionHandler:
    {(internet:Bool) -> Void in

        if (internet)
        {
            // "Internet" aka Google URL reachable
        }
        else
        {
            // No "Internet" aka Google URL un-reachable
        }
    })
}

Objective-C 代码

typedef void(^connection)(BOOL);

- (void)checkInternet:(connection)block
{
    NSURL *url = [NSURL URLWithString:@"http://www.google.com/"];
    NSMutableURLRequest *headRequest = [NSMutableURLRequest requestWithURL:url];
    headRequest.HTTPMethod = @"HEAD";

    NSURLSessionConfiguration *defaultConfigObject = [NSURLSessionConfiguration ephemeralSessionConfiguration];
    defaultConfigObject.timeoutIntervalForResource = 10.0;
    defaultConfigObject.requestCachePolicy = NSURLRequestReloadIgnoringLocalAndRemoteCacheData;

    NSURLSession *defaultSession = [NSURLSession sessionWithConfiguration:defaultConfigObject delegate:self delegateQueue: [NSOperationQueue mainQueue]];

    NSURLSessionDataTask *dataTask = [defaultSession dataTaskWithRequest:headRequest
        completionHandler:^(NSData *data, NSURLResponse *response, NSError *error)
    {
        if (!error && response)
        {
            block([(NSHTTPURLResponse *)response statusCode] == 200);
        }
    }];
    [dataTask resume];
}

- (void)yourMethod
{
    [self checkInternet:^(BOOL internet)
    {
         if (internet)
         {
             // "Internet" aka Google URL reachable
         }
         else
         {
             // No "Internet" aka Google URL un-reachable
         }
    }];
}

3
看起来这是最快的方法。 - Pavel
3
注意:根据我的经验,这个解决方案并不总是有效的。在许多情况下,响应时间很长后返回403错误。尽管这个解决方案看起来完美,但不能保证100%的结果。 - Mustafa
4
请使用www.appleiphonecell.com - 苹果公司专门为这个目的创建了这个网址。 - Smikey
1
我使用了appleiphonecell,因为它是苹果自己的网站,可以在中国使用,并且速度非常快。这加上你的答案为我提供了最近和最快的解决方案。谢谢! - Septronic
我的上一个评论是指ObjectiveC解决方案。 - eric
显示剩余8条评论

58

苹果提供 示例代码 以检查不同类型的网络可用性。或者你可以在iPhone开发者食谱中找到一个例子

注意:请参考@KHG对这个答案中使用苹果的reachability代码的评论。


谢谢。我发现Xcode 3.0的文档中也包含源代码,可以通过在文档中搜索“reachability”来找到。 - Brock Woolf
7
请注意,苹果公司的可达性示例代码的新修订版(09-08-09)是异步的。 - Daniel Hepper
链接已经失效。 - badsyntax

48

你可以使用苹果公司的 Reachability在此可用)。

#import "Reachability.h"

- (BOOL)networkConnection {
    return [[Reachability reachabilityWithHostName:@"www.google.com"] currentReachabilityStatus];
}

if ([self networkConnection] == NotReachable) { /* No Network */ } else { /* Network */ } //Use ReachableViaWiFi / ReachableViaWWAN to get the type of connection.

@Supertecnoboff 不,它是异步的。 - Aleksander Azizi

42

苹果提供了一个示例应用程序,可以完全实现此功能:

网络状态检测


8
请注意,Reachability示例仅检测哪些接口处于活动状态,而不是哪些接口与互联网有有效连接。 即使Reachability报告一切就绪,应用程序也应优雅地处理失败情况。 - rpetrich
3.0版本中情况好多了,因为系统将为用户呈现一个登录页面,当用户处于锁定的WiFi网络后,必须先登录才能使用……以前需要手动检查重定向(如果开发2.2.1应用程序仍需如此)。 - Kendall Helmstetter Gelner
我不会说Reachability应用程序完全符合要求。但是,它是添加所需功能的良好起点。 - Johan Karlsson
链接失效了!如果能够检查/修复,将不胜感激。非常感谢。 - Heider Sati

34

只有 Reachability 类已经更新。现在你可以使用:

Reachability* reachability = [Reachability reachabilityWithHostName:@"www.apple.com"];
NetworkStatus remoteHostStatus = [reachability currentReachabilityStatus];

if (remoteHostStatus == NotReachable) { NSLog(@"not reachable");}
else if (remoteHostStatus == ReachableViaWWAN) { NSLog(@"reachable via wwan");}
else if (remoteHostStatus == ReachableViaWiFi) { NSLog(@"reachable via wifi");}

1
除非自从4.0发布以来有所改变,否则该代码不是异步的,你肯定会在崩溃报告中看到它——我之前就遇到过这种情况。 - bpapa
1
我同意bpapa的观点。使用同步代码不是一个好主意。不过感谢提供信息。 - Brock Woolf

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接