Я пытаюсь преобразовать этот фрагмент кода в Swift. Я изо всех сил пытаюсь оторваться от земли из-за некоторых трудностей.
- (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)
{
return NO;
}
BOOL isReachable = flags & kSCNetworkFlagsReachable;
BOOL needsConnection = flags & kSCNetworkFlagsConnectionRequired;
return (isReachable && !needsConnection) ? YES : NO;
}
Первая и основная проблема, с которой я столкнулся, заключается в том, как определять структуры C. struct sockaddr_in zeroAddress;
Я полагаю, что в первой строке ( ) приведенного выше кода они определяют экземпляр, вызываемый zeroAddress
из структуры sockaddr_in (?). Я пробовал объявить var
вот так.
var zeroAddress = sockaddr_in()
Но я получаю сообщение об ошибке « Отсутствует аргумент для параметра sin_len» в вызове, что понятно, потому что эта структура принимает ряд аргументов. Я попробовал еще раз.
var zeroAddress = sockaddr_in(sin_len: sizeof(zeroAddress), sin_family: AF_INET, sin_port: nil, sin_addr: nil, sin_zero: nil)
Как и ожидалось, я получаю другую переменную ошибки, используемую в собственном начальном значении . Я тоже понимаю причину этой ошибки. В C они сначала объявляют экземпляр, а затем заполняют параметры. Насколько я знаю, это невозможно в Swift. Так что я действительно не знаю, что мне делать.
Я прочитал официальный документ Apple о взаимодействии с C API в Swift, но в нем нет примеров работы со структурами.
Может ли кто-нибудь помочь мне здесь? Я был бы очень признателен.
Спасибо.
ОБНОВЛЕНИЕ: Благодаря Мартину я смог решить первоначальную проблему. Но все же Swift не облегчает мне задачу. У меня несколько новых ошибок.
func connectedToNetwork() -> 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)
var defaultRouteReachability: SCNetworkReachabilityRef = SCNetworkReachabilityCreateWithAddress(UnsafePointer<Void>, UnsafePointer<zeroAddress>) // 'zeroAddress' is not a type
var flags = SCNetworkReachabilityFlags()
let didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, UnsafeMutablePointer<flags>) // 'flags' is not a type
defaultRouteReachability.dealloc(1) // 'SCNetworkReachabilityRef' does not have a member named 'dealloc'
if didRetrieveFlags == false {
return false
}
let isReachable: Bool = flags & kSCNetworkFlagsReachable // Cannot invoke '&' with an argument list of type '(@lvalue UInt32, Int)'
let needsConnection: Bool = flags & kSCNetworkFlagsConnectionRequired // Cannot invoke '&' with an argument list of type '(@lvalue UInt32, Int)'
return (isReachable && !needsConnection) ? true : false
}
РЕДАКТИРОВАТЬ 1: Хорошо, я изменил эту строку на эту,
var defaultRouteReachability: SCNetworkReachabilityRef = SCNetworkReachabilityCreateWithAddress(UnsafePointer<Void>(), &zeroAddress)
Новая ошибка, которую я получаю в этой строке, - «UnsafePointer» не может быть преобразована в «CFAllocator» . Как пройтиNULL
на Swift?
Также я изменил эту строку, и теперь ошибка исчезла.
let didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags)
РЕДАКТИРОВАТЬ 2: Я пропустил nil
эту строку, увидев этот вопрос. Но этот ответ противоречит приведенному здесь ответу . В нем говорится, что NULL
в Swift нет эквивалента .
var defaultRouteReachability: SCNetworkReachabilityRef = SCNetworkReachabilityCreateWithAddress(nil, &zeroAddress)
В любом случае я получаю новую ошибку, в которой говорится, что «sockaddr_in» не идентично «sockaddr» в строке выше.
Ответы:
(Этот ответ неоднократно расширялся из-за изменений в языке Swift, что немного сбивало с толку. Теперь я переписал его и удалил все, что относится к Swift 1.x. Старый код можно найти в истории редактирования, если кому-то понадобится Это.)
Вот как вы это сделаете в Swift 2.0 (Xcode 7) :
Пояснения:
Начиная с Swift 1.2 (Xcode 6.3), импортированные структуры C имеют инициализатор по умолчанию в Swift, который инициализирует все поля структуры нулями, поэтому структура адреса сокета может быть инициализирована с помощью
sizeofValue()
дает размер этой структуры, это должно быть преобразовано вUInt8
дляsin_len
:AF_INET
-Int32
это необходимо преобразовать в правильный тип дляsin_family
:withUnsafePointer(&zeroAddress) { ... }
передает адрес структуры замыканию, где он используется в качестве аргумента дляSCNetworkReachabilityCreateWithAddress()
.UnsafePointer($0)
Преобразование необходимо , потому что функция ожидает указательsockaddr
, а неsockaddr_in
.Значение, возвращаемое из,
withUnsafePointer()
является возвращаемым значением изSCNetworkReachabilityCreateWithAddress()
и имеет типSCNetworkReachability?
, т. Е. Является необязательным.guard let
Заявление (новая функция Swift 2.0) присваивает развернутое значение кdefaultRouteReachability
переменному , если она не являетсяnil
. В противном случаеelse
блок выполняется и функция возвращается.SCNetworkReachabilityCreateWithAddress()
возвращает управляемый объект. Вам не нужно выпускать его явно.Начиная с Swift 2,
SCNetworkReachabilityFlags
соответствуетOptionSetType
интерфейсу, подобному множеству. Вы создаете пустую переменную флагов с помощьюи проверьте наличие флагов с помощью
Второй параметр
SCNetworkReachabilityGetFlags
имеет типUnsafeMutablePointer<SCNetworkReachabilityFlags>
, что означает, что вы должны передать адрес переменной flags.Также обратите внимание, что регистрация обратного вызова уведомителя возможна в Swift 2, сравните Работа с API C из Swift и Swift 2 - UnsafeMutablePointer <Void> с object .
Обновление для Swift 3/4:
Небезопасные указатели больше не могут быть просто преобразованы в указатель другого типа (см. - SE-0107 UnsafeRawPointer API ). Вот обновленный код:
источник
withUnsafePointer(&zeroAddress)
вызывает следующее закрытие{ ...}
с адресом вzeroAddress
качестве аргумента. Внутри закрытия$0
стоит этот аргумент. - Извините, все это невозможно объяснить в нескольких предложениях. Посмотрите документацию о закрытии в книге Swift. $ 0 - это «сокращенное имя аргумента».true
если Wi-Fi не подключен и 4G включен, но пользователь указал, что приложение не может использовать сотовые данные. Какие-либо решения?let cellular = flags.contains(.IsWWAN)
вы можете вернуть знак вместо логического, например:func connectedToNetwork() -> (connected: Bool, cellular: Bool)
Swift 3, IPv4, IPv6
На основе ответа Мартина Р.
источник
import SystemConfiguration
Это не имеет ничего общего со Swift, но лучшее решение - НЕ использовать Reachability для определения, находится ли сеть в сети. Просто установите соединение и обработайте ошибки в случае сбоя. Установление соединения может иногда запускать бездействующие автономные радиостанции.
Единственное допустимое использование Reachability - использовать ее для уведомления вас, когда сеть переходит из автономного режима в онлайн. На этом этапе вы должны повторить неудачные попытки подключения.
источник
NSURLSession
API сNSURLSessionConfiguration.allowsCellularAccess = false
.Лучшее решение - использовать
ReachabilitySwift
класс , написанный наSwift 2
и использующийSCNetworkReachabilityRef
.Просто и легко:
Работает как шарм.
наслаждаться
источник
SCNetworkReachability
класс в Swift, это предложение о зависимости, которую следует использовать для проверки действительного сетевого подключения.обновлен ответ juanjo для создания экземпляра singleton
использование
источник
Это в Swift 4.0
Я использую этот фреймворк https://github.com/ashleymills/Reachability.swift
и устанавливаю Pod ..
В AppDelegate
Экран reachabilityViewController появится, если интернета нет
источник
Swift 5, используя
NWPathMonitor
источник