iOS网络性能监控

现在的Native App平台化趋势越来越明显,网络层架构也越来越复杂。一个App基本都有多个不同的网络模块。 从简单的业务数据的HTTP/HTTPS(基于NSURLConnection或者NSURLSession),到WebView的WebCore网络层,到基于TCP长连接的推送模块,到各种第三方组件比如统计、日志上报各自的网络层,或者很多app采用基于TCP的私有协议等等,网络层越来越复杂,对Native开发者来说越来越像一个黑盒模块。 Native开发者只能着眼于业务开发,对网络层的异常、性能等等问题一无所知。

初识iOS网络层API

让我们剥开网络相关的SDK,一层一层地看每一层做了些什么。

AFNetworking

AFNetworking是对NSURLConnection/NSURLSession的封装。增加了如下逻辑

  • 封装成NSOperation的形式,提供了resume/cancel等等处理。
  • 增加了NSData的文件处理,上传/下载
  • 方便处理JSON/XML数据
  • 方便处理HTTPS
  • 有Reachablity的API

NSURLSession/NSURLConnection

NSURLSession/NSURLConnection 都是基于CFNetwork的。

NSURLConnection - CFURLConnection的封装。提供了create,start,cancel,send(同步或者异步),设置回调,设置runloop等函数。

NSURLSession/NSURLSessionTask - NSCFURLSession/NSCFURLSessionTask等等的封装。

NSURLXXX这一层主要处理了:

  • 把CFNetwork的blockhandler封装成delegate的方式
  • 处理NSURLProtocol相关的代理
  • 处理NSURLCache的缓存相关,是对CFURLCache的封装
  • 封装sendAsync/和sendSync方法。
  • 把CFURLResponse的statusCode转化成String

CFNetwork

CFNetwork 展示了如何把字节流封装成HTTP协议的请求收发。

[图片上传失败...(image-4f1c4b-1535019216015)]

  • CFURLRequest由用户创建,里面包括URL/header/body这些请求的信息。然后CFURLRequest会被转换成CFHTTPMessage的格式。
  • CFHTTPMessage里主要是HTTP协议的定义和转换,把每一个请求request转换成标准的HTTP格式的文本。
  • CFURLConnection 里主要是处理请求任务,包括pthread线程、CFRunloop,请求队列的管理等等。所以提供了start、cancel等等操作的api。也有操作CFReadStream等API
  • CFHost:负责DNS,在有CFHostStartInfoResolution等函数,基于dns_async_startgetaddrinfo_async_start方法。在iOS8/9基于getaddrinfo。主要是同步调用和异步调用的区别。
  • CFURLCache/CFURLCredential/CFHTTPCookie:处理缓存/证书/cookie相关的逻辑,都有对应的NS类。

主要的数据交换调用基于CFStream的API。

CFStream

借助CFSocketStream,封装BSD Socket,和SecurityTransport(SSL调用)。

由于BSD Socket都是同步调用。所以CFStream这一层主要是Runloop逻辑,锁,dowhile等待等等。
类似BSD socket一样是数据流输入/输出的API。

CFStream 创建时要传入一堆callback,包括open/close,read/write等等。比如CFSocketStream,封装了BSD Socket的作为callback传入CFStream

CFSocketStream 也包括了DNS、SSL连接、Connect握手等等逻辑。

BSD Socket

有多组API,包括connect/shutdown,send/recv,read/write,recvfrom/sendto,recvmsg/sendmsg.
作为客户端一般不使用accept/bind.

send/recvread/write的区别在于多了一个flags参数。当flag为0时,send等同于write。

对于发送消息。send只可用于基于连接的套接字,sendtosendmsg既可用于无连接的socket,也可用于基于连接的socket。除了socket设置为非阻塞模式,调用将会阻塞直到数据被发送完。

DNS方法:
getaddrinfo 是对 gethostbyname/gethostbyaddr 的替代,支持了ipv6,返回一个地址struct链表。在iOS8/9中使用。
getaddrinfo_async_start 在iOS10中使用,支持了异步。

监控什么

iOS-Monitor-Platform 这篇文章提出了一些监控指标(然而他提供的方法并不能监控到)。

  • TCP 建立连接时间
  • DNS 时间
  • SSL 时间
  • 首包时间
  • 响应时间
  • HTTP 错误率
  • 网络错误率
  • 流量

APM厂听云提供的一些监控指标:

  • TOP5 响应时间最慢主机
  • TOP5吞吐率最高主机
  • TOP5 DNS时间最慢地域
  • TCP建连最慢主机
  • 连接次数最多主机

HTTP抓包工具Charles提供的监控指标:

  • Request Start Time
  • Request End Time
  • Response Start Time
  • Response End Time
  • Duration
  • DNS
  • Connect
  • SSL Handshake
  • Latency

当然如果可行的话,想每一个细节都监控到。但是很多数据都有实现成本,本文用最低的成本力求收集尽可能多的指标。

具体实现

HTTP 的监控

HTTP的监控最佳的实践当然就是利用NSURLSession的NSURLSessionTaskMetrics。

[图片上传失败...(image-7b6c8-1535019216015)]

想探究NSURLSessionTaskMetrics的实现,如果反编译CFNetwork的源码,可以看到-[NSURLSessionTaskMetrics _initWithPerformanceTiming] 这个方法,说明是来自一个叫TimingPerformance的类。
TimingPerformance的初始化方法代码如下,可以看到这里定义了所有NSURLSessionTaskMetrics时间节点需要的key,几乎完全一致。然后初始化时利用CFAbsoluteTimeGetCurrent函数来记录初始化的时间。

int __ZN17PerformanceTimingC2Ev() {
    rbx = rdi;
    CFObject::CFObject();
    ...
    *(rbx + 0x20) = @"_kCFNTimingDataRedirectStart";
    *(rbx + 0x30) = @"_kCFNTimingDataRedirectEnd";
    *(rbx + 0x40) = @"_kCFNTimingDataFetchStart";
    *(rbx + 0x50) = @"_kCFNTimingDataDomainLookupStart";
    *(rbx + 0x60) = @"_kCFNTimingDataDomainLookupEnd";
    *(rbx + 0x70) = @"_kCFNTimingDataConnectStart";
    *(rbx + 0x80) = @"_kCFNTimingDataConnectEnd";
    *(rbx + 0x90) = @"_kCFNTimingDataSecureConnectionStart";
    *(rbx + 0xa8) = @"_kCFNTimingDataRequestStart";
    *(rbx + 0xb8) = @"_kCFNTimingDataRequestEnd";
    *(rbx + 0xc8) = @"_kCFNTimingDataResponseStart";
    *(rbx + 0xd8) = @"_kCFNTimingDataResponseEnd";
    *(rbx + 0xe8) = @"_kCFNTimingDataRedirectCountW3C";
    *(rbx + 0xf8) = @"_kCFNTimingDataRedirectCount";
    *(rbx + 0x108) = @"_kCFNTimingDataTaskResumed";
    *(rbx + 0x118) = @"_kCFNTimingDataConnectCreate";
    *(rbx + 0x128) = @"_kCFNTimingDataTCPConnected";
    *(rbx + 0x138) = @"_kCFNTimingDataFirstWrite";
    *(rbx + 0x148) = @"_kCFNTimingDataFirstRead";
    *(rbx + 0x158) = @"_kCFNTimingDataConnectionInit";
    *(rbx + 0x168) = @"_kCFNTimingDataConnected";
    ....
    *(rbx + 0x1f0) = @"_kCFNTimingDataTimingDataInit";
    ...
    CFAbsoluteTimeGetCurrent();
    ....
    return rax;
}

这个类是怎么使用的呢,可以看到[NSCFURLSessionTask resume]这个方法里:

void -[__NSCFURLSessionTask resume](void * self, void * _cmd) {
    rbx = self;
            ...
            __setRecordForKeyInternalPerformanceTiming(@"streamTask-resume");
            r15 = rbx->_performanceTiming;
            if (r15 != 0x0) {
                    PerformanceTiming::Class();
                    xmm0 = intrinsic_movsd(xmm0, *(r15 + 0x110));
                    xmm0 = intrinsic_ucomisd(xmm0, 0x0);
                    if ((xmm0 == 0x0) && (!CPU_FLAGS & P)) {
                            CFAbsoluteTimeGetCurrent();
                            *(r15 + 0x110) = intrinsic_movsd(*(r15 + 0x110), xmm0);
                    }
            }
            __setRecordForKeyInternalPerformanceTiming(@"start-task-resume-to-loader-start-load");
              ...
    return;
}

可以看到这里rbx寄存器存储的就是NSCFURLSessionTask对象,这个对象有一个成员变量就是_performanceTiming,放在r15这个寄存器里。上面的代码可以看到(0x108)对应的就是_kCFNTimingDataTaskResumed这个key,而这里xmm0寄存器是个浮点数存储的寄存器,存储的是(r15 + 0x110),对应应该是,然后判断xmm0是否为空,如果是空的话,就调用CFAbsoluteTimeGetCurrent函数获取当前CPU时间,然后再赋给(r15 + 0x110),对应的应该就是_kCFNTimingDataTaskResumed这个key对应的value。

至于__setRecordForKeyInternalPerformanceTiming 这个函数,可以看到它的key并不存在于PerformanceTiming对象初始化的时候,它应该是InternalPerformanceTiming,这是个不同的类,可能是PerformanceTiming的子类。他的key是不同的,判断是这个库内部使用的,并没有作为NSURLSessionTaskMetrics传递出去。

发现__ZN17PerformanceTiming32fillW3NavigationTimingAWDMetricsEP27PerformanceTimingAWDMetrics,__ZN17PerformanceTiming30fillStreamTaskTimingAWDMetricsEP26StreamTaskTimingAWDMetrics这两个函数,说明PerformanceTimingW3NavigationTiming,以及StreamTaskTiming这几个东西的AWDMetrics是可以互相转化的。W3NavigationTiming很容易想到是用于WebView的Timing的API。

NSURLSessionTaskMetrics的优点是苹果帮咱实现了,但是有很严重的缺点是只能适用于iOS10以后的NSURLSession。NSURLConnection是用不了的。iOS10以下也是用不了的。 (见后文重大发现)

其它方案的分析

对于iOS10以下的NSURLSession以及NSURLConnection,想要打点统计时间点挺困难的,主要困难点在不同SDK的API调用不同,比如iOS8和9的DNS,可以hook到getaddrinfo函数,iOS10有时可以hook到getaddrinfo_async_start函数,但是对于iOS11,我尝试了各种跟DNS相关的函数,完全hook不到。反编译CFNetwok出来的跟DNS相关的函数,也都没有被NSURLSession/NSURLConnection调用。 SSL的情况也非常类似,目前只知道iOS8/9会通过SecurityTransport的SSLHandshake/SSLRead/SSLWrite等函数,但是iOS10以上就完全懵逼。这些尝试只能宣告失败,告一段落了。

有的文章认为是iOS10之后系统屏蔽了某些BSD Socket函数的hook,比如connect/read/write 等等。 据我观察并不是这样,BSD socket还是能够hook到,只是大部分情况下不调用这些API了,少数情况还是有使用的。 如果真的被屏蔽了,应该是完全hook不到的。

有些文章写了说监控HTTP,可以采用NSURLProtocol拦截请求的方式(比如听云)。我都是持怀疑态度的,因为监控性能,如果没有DNS/SSL相关的监控就失去了大部分意义,而监控request/response的就不需要用hook这种方式了(完全可以在自己封装的网络层部分实现)。 而针对于NSURL相关API的hook是统计不到DNS/SSL的,因为它们不在这一层实现。

也有些文章说hook CFStream的方式(比如网易APM)。 但是如果看到CFStream的实现就知道CFStream是对BSD Socket的封装,Open/Close/read/Write 如果看CFSocketStream的源码,这些API都还是BSD Socket实现的,就是说hook CFStream 和 BSD socket没有很大的区别。 还是没有hook到DNS/SSL的点上。

WebView 的监控

WebView的监控是相对简单的,主要是Timing API。

[图片上传失败...(image-56e48e-1535019216015)]

好处是兼容性很好,目前UIWebView和WKWebView都支持,iOS9以上都支持。因为是浏览器的API。

WebCore里,跟这个timing相关的API主要是PerformanceTiming类:

class PerformanceTiming : public RefCounted<PerformanceTiming>, public DOMWindowProperty {
public:
    static Ref<PerformanceTiming> create(Frame* frame) { return adoptRef(*new PerformanceTiming(frame)); }

    unsigned long long navigationStart() const;
    unsigned long long unloadEventStart() const;
    unsigned long long unloadEventEnd() const;
    unsigned long long redirectStart() const;
    unsigned long long redirectEnd() const;
    unsigned long long fetchStart() const;
    //...省略部分函数
    unsigned long long domContentLoadedEventStart() const;
    unsigned long long domContentLoadedEventEnd() const;
    unsigned long long domComplete() const;
    unsigned long long loadEventStart() const;
    unsigned long long loadEventEnd() const;

private:
    explicit PerformanceTiming(Frame*);
    const DocumentTiming* documentTiming() const;
    DocumentLoader* documentLoader() const;
    LoadTiming* loadTiming() const;
};

} // namespace WebCore

头文件里有一堆getter函数的定义,同时初始化方法只有一个,入参是单一的Frame对象,说明一个Frame对象就能够提供到这些所有的参数。

unsigned long long PerformanceTiming::requestStart() const
{
    DocumentLoader* loader = documentLoader();
    if (!loader)
        return connectEnd();

    const NetworkLoadMetrics& timing = loader->response().deprecatedNetworkLoadMetrics();
    ASSERT(timing.requestStart >= 0_ms);
    return resourceLoadTimeRelativeToFetchStart(timing.requestStart);
}
unsigned long long PerformanceTiming::domInteractive() const
{
   const DocumentTiming* timing = documentTiming();
   if (!timing)
       return 0;
   return monotonicTimeToIntegerMilliseconds(timing->domInteractive);
}
unsigned long long PerformanceTiming::loadEventStart() const
{
    LoadTiming* timing = loadTiming();
    if (!timing)
        return 0;
    return monotonicTimeToIntegerMilliseconds(timing->loadEventStart());
}

再看cpp文件就知道,PerformanceTiming是对Frame类中已经统计好的参数的一个封装,内部并没有逻辑。数据其实就是来自于NetworkLoadMetricsDocumentTimingLoadTiming 三部分。也很容易理解就是分别对应网络请求相关的性能统计、对应DOM加载相关的和WebView加载相关的性能统计。

有一个细节就是NetworkLoadMetrics里有0ms的判断,保证NetworkLoadMetrics返回的相关数据大于0。而DocumentTimingLoadTiming返回的数据为空时就是0。实际上使用这一系列数据时确实会出现一部分参数为0的情况,而且跟调用PerformanceTiming的接口有关。

WebCore的类的架构如下图。
[图片上传失败...(image-449013-1535019216015)]
那WebView里,网络是怎么一层层调用的呢? 追踪WKWebView的loadRequest:方法,调用栈应该是这样的:

- (WKNavigation *)loadRequest:(NSURLRequest *)request
void WebPage::loadRequest(const LoadParameters& loadParameters)
void UserInputBridge::loadRequest(FrameLoadRequest&& request, InputSource)
void FrameLoader::load(FrameLoadRequest&& request)
void FrameLoader::load(DocumentLoader* newDocumentLoader)
void FrameLoader::loadWithDocumentLoader(DocumentLoader* loader, FrameLoadType type, FormState* formState, AllowNavigationToInvalidURL allowNavigationToInvalidURL)
void FrameLoader::continueLoadAfterNavigationPolicy(const ResourceRequest& request, FormState* formState, bool shouldContinue, AllowNavigationToInvalidURL allowNavigationToInvalidURL)
void DocumentLoader::startLoadingMainResource()
void ResourceLoader::start()
void ResourceHandle::createNSURLConnection(id delegate, bool shouldUseCredentialStorage, bool shouldContentSniff, SchedulingBehavior, NSDictionary *connectionProperties);

ResourceLoader是资源加载,而真正操作网络请求的类在ResourceHandle。看代码就发现WebCore的网络层在iOS上也是基于NSURLConnection的。可以通过AOP的方式hook到某个位置,然后使用NSURLConnection的API进行操作。

有一处比较有意思,WebCore中实现了一个NSURLSession,叫WebCoreNSURLSession。这个类似乎只在MediaPlayer里面使用。相同的是他们也有类似的API,比如dataTaskWithRequest:等等,但是内部实现不一样,WebCoreNSURLSession也是基于ResourceLoader的子类。而NSURLSession是基于CFNetwork。

使用Timing系列的API也有需要注意的细节。WebCore内核在iOS上和在Mac的Safari上是不一样的。iOS10以后的WKWebView才实现.toJSON()。如果是UIWebView,或者是iOS10以下的WKWebView,需要先执行一段js脚本,方便我们把js对象转换为json。

NSString *funcStr = @"function flatten(obj) {"
        "var ret = {}; "
        "for (var i in obj) { "
        "ret[i] = obj[i];"
        "}"
        "return ret;}";
[webView stringByEvaluatingJavaScriptFromString:funcStr];

TCP的监控

一般App的网络层长连接,会基于TCP实现自定义的协议或者使用Websocket。有的app会基于BSD Socket封装(比如微信的mars)。有的会先利用一些开源的框架比如 CocoaAsyncSocket 或者 SocketRocket,然后再进行封装。

CocoaAsyncSocket

CocoaAsyncSocket是基于BSD Socket,CFStream,SecurityTransport的封装,封装成TCP/UDP协议。这几个API的共同之处在于都是数据流读写的形式。BSD Socket主要是同步阻塞,而CFStream是异步的。

既然是数据流读写,所以CocoaAsyncSocket肯定是包括数据流的处理和转换了。主要是缓冲区,ReadBuffer/WriteBuffer,判断读取的结尾CRLF, 读取的长度length和读取的超时机制等等。
CocoaAsyncSocket也封装了DNS、ipv4和ipv6、SSL等等逻辑。

SocketRocket

是基于NSStream 的封装,不同于CocoaAsyncSocket的传输层协议, 支持HTTP/WebSocket的应用层协议,定义了header的字段等等。
由于是基于数据流读写的,所以也包括readBuffer/WriteBuffer等数据处理逻辑。
也包括Runloop,线程等异步处理逻辑和阻塞同步逻辑。
还实现了PingPong这样的,跟服务端配合的保活逻辑。

所以TCP的监控可以hook BSD Socket 的API,包括 connect/disconnect/read/write等等调用,如果是同步调用,所以可以在执行函数前后埋点计算时间。
也需要hook DNS方面的API,比如 gethostbyname/getaddrinfo等同步调用的以及getaddrinfo_async_start等等异步调用的API。
也可以hook SSL 方面的API,比如 SSLHandshake/SSLRead/SSLWrite,实现对SSL连接的监测。

剧情反转,重大发现 (这一段是后来加的)

HTTP的性能监控因为NSURLSessionTaskMetrics的兼容性问题似乎已经穷途末路了,但是在写第二部分WebView的时候突然有了一个巨大的发现,这个发现来自于在看WebCore的源码的时候发现了一些神奇的东西。

#if !HAVE(TIMINGDATAOPTIONS)
void setCollectsTimingData()
{
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        [NSURLConnection _setCollectsTimingData:YES];
        ...
    });
}
#endif

这是说NSURLConnection本身有一套TimingData的收集API,只是没有暴露给开发者而已,但是WebCore里一直在用...苹果你为啥这么小气?!
然后就很轻易地在runtime header里找到了NSURLConnection的_setCollectsTimingData: API,还有_timingData的API。
这货iOS8以后都是支持的,iOS8之前也许也支持了。

那么NSURLSession呢,是不是也类似?果然。在iOS9之前,也只需要设置_setCollectsTimingData:就好了。
搜了一下google和github,我应该是第一个发现这个私有API的人...

所以很神奇地,很轻易地,就实现了NSURLConnection和NSURLSession全套的支持....

总结

我们几乎可以用很少的代码实现HTTP/WebView/TCP跨框架的大部分网络性能数据收集。如果把兼容性整理成一张表的话可以看到我们几乎支持了大部分的场景。

iOS SDK NSURLConnetion NSURLSession UIWebView WKWebView TCP
8.4 YES YES via TCP via TCP YES
9.3 YES YES YES YES YES
10.3 YES YES YES YES YES
11.3 YES YES YES YES YES

NetworkTracker 是我封装的一部分代码。并将监控结果简单地画了个图表。还是比较直观的。

Group.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容

  • 作者:敖志敏本文为原创文章,转载请注明作者及出处 国内移动网络环境非常复杂,WIFI、4G、3G、2.5G(Edg...
    沪江技术学院阅读 7,949评论 13 80
  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,975评论 3 119
  • 我喜欢水也热爱山。居家北国。相比南方而言,北方干旱水少但不乏名山大川;四季轮换而言,北国四季分明更彰山岳魅力。如果...
    山水依情阅读 284评论 0 2
  • 他说,以前的我,习惯性地选择宁愿被别人伤害,也不愿伤害别人,以为这就是解决问题的最好方法。后来的后来,我终于明白...
    林姝迩阅读 288评论 8 0
  • 五年前,因为个人学习需要去了趟北京。 当去北京的南京人在火车上遇到了来南京旅行的北京人时,话题就多了,于是一路聊到...
    石佛寺车站阅读 185评论 0 2