我想检查一下我是否在使用Cocoa Touch库的iOS上或在使用Cocoa库的macOS上建立了Internet连接。

我想出了一种使用NSURL进行连接的方法。我这样做的方式似乎有点不可靠(因为即使Google可能有一天会倒闭,依靠第三方也似乎很糟糕),而且我可以检查是否有其他网站的回应(如果Google没有回应),确实浪费了资源,并给我的应用程序带来了不必要的开销。

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


我做得不好(更不用说在iOS 3.0和macOS 10.4中弃用了stringWithContentsOfURL),如果是这样,什么是更好的方法来实现这一目标?

评论

而是返回(BOOL)URLString;或者甚至更好,返回!! URLString或返回URLString!= nil

我不知道您的用例是什么,但是如果可以的话,最好尝试该请求并处理任何错误,例如出现缺乏连接的情况。如果您无法执行此操作,那么在这种情况下,这里有很多好的建议。

您的解决方案很聪明,我更喜欢。您还可以使用NSString * URLString = [NSString stringWithContentsOfURL:[NSURL URLWithString:@“ https://twitter.com/getibox”]编码:NSUTF8StringEncoding错误:无];摆脱烦人的警告。

尝试从下面的链接使用Reachability类,它将为您工作github.com/tonymillion/Reachability

对于最近找到此答案的用户:stackoverflow.com/a/8813279

#1 楼

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


Swift

1)通过CocoaPods或迦太基安装: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


4)然后在您可以调用的View Controller的.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文件之一重命名为其他名称才能解决此问题。

注意:您使用的域无关紧要。它只是测试通往任何域的网关。

评论


@gonzobrains:您使用的域无关紧要。它只是测试通往任何域的网关。

–我被抢了
2011年6月14日21:33

@KaanDedeoglu只是一个示例,使用任何您想要的域。它只是检查到Internet的网关,而不是实际可用的域。

–我被抢了
13年5月6日在16:04

哦,顺便说一句,您还需要将SystemConfiguration.framework也添加到项目中(对于方法1)。

– wiseindy
13年8月16日在18:55

使用www.appleiphonecell.com而不是google.com-正是出于这个原因,此网址是由Apple创建的。

– Smikey
15年7月25日在13:55

现在(2018年)使用www.appleiphonecell.com是一个糟糕的选择。现在,它将重定向到一个大于40kB的html文件Apple.com。返回google.com-只有11kB。顺便说一句google.com/m的大小相同,但据报告慢了120毫秒。

– BlueskyMed
18/12/9在17:31



#2 楼

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

//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)的情况下可以正常工作

–威廉
2012年5月5日,2:30

对于那些像我上面的代码一样复制和粘贴的人。还要手动添加SystemConfiguration.framework,否则会出现链接错误。

– Iftikhar Ali Ansari
15年1月14日在13:56

如果我连上WiFi,总能到达。 WiFi并不意味着它已经可以连接互联网。我想验证互联网连接,即使它具有WiFi连接。你能帮我吗?

–wesley
19年1月29日在11:55

#3 楼

使用Apple的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这不会异步执行Internet检查,这就是为什么它“更苗条” ...您应该始终通过订阅通知来异步执行此操作,以免在此过程中挂断应用程序。

–我被抢了
2012年5月3日下午5:29

谢谢,即使您正在使用wifi adsl且adsl未连接,这项工作也正是我所需要的。

–威廉
2012年5月5日下午2:27

这会泄漏内存-需要使用CFRelease()释放“可读性”结构(对象,事物)。

–罗素·马尔(Russell Mull)
2012年11月20日在8:52

@RussellMull ...任何想法如何解决泄漏?

– i_raqz
2013年9月15日5:23



很奇怪,这个答案比我的答案(在标记为重复的问题上)早2年,与我的答案完全相同,但直到今天我才看到它。

–ArtOfWarfare
2014年1月30日5:32

#4 楼

这曾经是正确的答案,但现在已经过时,因为您应该订阅通知以获取可达性。此方法同步检查:


您可以使用Apple的Reachability类。它还将允许您检查是否已启用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) { }


SDK并未附带Reachability类,而是该Apple示例应用程序的一部分。只需下载它,然后将Reachability.h / m复制到您的项目中即可。另外,还必须将SystemConfiguration框架添加到您的项目中。

评论


请参阅以上有关不使用可访问性的评论。在异步模式下使用它并订阅它发送的通知-不。

–肯德尔·赫尔姆斯特·盖尔纳(Kendall Helmstetter Gelner)
09年7月6日在4:29

这段代码是对可设置性类使用委托方法之前需要设置的内容的良好起点。

–布鲁克·伍尔夫(Brock Woolf)
09年7月6日在13:42

#5 楼

这是一个非常简单的答案:

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的移动网站,但是如果我有一个可靠的网络服务器,我会上传一个只有一个字符的小文件,以实现最大速度。

如果检查设备是否以某种方式连接到了互联网就是您想要做的所有事情,我绝对建议您使用此简单的解决方案。如果您需要了解用户的连接方式,则可以使用“可达性”。

小心:这将在加载网站时短暂阻止您的线程。就我而言,这不是问题,但是您应该考虑一下(Brad指出了这一点)。

评论




Apple文档建议不要执行此操作,因为它可能会阻塞速度较慢的网络上的线程,并可能导致应用在iOS中终止

–布拉德·托马斯(Brad Thomas)
2014年10月7日17:44


大声笑,我相信Google非常感谢您建议人们将其用作互联网支票资源。

–mxcl
18年4月3日在15:28

#6 楼

这是我在应用程序中执行的操作:虽然200状态响应代码不能保证任何功能,但对我来说足够稳定。这不需要像这里发布的NSData答案那样多的负载,因为我只需检查HEAD响应即可。

快速代码

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
         }
    }];
}


评论


看来这是最快的方法

–帕维尔
13年3月21日在8:28

警告:根据我的经验,此解决方案并非始终有效。在许多情况下,经过一段甜蜜的时光后,返回的响应是403。该解决方案似乎是完美的,但不能保证100%的结果。

–穆斯塔法
13年5月9日在7:54

自2014年6月起,由于中国政府现已完全封锁google.com,该操作将在中国大陆失败。 (虽然google.cn可以运行,但是在中国大陆没有baidu.com,没有互联网)可能要ping通您需要与之通信的任何服务器。

– prewett
2014年7月28日在5:28



请改用www.appleiphonecell.com-苹果正是出于这个原因创建了该网址。

– Smikey
15年7月25日在13:53

我使用appleiphonecell,因为它是Apple自己的,也可以在中国使用,并且它是一个非常快速的网站。结合您的回答,这为我提供了最近,最快的解决方案。谢谢

–Septronic
2015年11月9日14:18



#7 楼

Apple提供了示例代码来检查不同类型的网络可用性。或者,在iPhone开发人员手册中有一个示例。

注意:有关使用Apple可达性代码的信息,请参阅@KHG关于此答案的评论。

评论


谢谢。我发现3.0中的Xcode文档还包含源代码,该源代码是通过在文档中搜索“可达性”而找到的。

–布鲁克·伍尔夫(Brock Woolf)
09年7月5日在11:37

请注意,Apple的可达性示例代码的新修订版(09-08-09)是异步的。

–丹尼尔·赫珀(Daniel Hepper)
2010-1-17在21:03

#8 楼

您可以使用by(在此处提供)使用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
18年11月8日在19:10

#9 楼

Apple提供了一个示例应用程序,该应用程序完全可以做到这一点:

可达性

评论


您应该注意,“可达性”示例仅检测哪些接口处于活动状态,而不检测哪些接口与Internet的有效连接。即使在可达性报告一切准备就绪时,应用程序也应该适当地处理故障。

–rpetrich
09年7月5日在11:17

令人高兴的是,在3.0中情况要好得多,因为系统将在锁定的WiFi后面为用户显示一个登录页面,您必须登录才能使用...您曾经必须手动检查重定向(并且您仍然如果开发2.2.1应用程序)

–肯德尔·赫尔姆斯特·盖尔纳(Kendall Helmstetter Gelner)
09年7月6日在4:28

我不会说“可达性”应用程序确实可以满足要求。不过,这是添加所需功能的良好起点,

–约翰·卡尔森(Johan Karlsson)
19年1月9日在13:46

链接断开!如果可以检查/修复,将不胜感激,非常感谢

– Heider Sati
5月22日10:01

#10 楼

仅可达性类已更新。您现在可以使用:

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");}


评论


除非自4.0版本发布以来没有任何更改,否则该代码不是异步的,您可以保证看到它出现在“崩溃报告”中-以前是我发生的。

– bpapa
2010年8月27日在15:10

我同意bpapa。使用同步代码不是一个好主意。谢谢你的信息

–布鲁克·伍尔夫(Brock Woolf)
2010年8月27日在18:28

#11 楼

iOS 5的可达性版本为darkseed / Reachability.h。不是我的! =)

#12 楼

在这里有一个美观的,使用ARC和GCD的可达性现代化方法:

可达性

#13 楼

如果您使用的是AFNetworking,则可以使用其自己的实现来实现Internet可达性状态。 >
使用此方法的优点之一是,当可达性状态更改时,可以使用AFNetworking设置所需的行为。假设我已经创建了一个名为AFHTTPClient的单例子类blocks(如AFNetworking文档的“子类注释”所述),我将执行以下操作:

BKHTTPClient *httpClient = [BKHTTPClient sharedClient];
[httpClient setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status)
{
    if (status == AFNetworkReachabilityStatusNotReachable) 
    {
    // Not reachable
    }
    else
    {
        // Reachable
    }
}];


您也可以专门使用AFHTTPClientBKHTTPClient枚举检查Wi-Fi或WLAN连接(更多信息请参见此处)。

#14 楼

我在此讨论中使用了代码,但似乎工作正常(阅读整个线程!)。

我并未对每种可能的连接类型(例如ad Wic)进行详尽的测试。 -Fi)。

评论


这段代码并不是很好,因为它只是检查您是否与路由器建立了wifi连接,而不是是否可以访问网络。您可以使用wifi并继续启用以访问网络。

–鸭
09年11月21日在2:43

#15 楼

非常简单....尝试以下步骤:

步骤1:将SystemConfiguration框架添加到您的项目中。


步骤2:将以下代码导入到您的项目中header文件。

#import <SystemConfiguration/SystemConfiguration.h>



步骤3:使用以下方法



类型1:

- (BOOL) currentNetworkStatus {
    [UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
    BOOL connected;
    BOOL isConnected;
    const char *host = "www.apple.com";
    SCNetworkReachabilityRef reachability = SCNetworkReachabilityCreateWithName(NULL, host);
    SCNetworkReachabilityFlags flags;
    connected = SCNetworkReachabilityGetFlags(reachability, &flags);
    isConnected = NO;
    isConnected = connected && (flags & kSCNetworkFlagsReachable) && !(flags & kSCNetworkFlagsConnectionRequired);
    CFRelease(reachability);
    return isConnected;
}






类型2:

导入标头:#import "Reachability.h"

- (BOOL)currentNetworkStatus
{
    Reachability *reachability = [Reachability reachabilityForInternetConnection];
    NetworkStatus networkStatus = [reachability currentReachabilityStatus];
    return networkStatus != NotReachable;
}




步骤4:使用方法:

- (void)CheckInternet
{
    BOOL network = [self currentNetworkStatus];
    if (network)
    {
        NSLog(@"Network Available");
    }
    else
    {
        NSLog(@"No Network Available");
    }
}


评论


类型1是异步的吗?

– Supertecnoboff
18-10-3在17:27

我想从您的答案中修复2型。我添加了可达性类,并尝试通过使用以上答案来检查连接验证。即使我连接了WiFi,它也始终可以访问,但没有互联网连接。 WiFi并不意味着它已经可以连接互联网。我想验证互联网连接,即使它具有WiFi连接。你能帮我吗?

–wesley
19年1月29日在11:54

#16 楼

-(void)newtworkType {

 NSArray *subviews = [[[[UIApplication sharedApplication] valueForKey:@"statusBar"] valueForKey:@"foregroundView"]subviews];
NSNumber *dataNetworkItemView = nil;

for (id subview in subviews) {
    if([subview isKindOfClass:[NSClassFromString(@"UIStatusBarDataNetworkItemView") class]]) {
        dataNetworkItemView = subview;
        break;
    }
}


switch ([[dataNetworkItemView valueForKey:@"dataNetworkType"]integerValue]) {
    case 0:
        NSLog(@"No wifi or cellular");
        break;

    case 1:
        NSLog(@"2G");
        break;

    case 2:
        NSLog(@"3G");
        break;

    case 3:
        NSLog(@"4G");
        break;

    case 4:
        NSLog(@"LTE");
        break;

    case 5:
        NSLog(@"Wifi");
        break;


    default:
        break;
}
}


评论


即使设备已连接到Wifi或其他某种网络类型,互联网连接仍然可能不可用。简单测试:连接到家庭wifi,然后拔下电缆调制解调器。仍连接到wifi,但互联网零。

–我被抢了
13年8月22日在14:35

#17 楼

- (void)viewWillAppear:(BOOL)animated
{
    NSString *URL = [NSString stringWithContentsOfURL:[NSURL URLWithString:@"http://www.google.com"]];

    return (URL != NULL ) ? YES : NO;
}


或使用Reachability类。

有两种方法可以使用iPhone SDK检查Internet可用性:

1。检查Google页面是否打开。

2。可达性类

有关更多信息,请参阅可达性(Apple Developer)。

评论


iPhone SDK中有两种检查互联网可用性的方法:1)检查Google页面是否打开。

–IOS Rocks
2012年11月22日12:04



-1:这是一种同步方法,它将在尝试连接到google.com时阻止主线程(更改应用程序UI的主线程)。如果您的用户的数据连接速度非常慢,则电话将表现为该过程无响应。

–我被抢了
13年3月25日在23:05

#18 楼

使用http://huytd.github.io/datatify/。比自己添加库和编写代码要容易。

#19 楼

首先:在框架中添加CFNetwork.framework

代码:ViewController.m

#import "Reachability.h"

- (void)viewWillAppear:(BOOL)animated
{
    Reachability *r = [Reachability reachabilityWithHostName:@"www.google.com"];
    NetworkStatus internetStatus = [r currentReachabilityStatus];

    if ((internetStatus != ReachableViaWiFi) && (internetStatus != ReachableViaWWAN))
    {
        /// Create an alert if connection doesn't work
        UIAlertView *myAlert = [[UIAlertView alloc]initWithTitle:@"No Internet Connection"   message:NSLocalizedString(@"InternetMessage", nil)delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles:nil];
        [myAlert show];
        [myAlert release];
    }
    else
    {
         NSLog(@"INTERNET IS CONNECT");
    }
}


#20 楼

首先下载可到达性类,然后在Xcode中放入reachability.h和reachabilty.m文件。

最好的方法是制作一个通用的函数类(NSObject),以便可以在任何类中使用它。这是用于检查网络连接可达性的两种方法:

+(BOOL) reachabiltyCheck
{
    NSLog(@"reachabiltyCheck");
    BOOL status =YES;
    [[NSNotificationCenter defaultCenter] addObserver:self
                                          selector:@selector(reachabilityChanged:)
                                          name:kReachabilityChangedNotification
                                          object:nil];
    Reachability * reach = [Reachability reachabilityForInternetConnection];
    NSLog(@"status : %d",[reach currentReachabilityStatus]);
    if([reach currentReachabilityStatus]==0)
    {
        status = NO;
        NSLog(@"network not connected");
    }
    reach.reachableBlock = ^(Reachability * reachability)
    {
        dispatch_async(dispatch_get_main_queue(), ^{
        });
    };
    reach.unreachableBlock = ^(Reachability * reachability)
    {
        dispatch_async(dispatch_get_main_queue(), ^{
        });
    };
    [reach startNotifier];
    return status;
}

+(BOOL)reachabilityChanged:(NSNotification*)note
{
    BOOL status =YES;
    NSLog(@"reachabilityChanged");
    Reachability * reach = [note object];
    NetworkStatus netStatus = [reach currentReachabilityStatus];
    switch (netStatus)
    {
        case NotReachable:
            {
                status = NO;
                NSLog(@"Not Reachable");
            }
            break;

        default:
            {
                if (!isSyncingReportPulseFlag)
                {
                    status = YES;
                    isSyncingReportPulseFlag = TRUE;
                    [DatabaseHandler checkForFailedReportStatusAndReSync];
                }
            }
            break;
    }
    return status;
}

+ (BOOL) connectedToNetwork
{
    // Create zero addy
    struct sockaddr_in zeroAddress;
    bzero(&zeroAddress, sizeof(zeroAddress));
    zeroAddress.sin_len = sizeof(zeroAddress);
    zeroAddress.sin_family = AF_INET;

    // Recover reachability flags
    SCNetworkReachabilityRef defaultRouteReachability = SCNetworkReachabilityCreateWithAddress(NULL, (struct sockaddr *)&zeroAddress);
    SCNetworkReachabilityFlags flags;
    BOOL didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags);
    CFRelease(defaultRouteReachability);
    if (!didRetrieveFlags)
    {
        NSLog(@"Error. Could not recover network reachability flags");
        return NO;
    }
    BOOL isReachable = flags & kSCNetworkFlagsReachable;
    BOOL needsConnection = flags & kSCNetworkFlagsConnectionRequired;
    BOOL nonWiFi = flags & kSCNetworkReachabilityFlagsTransientConnection;
    NSURL *testURL = [NSURL URLWithString:@"http://www.apple.com/"];
    NSURLRequest *testRequest = [NSURLRequest requestWithURL:testURL  cachePolicy:NSURLRequestReloadIgnoringLocalCacheData timeoutInterval:20.0];
    NSURLConnection *testConnection = [[NSURLConnection alloc] initWithRequest:testRequest delegate:self];
    return ((isReachable && !needsConnection) || nonWiFi) ? (testConnection ? YES : NO) : NO;
}


现在您可以通过调用此类方法检查任何类中的网络连接。

#21 楼

还有另一种方法可以使用iPhone SDK检查Internet连接。

尝试为网络连接实现以下代码。

#import <SystemConfiguration/SystemConfiguration.h>
#include <netdb.h>

/**
     Checking for network availability. It returns
     YES if the network is available.
*/
+ (BOOL) connectedToNetwork
{

    // Create zero addy
    struct sockaddr_in zeroAddress;
    bzero(&zeroAddress, sizeof(zeroAddress));
    zeroAddress.sin_len = sizeof(zeroAddress);
    zeroAddress.sin_family = AF_INET;

    // Recover reachability flags
    SCNetworkReachabilityRef defaultRouteReachability =
        SCNetworkReachabilityCreateWithAddress(NULL, (struct sockaddr *)&zeroAddress);
    SCNetworkReachabilityFlags flags;

    BOOL didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags);
    CFRelease(defaultRouteReachability);

    if (!didRetrieveFlags)
    {
        printf("Error. Could not recover network reachability flags\n");
        return NO;
    }

    BOOL isReachable = ((flags & kSCNetworkFlagsReachable) != 0);
    BOOL needsConnection = ((flags & kSCNetworkFlagsConnectionRequired) != 0);

    return (isReachable && !needsConnection) ? YES : NO;
}


#22 楼

我发现它很容易使用库SimplePingHelper。

示例代码:chrishulbert / SimplePingHelper(GitHub)

#23 楼


下载可访问性文件https://gist.github.com/darkseed/1182373
,并在框架中添加CFNetwork.framework和'SystemConfiguration.framework'
执行#import“ Reachability.h”



首先:在框架中添加CFNetwork.framework

代码:ViewController.m

- (void)viewWillAppear:(BOOL)animated
{
    Reachability *r = [Reachability reachabilityWithHostName:@"www.google.com"];
    NetworkStatus internetStatus = [r currentReachabilityStatus];

    if ((internetStatus != ReachableViaWiFi) && (internetStatus != ReachableViaWWAN))
    {
        /// Create an alert if connection doesn't work
        UIAlertView *myAlert = [[UIAlertView alloc]initWithTitle:@"No Internet Connection"   message:NSLocalizedString(@"InternetMessage", nil)delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles:nil];
        [myAlert show];
        [myAlert release];
    }
    else
    {
         NSLog(@"INTERNET IS CONNECT");
    }
}


#24 楼

Swift 3 / Swift 4

必须先导入

import SystemConfiguration


您可以使用以下方法检查互联网连接

func isConnectedToNetwork() -> Bool {

    var zeroAddress = sockaddr_in()
    zeroAddress.sin_len = UInt8(MemoryLayout.size(ofValue: zeroAddress))
    zeroAddress.sin_family = sa_family_t(AF_INET)

    let defaultRouteReachability = withUnsafePointer(to: &zeroAddress) {
        q4312078q.withMemoryRebound(to: sockaddr.self, capacity: 1) {zeroSockAddress in
            SCNetworkReachabilityCreateWithAddress(nil, zeroSockAddress)
        }
    }

    var flags = SCNetworkReachabilityFlags()
    if !SCNetworkReachabilityGetFlags(defaultRouteReachability!, &flags) {
        return false
    }
    let isReachable = (flags.rawValue & UInt32(kSCNetworkFlagsReachable)) != 0
    let needsConnection = (flags.rawValue & UInt32(kSCNetworkFlagsConnectionRequired)) != 0
    return (isReachable && !needsConnection)

}


#25 楼

可达性类可以确定设备是否可以使用Internet连接...

但是在访问Intranet资源的情况下:

Ping Intranet具有可达性类的服务器始终返回true。

因此,在这种情况下,一种快速的解决方案是创建一个称为pingme的Web方法以及该服务上的其他Web方法。
pingme应该返回一些内容。

所以我在常用功能上写了以下方法

-(BOOL)PingServiceServer
{
    NSURL *url=[NSURL URLWithString:@"http://www.serveraddress/service.asmx/Ping"];

    NSMutableURLRequest *urlReq=[NSMutableURLRequest requestWithURL:url];

    [urlReq setTimeoutInterval:10];

    NSURLResponse *response;

    NSError *error = nil;

    NSData *receivedData = [NSURLConnection sendSynchronousRequest:urlReq
                                                 returningResponse:&response
                                                             error:&error];
    NSLog(@"receivedData:%@",receivedData);

    if (receivedData !=nil)
    {
        return YES;
    }
    else
    {
        NSLog(@"Data is null");
        return NO;
    }
}


上面的方法对我来说是如此有用,所以无论何时我尝试发送一些数据到服务器,我总是使用这种低超时URLRequest来检查Intranet资源的可访问性。

#26 楼

自己执行此操作非常简单。以下方法将起作用。只需确保不允许使用名称传递HTTP,HTTPS等主机名协议。

-(BOOL)hasInternetConnection:(NSString*)urlAddress
{
    SCNetworkReachabilityRef ref = SCNetworkReachabilityCreateWithName(kCFAllocatorDefault, [urlAddress UTF8String]);
    SCNetworkReachabilityFlags flags;
    if (!SCNetworkReachabilityGetFlags(ref, &flags))
    {
        return NO;
    }
    return flags & kSCNetworkReachabilityFlagsReachable;
}


这是一种快速简单且轻松的方式。

#27 楼

除了可访问性之外,您还可以使用Simple Ping帮助程序库。它的工作原理非常好,并且易于集成。

#28 楼

我认为这是最好的答案。

“是”表示已连接。 “否”表示已断开连接。

#import "Reachability.h"

 - (BOOL)canAccessInternet
{
    Reachability *IsReachable = [Reachability reachabilityForInternetConnection];
    NetworkStatus internetStats = [IsReachable currentReachabilityStatus];

    if (internetStats == NotReachable)
    {
        return NO;
    }
    else
    {
        return YES;
    }
}


#29 楼


步骤1:在您的项目中添加可访问性类。
步骤2:导入可访问性类

步骤3:创建以下函数

- (BOOL)checkNetConnection {
    self.internetReachability = [Reachability reachabilityForInternetConnection];
    [self.internetReachability startNotifier];
    NetworkStatus netStatus = [self.internetReachability currentReachabilityStatus];
    switch (netStatus) {
        case NotReachable:
        {
            return NO;
        }

        case ReachableViaWWAN:
        {
             return YES;
        }

        case ReachableViaWiFi:
        {
             return YES;
        }
    }
}



步骤4:调用以下函数:

if (![self checkNetConnection]) {
    [GlobalFunctions showAlert:@""
                     message:@"Please connect to the Internet!"
                     canBtntitle:nil
                     otherBtnTitle:@"Ok"];
    return;
}
else
{
    Log.v("internet is connected","ok");
}




评论


Apple的文档指出:注意:可达性无法告诉您的应用程序是否可以连接到特定主机,仅可以使用可能允许连接的接口以及该接口是否为WWAN即可。

– noobsmcgoobs
16年4月22日在19:55

#30 楼

在(iOS)Xcode 8,Swift 3.0中检查Internet连接可用性。这是检查网络可用性的简单方法,就像我们的设备是否连接到任何网络一样。我已经设法将其翻译为Swift 3.0,并在这里完成了最终代码。现有的Apple Reachability类和其他第三方库似乎太复杂了,无法转换为Swift。
这对于3G,4G和WiFi连接都有效。
不要忘记添加“ SystemConfiguration.framework”给您的项目构建者。

//Create new swift class file Reachability in your project.
import SystemConfiguration
public class InternetReachability {

class func isConnectedToNetwork() -> Bool {
   var zeroAddress = sockaddr_in(sin_len: 0, sin_family: 0, sin_port: 0, sin_addr: in_addr(s_addr: 0), sin_zero: (0, 0, 0, 0, 0, 0, 0, 0))
   zeroAddress.sin_len = UInt8(sizeofValue(zeroAddress))
   zeroAddress.sin_family = sa_family_t(AF_INET)
   let defaultRouteReachability = withUnsafePointer(&zeroAddress) {
          SCNetworkReachabilityCreateWithAddress(nil, UnsafePointer(q4312078q)).takeRetainedValue()
   }
   var flags: SCNetworkReachabilityFlags = 0
   if SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags) == 0 {
          return false
   }
   let isReachable = (flags & UInt32(kSCNetworkFlagsReachable)) != 0
   let needsConnection = (flags & UInt32(kSCNetworkFlagsConnectionRequired)) != 0

   return isReachable && !needsConnection
  }
}

// Check network connectivity from anywhere in project by using this code.
 if InternetReachability.isConnectedToNetwork() == true {
         print("Internet connection OK")
  } else {
         print("Internet connection FAILED")
  }


评论


MacOS版本?否则,主要问题将无法回答!

– Heitor
19年6月1日在5:11

请检查主要答案标题。

–ViJay Avhad
19年6月1日在6:22