iOS обнаруживает 3G или WiFi
Я не уверен, возможно ли это, но у меня есть такой сценарий.
У меня есть веб-сайт, отображаемый в моем UIWebView, который имеет ссылку, установленную в UISegmentedController. Их веб-сайт может обнаружить, если вы находитесь на Wi-Fi или 3G-сети.
Теперь сегментированный контроллер указывает на 2 разные страницы: 1 - Экран входа в систему для iPhone 2 - Домашняя страница после входа в систему.
Теперь вот вопрос:
Могу ли я запрограммировать свое приложение, чтобы определить, относится ли оно к WIFI или 3G (я знаю, что вы можете это сделать), но затем, основываясь на ответе, перейдите к сегменту 1 или 2
Вроде как это:
if (iPhone device is on 3g) {
Go to Segment 1;
} else {
Go to Segment 0;
}
13 ответов
Используя код, предоставленный Apple здесь
Reachability *reachability = [Reachability reachabilityForInternetConnection];
[reachability startNotifier];
NetworkStatus status = [reachability currentReachabilityStatus];
if(status == NotReachable)
{
//No internet
}
else if (status == ReachableViaWiFi)
{
//WiFi
}
else if (status == ReachableViaWWAN)
{
//3G
}
Если вы не хотите импортировать библиотеку Reachability или иметь дело с уведомителями, вы можете использовать этот простой синхронный метод:
typedef enum {
ConnectionTypeUnknown,
ConnectionTypeNone,
ConnectionType3G,
ConnectionTypeWiFi
} ConnectionType;
+ (ConnectionType)connectionType
{
SCNetworkReachabilityRef reachability = SCNetworkReachabilityCreateWithName(NULL, "8.8.8.8");
SCNetworkReachabilityFlags flags;
BOOL success = SCNetworkReachabilityGetFlags(reachability, &flags);
CFRelease(reachability);
if (!success) {
return ConnectionTypeUnknown;
}
BOOL isReachable = ((flags & kSCNetworkReachabilityFlagsReachable) != 0);
BOOL needsConnection = ((flags & kSCNetworkReachabilityFlagsConnectionRequired) != 0);
BOOL isNetworkReachable = (isReachable && !needsConnection);
if (!isNetworkReachable) {
return ConnectionTypeNone;
} else if ((flags & kSCNetworkReachabilityFlagsIsWWAN) != 0) {
return ConnectionType3G;
} else {
return ConnectionTypeWiFi;
}
}
Импортируйте доступность Apple и попробуйте это,
#import "Reachability.h"
#import <CoreTelephony/CTTelephonyNetworkInfo.h>
//Try this
Reachability *reachability = [Reachability reachabilityForInternetConnection];
[reachability startNotifier];
NetworkStatus status = [reachability currentReachabilityStatus];
if(status == NotReachable)
{
NSLog(@"none");
//No internet
}
else if (status == ReachableViaWiFi)
{
NSLog(@"Wifi");
//WiFi
}
else if (status == ReachableViaWWAN)
{
NSLog(@"WWAN");
//connection type
CTTelephonyNetworkInfo *netinfo = [[CTTelephonyNetworkInfo alloc] init];
_carrier = [[netinfo subscriberCellularProvider] carrierName];
if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyGPRS]) {
NSLog(@"2G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyEdge]) {
NSLog(@"2G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyWCDMA]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyHSDPA]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyHSUPA]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMA1x]) {
NSLog(@"2G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORev0]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORevA]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORevB]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyeHRPD]) {
NSLog(@"3G");
} else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyLTE]) {
NSLog(@"4G");
}
}
Ссылки (Ссылки могут прерваться в будущем):
При использовании iOS 12 или новее вы можете использовать NWPathMonitor
вместо доисторического Reachability
учебный класс:
import Network // Put this on top of your class
let monitor = NWPathMonitor()
monitor.pathUpdateHandler = { path in
if path.status != .satisfied {
// Not connected
}
else if path.usesInterfaceType(.cellular) {
// Cellular 3/4/5g connection
}
else if path.usesInterfaceType(.wifi) {
// Wi-fi connection
}
else if path.usesInterfaceType(.wiredEthernet) {
// Ethernet connection
}
}
monitor.start(queue: DispatchQueue.global(qos: .background))
Я сделал довольно простую основанную на блоках оболочку Reachability, которая удаляет весь устаревший C-подобный код Reachability, вылитый в гораздо более какао-форму.
Использование как:
[EPPZReachability reachHost:hostNameOrIPaddress
completition:^(EPPZReachability *reachability)
{
if (reachability.reachableViaCellular) [self doSomeLightweightStuff];
}];
Смотрите раздел "Доступность с блоками для повседневного использования" в блоге eppz! Или получите его прямо в разделе "Доступность eppz!" На GitHub.
Он также работает с IP-адресами, что оказалось довольно редкой функцией оболочки Reachability.
Для быстрого мы можем использовать:
func getNetworkType()->String {
do{
let reachability:Reachability = try Reachability.reachabilityForInternetConnection()
do{
try reachability.startNotifier()
let status = reachability.currentReachabilityStatus
if(status == .NotReachable){
return ""
}else if (status == .ReachableViaWiFi){
return "Wifi"
}else if (status == .ReachableViaWWAN){
let networkInfo = CTTelephonyNetworkInfo()
let carrierType = networkInfo.currentRadioAccessTechnology
switch carrierType{
case CTRadioAccessTechnologyGPRS?,CTRadioAccessTechnologyEdge?,CTRadioAccessTechnologyCDMA1x?: return "2G"
case CTRadioAccessTechnologyWCDMA?,CTRadioAccessTechnologyHSDPA?,CTRadioAccessTechnologyHSUPA?,CTRadioAccessTechnologyCDMAEVDORev0?,CTRadioAccessTechnologyCDMAEVDORevA?,CTRadioAccessTechnologyCDMAEVDORevB?,CTRadioAccessTechnologyeHRPD?: return "3G"
case CTRadioAccessTechnologyLTE?: return "4G"
default: return ""
}
// Get carrier name
}else{
return ""
}
}catch{
return ""
}
}catch{
return ""
}
}
Метод класса следующий
+(NSString*)connectedNetworkType {
Reachability *reachability = [Reachability reachabilityForInternetConnection];
[reachability startNotifier];
NetworkStatus status = [reachability currentReachabilityStatus];
if(status == NotReachable) {
NSLog(@"none");
//No internet
}
else if (status == ReachableViaWiFi) {
NSLog(@"Wifi");
//WiFi
return @"Wifi";
}
else if (status == ReachableViaWWAN){
NSLog(@"WWAN");
//connection type
CTTelephonyNetworkInfo *netinfo = [[CTTelephonyNetworkInfo alloc] init];
// _carrier = [[netinfo subscriberCellularProvider] carrierName];
if (([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyGPRS])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyEdge])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMA1x])) {
NSLog(@"2G");
return @"2G";
}
else if (([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyWCDMA])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyHSDPA])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyHSUPA])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORev0])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORevA])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyCDMAEVDORevB])
||([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyeHRPD])){
NSLog(@"3G");
return @"3G";
}
else if ([netinfo.currentRadioAccessTechnology isEqualToString:CTRadioAccessTechnologyLTE]) {
NSLog(@"4G");
return @"4G";
}
}
return @"-1";//default unknown
}
#import <ifaddrs.h>
#import <arpa/inet.h>
BOOL CheckWiFi() {
struct ifaddrs *interfaces = NULL;
struct ifaddrs *temp_addr = NULL;
BOOL hasWifi = NO;
int err = getifaddrs(&interfaces);
if(err == 0) {
temp_addr = interfaces;
while(temp_addr) {
if(temp_addr->ifa_addr->sa_family == AF_INET) {
struct sockaddr_in *addr = (struct sockaddr_in *)temp_addr->ifa_addr;
if(memcmp(temp_addr->ifa_name, "en", 2) == 0) {
hasWifi = YES;
break;
}
}
temp_addr = temp_addr->ifa_next;
}
}
freeifaddrs(interfaces);
return hasWifi;
}
Чтобы проверить, если вы находитесь в Wi-Fi, это экономит дорогостоящую проверку подключения. Проверьте для ifa_name "bridge", чтобы проверить доступ к Интернету.
Если вы используете Xamarin или Monotouch, вы можете использовать адаптированный класс Reachability из репозитория Xamarin GitHub:
https://github.com/xamarin/monotouch-samples/blob/master/ReachabilitySample/reachability.cs
Так что добавьте его в свой проект и позвоните Reachability.InternetConnectionStatus()
Если вы не хотите использовать Reachability или использовать Swift, простой ответ здесь
/questions/35954449/kak-ispolzovat-scnetworkreachability-v-swift/56560465#56560465
версия Swift с поддержкой 5G; на основе нескольких ответов.
import Foundation
import SystemConfiguration
import CoreTelephony
/// Returns current network connection type(e.g. 4G, WIFI). It may take some time; so there is the suggestion not to use it on the main queue.
static func connectionType() -> ConnectionType {
guard let reachability = SCNetworkReachabilityCreateWithName(kCFAllocatorDefault, "www.google.com") else {
return .noConnection
}
var flags = SCNetworkReachabilityFlags()
SCNetworkReachabilityGetFlags(reachability, &flags)
guard flags.contains(.reachable) else { return .noConnection }
guard flags.contains(.isWWAN) else { return .wifi }
let networkInfo = CTTelephonyNetworkInfo()
let carrierTypeName: String?
if #available(iOS 12.0, *) {
carrierTypeName = networkInfo.serviceCurrentRadioAccessTechnology?.first?.value
} else {
carrierTypeName = networkInfo.currentRadioAccessTechnology
}
if #available(iOS 14.0, *) {
switch carrierTypeName {
case CTRadioAccessTechnologyNR, CTRadioAccessTechnologyNRNSA:
return .cellular5G
default:
break
}
}
switch carrierTypeName {
case CTRadioAccessTechnologyGPRS, CTRadioAccessTechnologyEdge, CTRadioAccessTechnologyCDMA1x:
return .cellular2G
case CTRadioAccessTechnologyLTE:
return .cellular4G
case nil:
return .unknown
default:
return .cellular3G
}
}
enum ConnectionType: CustomStringConvertible {
case noConnection, unknown, wifi, cellular2G, cellular3G, cellular4G, cellular5G
var description: String {
switch self {
case .noConnection:
return "no connection"
case .unknown:
return "unknown"
case .wifi:
return "WIFI"
case .cellular2G:
return "2G"
case .cellular3G:
return "3G"
case .cellular4G:
return "LTE"
case .cellular5G:
return "5G"
}
}
}
Вот обновленная версия для iOS 6 с включенным SimplePing от Apple. Он совместим с ARC, и я начал с исправления другого человека, связанного с доступностью. http://elbsolutions.com/projects/reachability-with-simpleping-wrapper/
Я надеюсь, что это помогает кому-то.
Используйте этот, построенный с Reachability и простой в использовании, всего несколько строк кода для интеграции. Имеет функцию обратного вызова, чтобы сообщить вам, когда изменилось соединение http://huytd.github.io/datatify/