Eu pesquisei sobre este tópico, mas encontrei muito poucos detalhes que foram úteis. Com esses detalhes, tentei preparar alguns códigos da seguinte maneira.
Nota: Compare os detalhes compartilhados nesta postagem com outras postagens antes de marcar como DUPLICATE, e não apenas pelo assunto.
- (NSArray *)getDataCountersForType:(int)type {
BOOL success;
struct ifaddrs *addrs = nil;
const struct ifaddrs *cursor = nil;
const struct sockaddr_dl *dlAddr = nil;
const struct if_data *networkStatisc = nil;
int dataSent = 0;
int dataReceived = 0;
success = getifaddrs(&addrs) == 0;
if (success) {
cursor = addrs;
while (cursor != NULL) {
if (cursor->ifa_addr->sa_family == AF_LINK) {
dlAddr = (const struct sockaddr_dl *) cursor->ifa_addr;
networkStatisc = (const struct if_data *) cursor->ifa_data;
if (type == WiFi) {
dataSent += networkStatisc->ifi_opackets;
dataReceived += networkStatisc->ifi_ipackets;
}
else if (type == WWAN) {
dataSent += networkStatisc->ifi_obytes;
dataReceived += networkStatisc->ifi_ibytes;
}
}
cursor = cursor->ifa_next;
}
freeifaddrs(addrs);
}
return [NSArray arrayWithObjects:[NSNumber numberWithInt:dataSent], [NSNumber numberWithInt:dataReceived], nil];
}
Este código coleta informações de uso da Internet de um dispositivo iPhone (e não apenas do meu aplicativo).
Agora, se eu usar a Internet via Wi-Fi ou 3G, recebo os dados (bytes) apenas em ifi_obytes
(enviados) e ifi_ibytes
(recebidos), mas acho que devo obter o uso de Wi-Fi em ifi_opackets
e ifi_ipackets
.
Também queria acrescentar que, se eu estou conectado a uma rede Wi-Fi, mas não estou usando internet, eu ainda obter um valor acrescentado para ifi_obytes
e ifi_ibytes
.
Pode ser que eu esteja errado na implementação ou no entendimento. Precisa de alguém para me ajudar.
Edit: Em vez de AF_LINK
eu tentei AF_INET
(em sockaddr_in
vez de sockaddr_dl
). Isso trava o aplicativo.
É importante entender que esses contadores são fornecidos desde a última inicialização do dispositivo.
Portanto, para fazer um uso efetivo deles, você deve acompanhar todas as amostras com o tempo de atividade do dispositivo (você pode usar
mach_absolute_time()
- consulte isso para obter mais informações)Depois de ter contadores de amostras + tempo de atividade, você pode obter melhores heurísticas quanto ao uso de dados ...
fonte
Para adicionar à resposta aceita, é importante perceber que a quantidade de dados exibida pela interface excede e reinicia a
0
cada vez4 GB
, especialmente se você estiver usando esse código para calcular a diferença entre duas leituras. Isso ocorre porqueifi_obytes
eifi_ibytes
areuint_32
e seu valor máximo é4294967295
.Além disso, recomendo usar
unsigned int
s para as variáveis que contêm os dados enviados e recebidos.int
S regulares têm metade do valor máximo de um número inteiro não assinado; portanto, ao adicionarifi_obytes
, isso pode causar um estouro.fonte
Versão rápida da resposta aceita. Também divido o código em unidades menores.
fonte
Corrigi o código fonte acima para a versão Swift3
fonte
Uma nova versão baseada nas versões anteriores, mas adaptada para Swift4 e Xcode 9
fonte
Desculpe pela mesma resposta novamente.
mas descobri que o UInt32 não é suficiente, por isso falha quando se torna muito grande.
Acabei de alterar o UInt32 para UInt64 e funciona bem.
fonte