Я намагаюся перетворити цей фрагмент коду в 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 вони спочатку оголошують екземпляр, а потім заповнюють параметри. Наскільки я знаю, у Свіфті це неможливо. Тож я в цей момент по-справжньому програю, що робити.
Я читав офіційний документ Apple про взаємодію з API API C в 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
}
EDIT 1: Гаразд, я змінив цей рядок на цей,
var defaultRouteReachability: SCNetworkReachabilityRef = SCNetworkReachabilityCreateWithAddress(UnsafePointer<Void>(), &zeroAddress)
Нова помилка, яку я отримую на цій лінії, - це "UnsafePointer", не конвертована у "CFAllocator" . Як вам пройтиNULL
у Свіфті?
Також я змінив цей рядок і зараз помилка відсутня.
let didRetrieveFlags = SCNetworkReachabilityGetFlags(defaultRouteReachability, &flags)
EDIT 2: Я пройшов nil
цей рядок, побачивши це питання. Але ця відповідь суперечить відповіді тут . Це говорить про те, що NULL
у Swift немає рівнозначного .
var defaultRouteReachability: SCNetworkReachabilityRef = SCNetworkReachabilityCreateWithAddress(nil, &zeroAddress)
У будь-якому випадку я отримую нову помилку: "sockaddr_in" не є тотожною "sockaddr" у наведеному вище рядку.