WebViewJavascriptBridge 分析oc向js发送消息处理流程

1.优秀开源代码解读之JS与iOS Native Code互调的优雅实现方案

2. WebViewJavascriptBridge 原理分析

过程分析如下:

- (void)sendMessage:(id)sender {

       //  responseCallbackjs返回数据后回调

    [_bridge send:@"A string sent from ObjC to JS" responseCallback:^(id response) {

        NSLog(@"sendMessage got response: %@", response);

    }];

}


- (void)send:(id)data responseCallback:(WVJBResponseCallback)responseCallback {

    [self _sendData:data responseCallback:responseCallback handlerName:nil];

}


- (void)_sendData:(id)data responseCallback:(WVJBResponseCallback)responseCallback handlerName:(NSString*)handlerName {

    NSMutableDictionary* message = [NSMutableDictionary dictionary];

    

    if (data) {

        message[@"data"] = data;

    }

    

    if (responseCallback) {

        NSString* callbackId = [NSString stringWithFormat:@"objc_cb_%ld", ++_uniqueId];

         //用callbackId 标示responseCallback并保存起来,供js返回数据后回调

        _responseCallbacks[callbackId] = [responseCallback copy];

        message[@"callbackId"] = callbackId;

    }

    

    if (handlerName) {

        message[@"handlerName"] = handlerName;

    }

    [self _queueMessage:message];

}


- (void)_queueMessage:(WVJBMessage*)message {

    if (_startupMessageQueue) {

        [_startupMessageQueue addObject:message];

    } else {

        [self _dispatchMessage:message];

    }

}


- (void)_dispatchMessage:(WVJBMessage*)message {

    NSString *messageJSON = [self _serializeMessage:message];

    [self _log:@"SEND" json:messageJSON];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\\" withString:@"\\\\"];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\"" withString:@"\\\""];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\'" withString:@"\\\'"];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\n" withString:@"\\n"];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\r" withString:@"\\r"];

    messageJSON = [messageJSON stringByReplacingOccurrencesOfString:@"\f" withString:@"\\f"];

     //调用javascript中的函数_handleMessageFromObjC

    NSString* javascriptCommand = [NSString stringWithFormat:@"WebViewJavascriptBridge._handleMessageFromObjC('%@');", messageJSON];

    if ([[NSThread currentThread] isMainThread]) {

        [_webView stringByEvaluatingJavaScriptFromString:javascriptCommand];

    } else {

        __strong WVJB_WEBVIEW_TYPE* strongWebView = _webView;

        dispatch_sync(dispatch_get_main_queue(), ^{

            [strongWebView stringByEvaluatingJavaScriptFromString:javascriptCommand];

        });

    }

}


//处理来自ObjC的消息【native端调用】

function _handleMessageFromObjC(messageJSON) {


    //如果接收队列对象存在则入队该消息,否则直接处理

if (receiveMessageQueue) {

receiveMessageQueue.push(messageJSON)

} else {

_dispatchMessageFromObjC(messageJSON)

}

}


//内部方法:处理来自objc的消息

function _dispatchMessageFromObjC(messageJSON) {

setTimeout(function _timeoutDispatchMessageFromObjC() {

var message = JSON.parse(messageJSON)

var messageHandler

if (message.responseId) {

var responseCallback = responseCallbacks[message.responseId]

if (!responseCallback) { return; }

responseCallback(message.responseData)

delete responseCallbacks[message.responseId]

} else {

var responseCallback

if (message.callbackId) {

var callbackResponseId = message.callbackId

                    //handle处理完后回调

responseCallback = function(responseData) {

_doSend({ responseId:callbackResponseId, responseData:responseData })

}

}

//走这里,此处handler等于WebViewJavascriptBridge.init时的_messageHandler

var handler = WebViewJavascriptBridge._messageHandler

if (message.handlerName) {

handler = messageHandlers[message.handlerName]

}

try {

handler(message.data, responseCallback)

} catch(exception) {

if (typeof console != 'undefined') {

console.log("WebViewJavascriptBridge: WARNING: javascript handler threw.", message, exception)

}

}

}

})

}


 //初始化操作,并定义默认的消息处理逻辑

bridge.init(function(message, responseCallback) {

log('JS got a message', message)

var data = { 'Js Responds':'Web!' }

log('JS responding with', data)

responseCallback(data)

})


//初始化方法,注入默认的消息处理器 ,默认的消息处理器用于在处理来自objc的消息时,如果该消息没有设置处理器,则采用默认处理器处理

function init(messageHandler) {

if (WebViewJavascriptBridge._messageHandler) { throw new Error('WebViewJavascriptBridge.init called twice') }

WebViewJavascriptBridge._messageHandler = messageHandler

var receivedMessages = receiveMessageQueue

receiveMessageQueue = null

for (var i=0; i<receivedMessages.length; i++) {

_dispatchMessageFromObjC(receivedMessages[i])

}

}


//内部方法:消息的发送

function _doSend(message, responseCallback) {

       //如果定义了回调

if (responseCallback) {

       //为回调对象产生唯一标识

var callbackId = 'cb_'+(uniqueId++)+'_'+new Date().getTime()

       //并存储到一个集合对象里

responseCallbacks[callbackId] = responseCallback

       //新增一个key-value- 'callbackId':callbackId

message['callbackId'] = callbackId

}

        //把发送的消息至于队列中,然后下面的函数通知oc来取message

sendMessageQueue.push(message)

        //产生一个srcurl scheme),供nativeshouldStartLoadWithRequest捕捉

messagingIframe.src = CUSTOM_PROTOCOL_SCHEME + '://' + QUEUE_HAS_MESSAGE

}



- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType {

    if (webView != _webView) { return YES; }

    NSURL *url = [request URL];

    __strong typeof(_webViewDelegate) strongDelegate = _webViewDelegate;

    if ([[url scheme] isEqualToString:kCustomProtocolScheme]) {

        if ([[url host] isEqualToString:kQueueHasMessage]) {

            [self _flushMessageQueue];//取出js放进去的消息,并处理

        } else {

            NSLog(@"WebViewJavascriptBridge: WARNING: Received unknown WebViewJavascriptBridge command %@://%@", kCustomProtocolScheme, [url path]);

        }

        return NO;

    } else if (strongDelegate && [strongDelegate respondsToSelector:@selector(webView:shouldStartLoadWithRequest:navigationType:)]) {

        return [strongDelegate webView:webView shouldStartLoadWithRequest:request navigationType:navigationType];

    } else {

        return YES;

    }

}

- (void)_flushMessageQueue {

       //直接调用js的函数取出消息

    NSString *messageQueueString = [_webView stringByEvaluatingJavaScriptFromString:@"WebViewJavascriptBridge._fetchQueue();"];

    

    id messages = [self _deserializeMessageJSON:messageQueueString];

    if (![messages isKindOfClass:[NSArray class]]) {

        NSLog(@"WebViewJavascriptBridge: WARNING: Invalid %@ received: %@", [messages class], messages);

        return;

    }

    for (WVJBMessage* message in messages) {

        if (![message isKindOfClass:[WVJBMessage class]]) {

            NSLog(@"WebViewJavascriptBridge: WARNING: Invalid %@ received: %@", [message class], message);

            continue;

        }

        [self _log:@"RCVD" json:message];


        NSString* responseId = message[@"responseId"];

        if (responseId) {

            WVJBResponseCallback responseCallback = _responseCallbacks[responseId];//获取到之前保存的回调函数

            responseCallback(message[@"responseData"]);

            [_responseCallbacks removeObjectForKey:responseId];//删除回调函数

        } else {

            WVJBResponseCallback responseCallback = NULL;

            NSString* callbackId = message[@"callbackId"];

            if (callbackId) {

                responseCallback = ^(id responseData) {

                    WVJBMessage* msg = @{ @"responseId":callbackId, @"responseData":responseData };

                    [self _queueMessage:msg];

                };

            } else {

                responseCallback = ^(id ignoreResponseData) {

                    // Do nothing

                };

            }

            

            WVJBHandler handler;

            if (message[@"handlerName"]) {

                handler = _messageHandlers[message[@"handlerName"]];

                if (!handler) {

                    NSLog(@"WVJB Warning: No handler for %@", message[@"handlerName"]);

                    return responseCallback(@{});

                }

            } else {

                handler = _messageHandler;

            }

            

            @try {

                id data = message[@"data"];

                handler(data, responseCallback);

            }

            @catch (NSException *exception) {

                NSLog(@"WebViewJavascriptBridge: WARNING: objc handler threw. %@ %@", message, exception);

            }

        }

    }

}


    //获得队列,将队列中的每个元素用分隔符分隔之后连成一个字符串【native端调用】

function _fetchQueue() {

var messageQueueString = JSON.stringify(sendMessageQueue)

sendMessageQueue = []

return messageQueueString

}


总结oc把oc的data和回调函数的ID发给js,js返回js的data和回调函数的ID,这样就完成了交互。


UDP是一种无连接的协议,因此在iOS中使用UDP发送和接收数据与TCP有所不同。以下是OC实现UDP发送和接收的基本步骤: 发送UDP包: 1. 创建UDP socket: ``` int udpSocket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); ``` 2. 设置发送地址和端口: ``` struct sockaddr_in serverAddr; memset(&serverAddr, 0, sizeof(serverAddr)); serverAddr.sin_family = AF_INET; serverAddr.sin_addr.s_addr = inet_addr("目标IP地址"); serverAddr.sin_port = htons(目标端口); ``` 3. 发送数据: ``` NSString *message = @"hello"; const char *msg = [message UTF8String]; sendto(udpSocket, msg, strlen(msg), 0, (struct sockaddr *)&serverAddr, sizeof(serverAddr)); ``` 接收UDP包: 1. 创建UDP socket: ``` int udpSocket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); ``` 2. 设置socket端口和地址: ``` struct sockaddr_in localAddr; memset(&localAddr, 0, sizeof(localAddr)); localAddr.sin_family = AF_INET; localAddr.sin_addr.s_addr = htonl(INADDR_ANY); localAddr.sin_port = htons(本地端口); bind(udpSocket, (struct sockaddr *)&localAddr, sizeof(localAddr)); ``` 3. 接收数据: ``` char buffer[1024]; struct sockaddr_in clientAddr; socklen_t clientAddrLen = sizeof(clientAddr); ssize_t recvLen = recvfrom(udpSocket, buffer, sizeof(buffer), 0, (struct sockaddr *)&clientAddr, &clientAddrLen); if (recvLen > 0) { NSString *message = [[NSString alloc] initWithBytes:buffer length:recvLen encoding:NSUTF8StringEncoding]; } ``` 注意:在使用UDP协议时,需要注意数据包的大小和可靠性问题,因为UDP是无连接的,数据包可能会丢失或乱序。在实际应用中,需要根据具体情况选择合适的协议和方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值