当前位置:   article > 正文

OpenHarmony教程指南-性能示例_openharmony 教程

openharmony 教程

介绍

本示例集成了条件渲染、动态加载以及HiDumper等场景来介绍如何提升应用性能。

效果预览

在这里插入图片描述

HiDumper使用说明:

1.点击性能示例主页的HiDumper按钮,进入HiDumper查看组件信息场景页。
1.点击HiDumper查看组件信息场景页的查看应用组件树进入场景页。
2.点击HiDumper查看组件信息场景页的查看if/else组件按钮,在场景中点击显隐切换按钮,查看图片显示隐藏。
3.点击HiDumper查看组件信息场景页的查看visibility属性按钮,在场景中依次点击Visible、Hidden、Visible、None,查看图片显示和隐藏。

常规加载页面使用说明:

1.点击性能示例主页的StaticImport按钮,进入常规加载入口场景页。
2.点击常规加载入口的点击跳转常规加载案例按钮,进入常规加载主页
3.点击常规加载主页的pageOne按钮,进入常规加载子页面

动态加载页面使用说明:

1.点击性能示例主页的DynamicImport按钮,进入动态加载入口场景页。
2.点击动态加载入口页的点击跳转动态加载案例按钮,进入动态加载主页
3.点击动态加载主页的pageOne按钮,进入动态加载子页面

ifOrVisibility使用说明:

1.点击性能示例主页的ifOrVisibility按钮,进入合理选择条件渲染合显隐控制场景页,其下有四组场景,八个页面。
2.分别点击频繁切换显隐控制页面 ,频繁切换条件渲染页面 ,进入频繁切换场景,可以通过点击Switch visible and hidden按钮,实现组件显示与隐藏间的切换,对比完成时延。
3.分别点击首页渲染条件渲染页面 ,首页渲染显隐控制页面,进入首页渲染场景,可以通过点击Show the Hidden on start按钮,可以显示首页渲染时,初始被隐藏的组件,对比首页加载渲染完成时延。
4.分别点击部分修改条件渲染+容器限制页面部分修改条件渲染模块,进入部分修改场景,可以通过点击Switch Hidden and Show按钮,可以对长列表中Text组件做部分修改,对比完成时延。
5.分别点击复杂子树条件渲染+组件复用页面、复杂子树:条件渲染页面,进入复杂子树场景,可以通过点击Change FlexAlign按钮,可以Flex组件中子组件布局方式,对比布局修改完成时延。

Grid高性能使用说明:

1.点击性能示例主页的使用LazyFoEach,cachedCount,组件复用按钮,进入懒加载+cachedCount+组件复用页面场景页。
2.点击性能示例主页的使用columnStart按钮,进入使用columnStart,columnEnd页面场景页,点击scrollToIndex:1900按钮,通过性能打点方式,查看滑动耗时。
3.点击性能示例主页的使用GridLayoutOptions按钮,进入使用GridLayoutOptions页面场景页,点击scrollToIndex:1900按钮,通过性能打点方式,查看滑动耗时。

SmartPerfEditor使用说明:

1.打开显隐切换页面,点击显示/隐藏Image按钮,使用DrawingDoc录制回放功能抓取绘制信息。

SmartPerfHost使用说明:

1.点击smartPerfHost示例主页的SmartPerfHost按钮,进入SmartPerfHost查看组件信息场景页。
2.点击SmartPerfHost查看FrameTimeline优化前场景页的FrameTimeline优化前代码按钮进入场景页。
3.点击SmartPerfHost查看FrameTimeline优化后场景页的FrameTimeline优化后代码按钮进入场景页。
4.点击SmartPerfHost查看AppStartup优化前场景页的AppStartup优化前代码按钮进入场景页。
5.点击SmartPerfHost查看AppStartup优化前场景页的AppStartup优化后代码按钮进入场景页。

Trace使用说明:

1.点击性能示例主页的Trace按钮,进入懒加载示例场景页。

BackgroundTask使用说明:

1.点击性能示例主页的BackgroundTask按钮,进入后台任务场景页,包括短时任务长时任务两个子页面,同意授予通知权限。
2.点击短时任务按钮进入场景页,点击开始计算按钮执行任务。
3.点击长时任务按钮进入场景页,授予通知权限和定位权限,和点击开启定位服务按钮开启定位,点击关闭定位服务关闭定位。

WaterFlow使用说明:

1.点击性能示例主页的WaterFlow按钮,进入WaterFlow场景页

多线程共享内存页面使用说明:

1.点击性能示例主页的MemoryShared按钮,进入多线程共享内存入口场景页。
2.点击多线程共享内存入口页的原子操作对比按钮,进入原子操作主页
3.点击多线程共享内存入口页的锁的使用按钮,进入锁的使用子页面

Native跨线程调用使用说明:

1.点击性能示例主页的NativeThreadsCallJS按钮,进入Native跨线程调用场景页
2.点击Native跨线程调用页的threadSafeTest按钮,验证使用安全函数跨线程调用。
3.点击Native跨线程调用页的libuvTest按钮,验证使用libuv跨线程调用。

多线程间大数据传输使用说明:

1.点击性能示例主页的ThreadDataTransfer按钮,进入多线程数据传输入口场景页。
2.拖动任务数滑动条调整线程任务数,点击传参方式按钮,改变线程传参方式。
3.拖动饱和度滑动条,调整图片饱和度。

主线程和子线程的通信使用说明:

1.点击性能示例主页面的ThreadCommunication按钮,进入主线程和子线程通信场景页。
2.点击场景一:独立的耗时任务按钮,进入独立的耗时任务场景页,点击加载图片按钮。
3.点击场景二:多个任务执行结果统一返回按钮,进入多个任务执行结果统一返回场景页,点击加载图片按钮。
4.点击场景三:TaskPool和主线程的即时通信按钮,进入TaskPool和主线程的即时通信场景页,点击加载图片按钮。
5.点击场景四:Worker和主线程的即时通信按钮,进入Worker和主线程的即时通信场景页,点击将图片变成4个按钮。
6.点击场景五:子线程同步调用主线程的接口按钮,进入子线程同步调用主线程的接口场景页,点击加载图片按钮。

Web组件提升性能使用说明:

1.连接网络后点击性能示例主页的WebPerformance按钮,进入Web组件提升性能场景页。
2.点击常规Web首页按钮进入常规加载Web首页。
3.点击进入网页按钮进入常规Web组件页。
4.点击下一页按钮进入常规Web切换页。
5.点击优化Web首页按钮进入优化加载Web首页。
6.点击进入网页按钮进入优化Web组件页。
7.点击下一页按钮进入优化Web切换页。

工程目录

  1. features
  2. |---/dynamicImport/src/main/ets // 动态加载HAR共享包
  3. |---|---/pages
  4. | | |---DynamicContentPageOne.ets // 动态加载子页面
  5. | | |---DynamicEntryView.ets // 动态加载入口页面
  6. | | |---DynaHome.ets // 动态加载主页
  7. |---/grid/src/main/ets // grid类功能HAR共享包
  8. | |---/view
  9. | | |---GridView.ets // 场景主页面模块
  10. | | |---GridLazyForEachView.ets // 懒加载+cachedCount+组件复用模块
  11. | | |---GridColumnStartView.ets // 使用columnStart,columnEnd模块
  12. | | |---GridLayoutOptionsView.ets // 使用GridLayoutOptions模块
  13. |---/hiDumper/src/main/ets // 列表类功能HAR共享包
  14. | |---/view
  15. | | |---ConditionView.ets // 显隐切换模块
  16. | | |---GridView.ets // 查看组件树模块
  17. | | |---HidumperHomeView.ets // 场景主页面模块
  18. | | |---VisibilityView.ets // 查看Visibility属性模块
  19. |---/ifOrVisibility/src/main/ets // 列表类功能HAR共享包
  20. | |---/util
  21. | | |---/Constants.ets // 常量
  22. | |---/model
  23. | | |---/SceneData.ets // 数据类型
  24. | | |---/SceneMsg.ets // 本地数据
  25. | |---/view
  26. | | |---HidumperHomeView.ets // 场景主页面模块
  27. | | |---/IfForStartUp
  28. | | | |---BetterUseIf.ets // 首页渲染:条件渲染模块
  29. | | | |---WorseUseVisibility.ets // 首页渲染:显隐控制模块
  30. | | |---/Reusable
  31. | | | |---AlignContentFlex.ets // Text子组件
  32. | | | |---IfWithoutReusable.ets // 复杂子树:条件渲染+模块
  33. | | | |---IfWithReusable.ets // 复杂子树:条件渲染+组件复用模块
  34. | | | |---MockComplexReusableSubBranch.ets // 模拟复用复杂子组件
  35. | | | |---MockComplexSubBranch.ets // 模拟复杂子组件
  36. | | |---/StackForRenderControl
  37. | | | |---RenderControlWithoutStack.ets // 部分修改:条件渲染+容器限制模块
  38. | | | |---RenderControlWithStack.ets // 部分修改:条件渲染模块
  39. | | |---/VisibilityForAnimate
  40. | | | |---BetterUseVisibility.ets // 频繁切换:显隐控制模块
  41. | | | |---WorseUseIf.ets // 频繁切换:条件渲染
  42. |---/smartPerfEditor/src/main/ets // 显隐类功能HAR共享包
  43. | |---/view
  44. | | |---SmartPerfEditorView.ets // 显隐切换模块
  45. |---/smartPerfHost/src/main/ets // SmartPerfHost功能HAR共享包
  46. | |---/view
  47. | | |---AppStartupBeforeOptimization.ets // 冷启动耗时优化前模块
  48. | | |---AppStartupAfterOptimization.ets // 冷启动耗时优化后模块
  49. | | |---FrameTimelineBeforeOptimization.ets // 丢帧测试优化前模块
  50. | | |---FrameTimelineAfterOptimization.ets // 丢帧测试优化后模块
  51. | | |---SmartPerfHostHomeView.ets // 场景主页面模块
  52. |---/staticImport/src/main/ets // 常规加载HAR共享包
  53. |---|---/pages
  54. | | |---StaticContentPageOne.ets // 常规加载子页面
  55. | | |---StaticContentPageTwo.ets // 常规加载子页面
  56. | | |---StaticEntryView.ets // 常规加载入口页面
  57. | | |---StaticHome.ets // 常规加载主页
  58. |---/ThreadCommunication/src/main/ets // 主线程和子线程通信
  59. |---|---/pages
  60. | | |---IconItemSource.ets // 图片信息类
  61. | | |---IconView.ets // 图片Item组件
  62. | | |---IndependentTask.ets // 独立的耗时任务页面
  63. | | |---MultiTask.ets // 多任务统一返回页面
  64. | | |---TaskSendDataUsage.ets // TaskPool和主线程的即时通信页面
  65. | | |---ThreadCommunicationHomePage.ets // 主线程和子线程通信入口页面
  66. | | |---WorkerCallGlobalUsage.ets // 子线程同步调用主线程的接口页面
  67. | | |---WorkerUsage.ets // Worker和主线程的即时通信页面
  68. |---/trace/src/main/ets // Trace功能HAR共享包
  69. |---|---/pages
  70. | | |---LazyForEachPage.ets // 懒加载示例场景页面
  71. |---|---/view
  72. | | |---IconView.ets // 懒加载示例场景子页面
  73. |---|---/viewmodel
  74. | | |---BasicDataSource.ets // 封装列表数据方法
  75. |---/backgroundTask/src/main/ets // backgroundTask类功能HAR共享包
  76. | |---/view
  77. | | |---BackgroundTaskHomeView.ets // 场景主页面模块
  78. | | |---TransientTaskView.ets // 短时任务页面
  79. | | |---LongTermTaskView.ets // 长时任务页面
  80. ---/waterFlow/src/main/ets // waterFlow功能HAR共享包
  81. | |---/pages
  82. | | |---Index.ets // 场景主页面模块
  83. | | |---WaterFlowDataSource.ets // 数据源
  84. ---/nativeThreadsCallJS/src/main/ets // native跨线程调用功能HAR共享包
  85. | |---/pages
  86. | | |---Index.ets // 场景主页面模块
  87. ---/nativeThreadsCallJS/src/main/cpp // native跨线程调用功能native功能模块
  88. | |---/types
  89. | | |---hello.cpp // 场景native功能代码
  90. products/phone/entry/src/main/ets
  91. |---/entryability
  92. | |---EntryAbility.ts // 封装整个模块启用,监听Ability对应的窗口等逻辑
  93. |---/model
  94. | |---/DataType.ets // 数据类型
  95. | |---/MockData.ets // 本地数据
  96. |---pages
  97. | |---/dynamicImport
  98. | | |---DynamicEntryView.ets // 动态加载入口页面
  99. | | |---DynamicHome.ets // 动态加载主页
  100. | | |---DynamicContentPageOne.ets // 动态加载内容页
  101. | |---/grid
  102. | | |---GridPage.ets // 场景主页面
  103. | | |---GridLazyForEachPage.etss // 懒加载+cachedCount+组件复用页面
  104. | | |---GridColumnStartPage.ets // 使用columnStart,columnEnd页面
  105. | | |---GridLayoutOptionsPage.ets // 使用GridLayoutOptions页面
  106. | |---/hiDumper
  107. | | |---ConditionPage.ets // 显隐切换页面
  108. | | |---GridPage.ets // 查看组件树页面
  109. | | |---HidumperHomePage.ets // 场景主页面
  110. | | |---VisibilityPage.ets // 查看Visibility属性页面
  111. | |---/ifOrVisibility
  112. | | |---IfOrVisibilityPage.ets // 合理选择条件渲染首页
  113. | | |---AnimateBetterUseVisibility.ets // 频繁切换:显隐控制页面
  114. | | |---AnimateWorseUseIf.ets // 频繁切换:条件渲染页面
  115. | | |---IfBetterWithReusable.ets // 首页渲染:条件渲染页面
  116. | | |---IfWorseWithoutReusable.ets // 首页渲染:显隐控制页面
  117. | | |---RenderControlBetterWithStack.ets // 部分修改:条件渲染+容器限制页面
  118. | | |---RenderControlWorseWithoutStack.ets // 部分修改:条件渲染页面
  119. | | |---StartBetterUseIf.ets // 复杂子树:条件渲染+组件复用页面
  120. | | |---StartWorseUseVisibility.ets // 复杂子树:条件渲染页面
  121. | |---/smartPerfEditor
  122. | | |---SmartPerfEditorPage.ets // 显隐切换页面
  123. | |---/smartPerfHost
  124. | | |---AppStartupBeforeOptimizationPage.ets // 冷启动耗时优化前页面
  125. | | |---AppStartupAfterOptimizationPage.ets // 冷启动耗时优化后页面
  126. | | |---FrameTimelineBeforeOptimizationPage.ets // 丢帧测试优化前页面
  127. | | |---FrameTimelineAfterOptimizationPage.ets // 丢帧测试优化后页面
  128. | | |---SmartPerfHostPage.ets // 场景主页面
  129. | |---/staticImport
  130. | | |---StaticEntryView.ets // 常规加载入口页面
  131. | | |---StaticHome.ets // 常规加载主页
  132. | |---/backgroundTask
  133. | | |---BackgroundTaskPage.ets // 后台任务入口页面
  134. | | |---TransientTask.ets // 短时任务页面
  135. | | |---LongTermTask.ets // 长时任务页面
  136. | |---/ThreadCommunication
  137. | |---|---/workers
  138. | | |---|---Worker.ts // worker子线程
  139. | | |---IndependentTaskPage.ets // 独立的耗时任务页面
  140. | | |---MultiTaskPage.ets // 多任务统一返回页面
  141. | | |---TaskSendDataUsagePage.ets // TaskPool和主线程的即时通信页面
  142. | | |---ThreadCommunicationHomePage.ets // 主线程和子线程通信入口页面
  143. | | |---WorkerCallGlobalUsagePage.ets // 子线程同步调用主线程的接口页面
  144. | | |---WorkerUsagePage.ets // Worker和主线程的即时通信页面
  145. | |---/trace
  146. | | |---TracePage.ets // 懒加载示例首页
  147. | |---/ThreadDataTransfer
  148. | | |---ThreadDataTransferHomePage.ets // 线程间大数据传输首页
  149. | |---/waterFlow
  150. | | |---WaterFlowPage.ets // WaterFlow示例首页
  151. | |---/memoryShared
  152. | | |---AtomicsUsage.ets // 原子操作页面
  153. | | |---LockUsage.ets // 锁应用页面
  154. | | |---MemorySharedHome.ets // 多线程共享内存入口页面
  155. | |---/nativeThreadsCallJS
  156. | | |---NativeThreadsCallJS.ets // native跨线程调用示例首页
  157. | |---/webPerformance
  158. | | |---WebHomePage.ets // Web提升性能示例首页
  159. | | |---WebBrowserPage.ets // Web组件网页
  160. | | |---WebInitializedPage.ets // Web优化性能主页
  161. | | |---WebUninitializedPage.ets // Web常规主页
  162. | |---/Index
  163. |---/utils
  164. | |---Logger.ets // 封装整个日志

具体实现

  • HiDumper场景模块
  • 查看组件树
  • 在页面上添加Grid,每个GridItem嵌套多个Stack,然后通过HiDumper抓取组件树信息。
  • 源码链接:GridView.ets
  • 查看if/else组件
  • 在页面上添加显隐切换按钮,添加一张图片,点击按钮后通过HiDumper抓取组件树信息。
  • 源码链接:ConditionView.ets
  • 查看visibility属性
  • 在页面上添加Visible、Hidden、None按钮,添加一张图片,点击按钮后通过HiDumper抓取组件树信息。
  • 源码链接:VisibilityView.ets
  • Navigation常规加载页面模块
  • 在Navigation组件加载内容页时,使用import常规加载子页面。
  • 源码链接:StaticHome.ets
  • Navigation动态加载页面模块
  • 在Navigation组件加载内容页时,使用await import实现动态按需加载。
  • 源码链接:DynamicHome.ets
  • IfOrVisibility四个场景,分别正反例,八个页面:
  • 针对显示和隐藏间频繁切换的场景
  • 反例:使用条件循环实现显示和隐藏间的切换; WorseUseIf.ets
  • 正例:使用显隐控制实现显示和隐藏间的切换; BetterUseVisibility.ets
  • 针对应用冷启动,加载绘制首页
  • 反例:对于首页初始时,不需要显示的组件,通过显隐控制进行隐藏; WorseUseVisibility.ets
  • 正例:对于首页初始时,不需要显示的组件,通过条件渲染进行隐藏; BetterUseIf.ets

针对反复切换条件渲染的控制分支,但切换项仅涉及页面中少部分组件的场景

  • 反例:没有使用容器限制条件渲染组件的刷新范围; RenderControlWithoutStack.ets
  • 正例:使用容器限制条件渲染组件的刷新范围; RenderControlWithStack.ets
  • 针对反复切换条件渲染的控制分支,且控制分支中的每种分支内,组件子树结构都比较复杂的场景
  • 反例:没有使用组件复用实现条件渲染控制分支中的复杂子组件; IfWithoutReusable.ets
  • 正例:使用组件复用实现条件渲染控制分支中的复杂子组件; IfWithReusable.ets
  • 高性能Grid二个场景,三个页面:
  • 针对Grid中使用懒加载,cachedCount,组件复用的场景
  • 在页面上添加Grid,GridItem使用懒加载,设置cachedCount,同时对GridItem中的子组件使用组件复用;GridColumnStartView.ets
  • 针对Grid中使用scrollToIndex滑动到指定位置的场景
  • 反例:使用columnStart,columnEnd设置GridItem大小,使用scrollToIndex滑动到指定GirdItem,通过性能打点方式查看滑动耗时情况;GridColumnStartView.ets
  • 正例:使用GridLayoutOptions设置GridItem大小,使用scrollToIndex滑动到指定GirdItem,通过性能打点方式查看滑动耗时情况;
    GridLayoutOptionsView.ets
    Sm
  • artPerfEditor一个场景,一个页面:
  • 针对显示图片中查看是否存在冗余绘制的场景在页面上添加显隐切换按钮,点击按钮后,显示一张图片,通过DrawingDoc录制回放功能抓取绘制信息;SmartPerfEditorView.ets
  • SmartPerfHost页面模块
  • 应用冷启动,加载绘制页面应用优化前AppStartupBeforeOptimization.ets应用优化后AppStartupAfterOptimization.ets
  • 列表场景丢帧问题优化应用优化前FrameTimelineBeforeOptimization.ets应用优化后FrameTimelineAfterOptimization.ets
  • 后台任务页面模块
  • 短时任务申请短时任务,应用后台执行耗时计算任务源码链接:TransientTaskView.ets
  • 长时任务模拟后台导航定位场景,申请定位类型长时任务,使用@ohos.geoLocationManager实现位置定位功能,必须在联网环境才能获取定位数据源码链接:LongTermTaskView.ets
  • 多线程共享内存页面模块
  • 原子操作场景使用原子或非原子操作,实现多线程操作累加器源码链接:AtomicsUsage.ets
  • 锁应用场景使用或不适用锁,实现多线程写入文件源码链接:LockUsage.ets
  • 多线程间大数据传输页面模块
  • 图片调整饱和度通过不同参数,实现多线程修改图片源码链接:TreadUtil.ets
  • Web提升性能页面模块
  • 常规场景使用Web组件常规实现网页加载源码链接:WebUninitializedPage.ets
  • 优化场景使用预加载预连接提升web性能源码链接:WebInitializedPage.ets
  • 主线程和子线程间的通信
  • 将独立的耗时任务放在子线程中执行使用TaskPool实现源码链接:IndependentTask.ets
  • 多个任务一并返回结果使用TaskPool+TaskGroup实现源码链接:MultiTask.ets
  • Task和主线程的数据通信使用TaskPool中的sendData()和onReceiveData()接口实现源码链接:TaskSendDataUsage.ets
  • 子线程同步调用主线程的接口在Worker中使用callGlobalCallObjectMethod()接口实现源码链接:WorkerCallGlobalUsage.etsWorker.ts
  • Worker和主线程的数据通信使用Worker的postMessage()和onmessage()接口实现源码链接:WorkerUsage.etsWorker.ts
  • 相关权限

ohos.permission.INTERNETohos.permission.LOCATIONohos.permission.LOCATION_IN_BACKGROUNDohos.permission.APPROXIMATELY_LOCATIONohos.permission.KEEP_BACKGROUND_RUNNING

依赖

不涉及。

约束与限制

1.本示例已适配API version 10版本SDK,SDK版本号(API Version 10 Release),镜像版本号(4.0Release)。
2.本示例需要使用DevEco Studio 版本号(4.0Release)及以上版本才可编译运行。

下载

如需单独下载本工程,执行如下命令:

  1. git init
  2. git config core.sparsecheckout true
  3. echo code/Performance/PerformanceLibrary/ > .git/info/sparse-checkout
  4. git remote add origin https://gitee.com/openharmony/applications_app_samples.git
  5. git pull origin master

那么要想成为一名鸿蒙高级开发,以上知识点是必须要掌握的,除此之外,还需要掌握一些鸿蒙应用开发相关的一些技术,需要我们共同去探索。

为了能够让大家跟上互联网时代的技术迭代,在这里我特邀了几位行业大佬整理出一份最新版的鸿蒙学习提升资料,有需要的小伙伴自行领取,限时开源,先到先得~~~~

领取以下高清学习路线原图请点击→《鸿蒙 (Harmony OS)开发学习手册》纯血鸿蒙HarmonyOS基础技能学习路线图
在这里插入图片描述

领取以上完整高清学习路线图,请点击→《鸿蒙基础入门学习指南》小编自己整理的部分学习资料(包含有高清视频、开发文档、电子书籍等)
在这里插入图片描述

以上分享的学习路线都适合哪些人跟着学习?

-应届生/计算机专业
通过学习鸿蒙新兴技术,入行互联网,未来高起点就业。
-0基础转行
提前布局新方向,抓住风口,自我提升,获得更多就业机会。
-技术提升/进阶跳槽
发展瓶颈期,提升职场竞争力,快速掌握鸿蒙技术,享受蓝海红利。

在这里插入图片描述

最后

鸿蒙开发学习是一个系统化的过程,从基础知识的学习到实战技能的锤炼,再到对前沿技术的探索,每一环节都至关重要。希望这份教程资料能帮助您快速入门并在鸿蒙开发之路上步步攀升,成就一番事业。让我们一起乘风破浪,拥抱鸿蒙生态的广阔未来!

如果你觉得这篇内容对你有帮助,我想麻烦大家动动小手给我:点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。

关注我,同时可以期待后续文章ing,不定期分享原创知识。

想要获取更多完整鸿蒙最新VIP学习资料,请点击→《鸿蒙HarmonyOS分布式项目实战

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/304522?site
推荐阅读
相关标签
  

闽ICP备14008679号