WebViewJavascriptBridge 原理分析

WebViewJavascriptBridge 原理分析

网上好多都是在介绍 WebViewJavascriptBridge如何使用,这篇文章就来说说 WebViewJavascriptBridge 设计原理。

主要从两个过程来讲一下:js调用UIViewController中的代码(Native),Native调用js

1.概述

首先有两个问题:

a.Native(中的UIWebView)是否可以直接调用js method(方法)? 可以。

b.js 是否可以直接调用Native的mthod?不行。

明确上述两个问题,那么上图就不难明白了,webpage中的js method和webview本地的method之间关系。那WebViewJavascriptBridge出现是否解决这个问题(这个问题就是让js可以直接调用native的method)呢?答案是否定的?没有本质还是用uiwebview的代理方法进行字段拦截(判断url的scheme),实现js间接调用native的method。

我们来看WebViewJavascriptBridge提供的demo:

主要的核心是下面两个,接下来我们就来讨论一下其设计原理。

2.js调用Native method

在概述中说过,js是不能直接调用native的method所以,需要借助- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType,这个方法大家不陌生,每次在重新定向URL的时候,这个方法就会被触发,通常情况,我们会在这里做一些拦截完成js和本地的间接交互什么的。那么WebViewJavascriptBridge也不另外,也是这么做。

我们先来看看在ExampleApp.html文件中点击一个按钮发起请求的代码:


1

2

3

4

5

6

7

8

9

10

var callbackButton = document.getElementById(‘buttons‘).appendChild(document.createElement(‘button‘))

        callbackButton.innerHTML = ‘Fire testObjcCallback‘

        callbackButton.onclick = function(e) {

            e.preventDefault()

            log(‘JS calling handler "testObjcCallback"‘)

            //1

            bridge.callHandler(‘testObjcCallback‘, {‘foo‘: ‘cccccccccccc‘}, function(response) {

                log(‘JS got response‘, response)

            })

        }

估计大家大体都能看懂,唯独有疑问的地方是:

?


1

2

3

4

bridge.callHandler(‘testObjcCallback‘, {‘foo‘: ‘cccccccccccc‘}, function(response) {

                log(‘JS got response‘, response)

            })

        }

这段代码先不说,上面代码就是一个按钮的普通单击事件方法。我们一起想一下,如果这个按钮需要被点击之后调用native中的funtion函数,之后需要把这个(native的)funtion函数处理结果返回给js中的方法继续处理。这个是我们需求,带着这个需求我们看一下这个方法,testObjcCallBack这个我们猜测一下应该native中的方法或者一个能够调用到方法的name/id,后面这个是个json{‘foo’:‘ccccccccccccc’},应该是个参数,那么后面这个方法一看log应该知道,是对native返回的result进行处理的方法。拿具体是不是呢?只要找到callHandler方法就知道了。

在文件WebViewJavascriptBridge.js.txt里面我们找找这个方法:

?


1

2

3

function callHandler(handlerName, data, responseCallback) {

        _doSend({ handlerName:handlerName, data:data }, responseCallback)

    }

这里又多了一个方法叫_doSend连个参数 第1个是字典key-value定义,第二个是一个方法的指针(看看上面的方法你就知道了),那我们必须在同一个文件里面看看能不能找到这个_doSend方法:

?


1

2

3

4

5

6

7

8

9

function _doSend(message, responseCallback) {

        if (responseCallback) {

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

            responseCallbacks[callbackId] = responseCallback

            message[‘callbackId‘] = callbackId

        }

        sendMessageQueue.push(message)

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

    }

找到了。

逐行分析一下,变量callbackId是个字符串,responseCallBacks[] 一看就知道是个字典 ,这个字典把回掉(我们猜测)的方法responseCallback给保存起来,这Key(也就是callbackId)应该是唯一的,通过计数和时间应该知道这个字符串应该是唯一的,message也是一个字典,这是给message添加了一个新的key-value。干嘛呢?我也不知道,我们来看看sendMessageQueue是什么,大家一个push就知道应该是个数组。他吧一个字典放到一个消息队列中(数组队列),让后产生一个src(url scheme)。

有两个变量我们看看:

?


1

2

var CUSTOM_PROTOCOL_SCHEME = ‘wvjbscheme‘

var QUEUE_HAS_MESSAGE = ‘__WVJB_QUEUE_MESSAGE__‘

干嘛用,肯定是给webview 的 delegate判断用的,你感觉呢?(肯定是)

下面是在文件:WebViewJavascriptBridge.m

好了到了这里大家猜猜这个要干嘛?肯定是要发url让web截取对吧?那还用问啊,肯定是啊,已经说过了js能不能调用native的funtion函数?不能。我们来看看这个messagingIframe是:

?


1

2

3

4

5

6

function _createQueueReadyIframe(doc) {

    messagingIframe = doc.createElement(‘iframe‘)

    messagingIframe.style.display = ‘none‘

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

    doc.documentElement.appendChild(messagingIframe)

}

原来就是iframe,这个就不同给大家解释了。好了src一产生就会出现什么,uiwebview代理回掉截获,此时我们把目光回到UIWebview的Native下面:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

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

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

    NSURL *url = [request URL];

    __strong WVJB_WEBVIEW_DELEGATE_TYPE* strongDelegate = _webViewDelegate;

    

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

    {

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

        {

            //会走这里

            [self _flushMessageQueue];

        }

        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;

    }

}

一看就头大,哈哈,是,我也头大。看看上面的注释说 会走这里,我们看看为什么会走那里,最外圈的if([url scheme])判断是

#define kCustomProtocolScheme @"wvjbscheme"

这个定义是什么意思,我们先不做解释,刚才我们说过js不能直接调用native的function,大家只要记住这点,接着往下走就是了。至于为什么走这里,自己看代码(上文有提到),我们看看_flushMessageQueue:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

- (void)_flushMessageQueue {

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

    //json转成数组

    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];

        //用于js回掉

        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) {

                    if (responseData == nil) {

                        responseData = [NSNull null];

                    }

                    

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

                    [self _queueMessage:msg];

                };

            } else {

                responseCallback = ^(id ignoreResponseData) {

                    // Do nothing

                };

            }

            

            WVJBHandler handler;

            if (message[@"handlerName"]) {

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

            } else {

                handler = _messageHandler;

            }

            if (!handler) {

                [NSException raise:@"WVJBNoHandlerException" format:@"No handler for message from JS: %@", message];

            }

            

            handler(message[@"data"], responseCallback);

        }

    }

}

这下牛逼了,不忍直视啊!这么多,哈哈,多不可怕,可怕是你坚持不下去了。

我们逐行来看:

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

我们必须回去到js文件中去,这里是webview直接调用js中的方法:


1

2

3

4

5

function _fetchQueue() {

    var messageQueueString = JSON.stringify(sendMessageQueue)

    sendMessageQueue = []

    return messageQueueString

}

谢天谢地这个方法代码不多,这个消息很眼熟,SendMessageQueue,刚才我们说什么来?他是一个字典,那里面有哪些东西,我么来看看

handlerName:handlerName,

data:data,

callbackId:callbackId

这个消息字典此时被取出来准备做什么,这里提示下我们已经走到webview 的delegate里面了,所以拿到这些信息肯定是调用native的method对吧?肯定是的。接着往下走,接着会把json字符串转成数组,然后进行判断,

?


1

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

有没有responseid,你说又没,肯定没有啊(你不行看看上面),所以就这这里了

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

WVJBResponseCallback responseCallback = NULL;

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

            if (callbackId) {

                responseCallback = ^(id responseData) {

                    if (responseData == nil) {

                        responseData = [NSNull null];

                    }

                    

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

                    [self _queueMessage:msg];

                };

            } else {

                responseCallback = ^(id ignoreResponseData) {

                    // Do nothing

                };

            }

            

            WVJBHandler handler;

            if (message[@"handlerName"]) {

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

            } else {

                handler = _messageHandler;

            }

            if (!handler) {

                [NSException raise:@"WVJBNoHandlerException" format:@"No handler for message from JS: %@", message];

            }

            

            handler(message[@"data"], responseCallback);

这部分是重点,到底他是怎么要调用本地function的,callbackId大家熟悉吧,判断是否为空,不为空给他指定一个block,这个不说了,block指定,此时不调用(手动调用才会执行),这个刚才说了用来处理native的function处理的result用于把处理后的值返回给js的,接着往下去,看到handler这个方法会从message找到handlerName,这里我们看一下多了一个_messageHandlers字典,从这个字典获取一个block(WVJBHandler是一个block),直接执行了。那我们看看_messageHandlers是怎么被添加block的:


1

2

3

- (void)registerHandler:(NSString *)handlerName handler:(WVJBHandler)handler {

    _messageHandlers[handlerName] = [handler copy];

}

那又是谁调用了这个方法:

找到了(在文件 ExampleAppViewController.m的viewdidload中),这里有方法testObjecCallback


1

2

3

4

[_bridge registerHandler:@"testObjcCallback" handler:^(id data, WVJBResponseCallback responseCallback) {

    NSLog(@"testObjcCallback called: %@", data);

    responseCallback(@"Response from testObjcCallback");

}];

有点乱了。刚才我们的思路都是倒推的,如果我们整过来,首先肯定是viewdidload初始化,初始化之后会把这个block加入到_messageHandlers中,之后因为js调用动态读取这个block调用,在调用之前,我们又把定一个block付给回掉处理的responseCallback的block,这个block在handler中调用而调用,有点绕,自己可以多想想。

我们接着来看看:


1

2

3

4

5

6

7

8

responseCallback = ^(id responseData) {

                    if (responseData == nil) {

                        responseData = [NSNull null];

                    }

                    

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

                    [self _queueMessage:msg];

                };

这个就是你绕的地方,他是后被定义的,所以一开不执行,只有在处理数据后回调才会被调用,这里有个方法_queueMessage:


1

2

3

4

5

6

7

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

    if (_startupMessageQueue) {

        [_startupMessageQueue addObject:message];

    } else {

        [self _dispatchMessage:message];

    }

}

这里面还有个方法:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

- (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"];

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

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

    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];

        });

    }

}

我们在回到WebViewJavascriptBridge.js.txt文件中看到

?


1

2

3

4

5

6

7

function _handleMessageFromObjC(messageJSON) {

    if (receiveMessageQueue) {

        receiveMessageQueue.push(messageJSON)

    } else {//肯定走这个  为什么呢?

        _dispatchMessageFromObjC(messageJSON)

    }

}

再来看看:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

function _dispatchMessageFromObjC(messageJSON) {

        setTimeout(function _timeoutDispatchMessageFromObjC() {

            var message = JSON.parse(messageJSON)

            var messageHandler

            var responseCallback

            if (message.responseId) {

                responseCallback = responseCallbacks[message.responseId]

                if (!responseCallback) { return; }

                responseCallback(message.responseData)

                delete responseCallbacks[message.responseId]

            } else {

                if (message.callbackId) {

                    var callbackResponseId = message.callbackId

                    responseCallback = function(responseData) {

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

                    }

                }

                

                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)

                    }

                }

            }

        })

    }

大家还记得我们返回的对象是:

?


1

@{ @"responseId":callbackId, @"responseData":responseData }

所以这里messageHandlers刚才也说过了用来存方法的,callbackId被换了个名字叫responseId意思一样,只要值没变就行,所以就会执行:


1

2

3

bridge.callHandler(‘testObjcCallback‘, {‘foo‘: ‘cccccccccccc‘}, function(response) {

                log(‘JS got response‘, response)

            })

中的方法,好了,完了。

总结一下:js这边 先把方法名字、参数、处理方法保存成一个字典在转成json字符串,在通过UIWebview调用js中某个方法把这个json字符串传到Native中去(不是通过url传的,这样太low了),同时把这个处理的方法以key-value形式放到一个js的字典中。

UIWebView在收到这个json之后,进行数据处理、还有js的回掉的处理方法(就是那个callbackId)处理完成后也会拼成一个key-value字典通过调用js传回去(可以直接调用js)。

js在接到这个json后,根据responseId读取responseCallbacks中处理方法进行处理Native code返回的数据。

3.Native调用js method

过程不是直接调用js,也是通过js调用Native过程一样的处理方式。

大体来看一下,先看一个按钮的单击事件:


1

2

3

4

5

6

- (void)callHandler:(id)sender {

    id data = @{ @"greetingFromObjC": @"Hi there, JS!" };

    [_bridge callHandler:@"testJavascriptHandler" data:data responseCallback:^(id response) {

        NSLog(@"testJavascriptHandler responded: %@", response);

    }];

}

看看callHandler:


1

2

3

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

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

}

看看_sendData:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

- (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];

        _responseCallbacks[callbackId] = [responseCallback copy];

        message[@"callbackId"] = callbackId;

    }

    

    if (handlerName) {

        message[@"handlerName"] = handlerName;

    }

    [self _queueMessage:message];

}

到_queueMessage:之后流程就和上面一样了,这里面native也有个:


1

2

3

4

5

6

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

       if (responseId) {

           WVJBResponseCallback responseCallback = _responseCallbacks[responseId];

           responseCallback(message[@"responseData"]);

           [_responseCallbacks removeObjectForKey:responseId];

       }

这个和js中的处理思想是一样的。

总结:native将方法名、参数、回到的id放到一个对象中传给js。

js根据方法名字调用相应方法,之后将返回数据和responseId拼装,最后通过src 重定向到UIWebview 的delegate。

native得到数据后根据responseId调用事先装入_responseCallbacks的block,动态读取调用,从而完成交互。

时间: 2025-01-07 20:00:16

WebViewJavascriptBridge 原理分析的相关文章

kafka producer实例及原理分析

1.前言 首先,描述下应用场景: 假设,公司有一款游戏,需要做行为统计分析,数据的源头来自日志,由于用户行为非常多,导致日志量非常大.将日志数据插入数据库然后再进行分析,已经满足不了.最好的办法是存日志,然后通过对日志的分析,计算出有用的数据.我们采用kafka这种分布式日志系统来实现这一过程. 步骤如下: 搭建KAFKA系统运行环境 如果你还没有搭建起来,可以参考我的博客: http://zhangfengzhe.blog.51cto.com/8855103/1556650 设计数据存储格式

android脱壳之DexExtractor原理分析[zhuan]

http://www.cnblogs.com/jiaoxiake/p/6818786.html内容如下 导语: 上一篇我们分析android脱壳使用对dvmDexFileOpenPartial下断点的原理,使用这种方法脱壳的有2个缺点: 1.  需要动态调试 2.  对抗反调试方案 为了提高工作效率, 我们不希望把宝贵的时间浪费去和加固的安全工程师去做对抗.作为一个高效率的逆向分析师, 笔者是忍不了的,所以我今天给大家带来一种的新的脱壳方法——DexExtractor脱壳法. 资源地址: Dex

android脱壳之DexExtractor原理分析

导语: 上一篇我们分析android脱壳使用对dvmDexFileOpenPartial下断点的原理,使用这种方法脱壳的有2个缺点: 1.  需要动态调试 2.  对抗反调试方案 为了提高工作效率, 我们不希望把宝贵的时间浪费去和加固的安全工程师去做对抗.作为一个高效率的逆向分析师, 笔者是忍不了的,所以我今天给大家带来一种的新的脱壳方法--DexExtractor脱壳法. 资源地址: DexExtractor源码:https://github.com/bunnyblue/DexExtracto

Adaboost算法原理分析和实例+代码(简明易懂)

Adaboost算法原理分析和实例+代码(简明易懂) [尊重原创,转载请注明出处] http://blog.csdn.net/guyuealian/article/details/70995333     本人最初了解AdaBoost算法着实是花了几天时间,才明白他的基本原理.也许是自己能力有限吧,很多资料也是看得懵懵懂懂.网上找了一下关于Adaboost算法原理分析,大都是你复制我,我摘抄你,反正我也搞不清谁是原创.有些资料给出的Adaboost实例,要么是没有代码,要么省略很多步骤,让初学者

Android视图SurfaceView的实现原理分析

附:Android控件TextView的实现原理分析 来源:http://blog.csdn.net/luoshengyang/article/details/8661317 在Android系统中,有一种特殊的视图,称为SurfaceView,它拥有独立的绘图表面,即它不与其宿主窗口共享同一个绘图表面.由于拥有独立的绘图表面,因此SurfaceView的UI就可以在一个独立的线程中进行绘制.又由于不会占用主线程资源,SurfaceView一方面可以实现复杂而高效的UI,另一方面又不会导致用户输

AbstractQueuedSynchronizer的介绍和原理分析(转)

简介 提供了一个基于FIFO队列,可以用于构建锁或者其他相关同步装置的基础框架.该同步器(以下简称同步器)利用了一个int来表示状态,期望它能够成为实现大部分同步需求的基础.使用的方法是继承,子类通过继承同步器并需要实现它的方法来管理其状态,管理的方式就是通过类似acquire和release的方式来操纵状态.然而多线程环境中对状态的操纵必须确保原子性,因此子类对于状态的把握,需要使用这个同步器提供的以下三个方法对状态进行操作: java.util.concurrent.locks.Abstra

linux中mmap系统调用原理分析与实现

参考文章:http://blog.csdn.net/shaoguangleo/article/details/5822110 linux中mmap系统调用原理分析与实现 1.mmap系统调用(功能)      void* mmap ( void * addr , size_t len , int prot , int flags ,int fd , off_t offset )      内存映射函数mmap, 负责把文件内容映射到进程的虚拟内存空间, 通过对这段内存的读取和修改,来实现对文件的

Android 4.4 KitKat NotificationManagerService使用详解与原理分析(一)__使用详解

概况 Android在4.3的版本中(即API 18)加入了NotificationListenerService,根据SDK的描述(AndroidDeveloper)可以知道,当系统收到新的通知或者通知被删除时,会触发NotificationListenerService的回调方法.同时在Android 4.4 中新增了Notification.extras 字段,也就是说可以使用NotificationListenerService获取系统通知具体信息,这在以前是需要用反射来实现的. 转载请

一个日期算法的原理分析

1.问题描述 在 OSC 问答频道有一个问题:时间算法:帮忙解答下 简单的复述一遍就是能够通过如下式子来计算month月day日是一年的第几天. 闰年是 day_of_year=(275*month)/9 - (month+9)/12 + day - 30 非闰年比这个少1天.可以简单的验证,这个式子中每个部分计算后都取整,整个结果总是对的. 我们知道1.3.5.7.8.10.12都是31天,2月的天数有点诡异,其他都是30天,正常情况下我们写程序会写很多if来判断月份,进而计算累积的天数.但是