Android系统架构开篇(深度解析)(android的系统架构中包含有哪几层?分别有什么作用?)
ccwgpt 2024-10-11 11:11 34 浏览 0 评论
阅读前请点击右上角“关注”,每天免费获取Android知识解析及面试解答。Android架构解析,只做职场干货,完全免费分享!
原文作者:Gityuan
一、引言
本文作为Android系统架构的开篇,起到提纲挈领的作用,从系统整体架构角度概要讲解Android系统的核心技术点,带领大家初探Android系统全貌以及内部运作机制。虽然Android系统非常庞大且错综复杂,需要具备全面的技术栈,但整体架构设计清晰。Android底层内核空间以Linux Kernel作为基石,上层用户空间由Native系统库、虚拟机运行环境、框架层组成,通过系统调用(Syscall)连通系统的内核空间与用户空间。对于用户空间主要采用C++和Java代码编写,通过JNI技术打通用户空间的Java层和Native层(C++/C),从而连通整个系统。
为了能让大家整体上大致了解Android系统涉及的知识层面,先来看一张Google官方提供的经典分层架构图,从下往上依次分为Linux内核、HAL、系统Native库和Android运行时环境、Java框架层以及应用层这5层架构,其中每一层都包含大量的子模块或子系统。
上图采用静态分层方式的架构划分,众所周知,程序代码是死的,系统运转是活的,各模块代码运行在不同的进程(线程)中,相互之间进行着各种错综复杂的信息传递与交互流,从这个角度来说此图并没能体现Android整个系统的内部架构、运行机理,以及各个模块之间是如何衔接与配合工作的。为了更深入地掌握Android整个架构思想以及各个模块在Android系统所处的地位与价值,计划以Android系统启动过程为主线,以进程的视角来诠释Android M系统全貌,全方位的深度剖析各个模块功能,争取各个击破。这样才能犹如庖丁解牛,解决、分析问题则能游刃有余。
二、Android架构
Google提供的5层架构图很经典,但为了更进一步透视Android系统架构,本文更多的是以进程的视角,以分层的架构来诠释Android系统的全貌,阐述Android内部的环环相扣的内在联系。
系统启动架构图
图解: Android系统启动过程由上图从下往上的一个过程是由Boot Loader引导开机,然后依次进入 -> Kernel -> Native -> Framework -> App,接来下简要说说每个过程:
关于Loader层:
- Boot ROM: 当手机处于关机状态时,长按Power键开机,引导芯片开始从固化在ROM里的预设代码开始执行,然后加载引导程序到RAM;
- Boot Loader:这是启动Android系统之前的引导程序,主要是检查RAM,初始化硬件参数等功能。
2.1 Linux内核层
Android平台的基础是Linux内核,比如ART虚拟机最终调用底层Linux内核来执行功能。Linux内核的安全机制为Android提供相应的保障,也允许设备制造商为内核开发硬件驱动程序。
- 启动Kernel的swapper进程(pid=0):该进程又称为idle进程, 系统初始化过程Kernel由无到有开创的第一个进程, 用于初始化进程管理、内存管理,加载Display,Camera Driver,Binder Driver等相关工作;
- 启动kthreadd进程(pid=2):是Linux系统的内核进程,会创建内核工作线程kworkder,软中断线程ksoftirqd,thermal等内核守护进程。kthreadd进程是所有内核进程的鼻祖。
2.2 硬件抽象层 (HAL)
硬件抽象层 (HAL) 提供标准接口,HAL包含多个库模块,其中每个模块都为特定类型的硬件组件实现一组接口,比如WIFI/蓝牙模块,当框架API请求访问设备硬件时,Android系统将为该硬件加载相应的库模块。
2.3 Android Runtime & 系统库
每个应用都在其自己的进程中运行,都有自己的虚拟机实例。ART通过执行DEX文件可在设备运行多个虚拟机,DEX文件是一种专为Android设计的字节码格式文件,经过优化,使用内存很少。ART主要功能包括:预先(AOT)和即时(JIT)编译,优化的垃圾回收(GC),以及调试相关的支持。
这里的Native系统库主要包括init孵化来的用户空间的守护进程、HAL层以及开机动画等。启动init进程(pid=1),是Linux系统的用户进程,init进程是所有用户进程的鼻祖。
- init进程会孵化出ueventd、logd、healthd、installd、adbd、lmkd等用户守护进程;
- init进程还启动servicemanager(binder服务管家)、bootanim(开机动画)等重要服务
- init进程孵化出Zygote进程,Zygote进程是Android系统的第一个Java进程(即虚拟机进程),Zygote是所有Java进程的父进程,Zygote进程本身是由init进程孵化而来的。
2.4 Framework层
- Zygote进程,是由init进程通过解析init.rc文件后fork生成的,Zygote进程主要包含:
- 加载ZygoteInit类,注册Zygote Socket服务端套接字
- 加载虚拟机
- 提前加载类preloadClasses
- 提前加载资源preloadResouces
- System Server进程,是由Zygote进程fork而来,System Server是Zygote孵化的第一个进程,System Server负责启动和管理整个Java framework,包含ActivityManager,WindowManager,PackageManager,PowerManager等服务。
- Media Server进程,是由init进程fork而来,负责启动和管理整个C++ framework,包含AudioFlinger,Camera Service等服务。
2.5 App层
- Zygote进程孵化出的第一个App进程是Launcher,这是用户看到的桌面App;
- Zygote进程还会创建Browser,Phone,Email等App进程,每个App至少运行在一个进程上。
- 所有的App进程都是由Zygote进程fork生成的。
2.6 Syscall && JNI
- Native与Kernel之间有一层系统调用(SysCall)层,见Linux系统调用(Syscall)原理;
- Java层与Native(C/C++)层之间的纽带JNI,见Android JNI原理分析。
三、通信方式
无论是Android系统,还是各种Linux衍生系统,各个组件、模块往往运行在各种不同的进程和线程内,这里就必然涉及进程/线程之间的通信。对于IPC(Inter-Process Communication, 进程间通信),Linux现有管道、消息队列、共享内存、套接字、信号量、信号这些IPC机制,Android额外还有Binder IPC机制,Android OS中的Zygote进程的IPC采用的是Socket机制,在上层system server、media server以及上层App之间更多的是采用Binder IPC方式来完成跨进程间的通信。对于Android上层架构中,很多时候是在同一个进程的线程之间需要相互通信,例如同一个进程的主线程与工作线程之间的通信,往往采用的Handler消息机制。
想深入理解Android内核层架构,必须先深入理解Linux现有的IPC机制;对于Android上层架构,则最常用的通信方式是Binder、Socket、Handler,当然也有少量其他的IPC方式,比如杀进程Process.killProcess()采用的是signal方式。下面说说Binder、Socket、Handler:
3.1 Binder
Binder作为Android系统提供的一种IPC机制,无论从系统开发还是应用开发,都是Android系统中最重要的组成,也是最难理解的一块知识点,想了解为什么Android要采用Binder作为IPC机制?可查看我在知乎上的回答。深入了解Binder机制,最好的方法便是阅读源码,借用Linux鼻祖Linus Torvalds曾说过的一句话:Read The Fucking Source Code。下面简要说说Binder IPC原理。
Binder IPC原理
Binder通信采用c/s架构,从组件视角来说,包含Client、Server、ServiceManager以及binder驱动,其中ServiceManager用于管理系统中的各种服务。
- 想进一步了解Binder,可查看Binder系列—开篇,Binder系列花费了13篇文章的篇幅,从源码角度出发来讲述Driver、Native、Framework、App四个层面的整个完整流程。根据有些读者反馈这个系列还是不好理解,这个binder涉及的层次跨度比较大,知识量比较广,建议大家先知道binder是用于进程间通信,有个大致概念就可以先去学习系统基本知识,等后面有一定功力再进一步深入研究Binder机制。
Binder原理篇
- Binder系列-开篇 - Gityuan博客 | 袁辉辉的Android博客
- Binder系列3-启动ServiceManager - Gityuan博客 | 袁辉辉的Android博客
- Binder系列4-获取ServiceManager - Gityuan博客 | 袁辉辉的Android博客
- Binder系列5-注册服务(addService) - Gityuan博客 | 袁辉辉的Android博客
- Binder系列6-获取服务(getService) - Gityuan博客 | 袁辉辉的Android博客
- Binder系列7-framework层分析 - Gityuan博客 | 袁辉辉的Android博客
- 进程的Binder线程池工作过程 - Gityuan博客 | 袁辉辉的Android博客
- 彻底理解Android Binder通信架构
- Binder系列10-总结 - Gityuan博客 | 袁辉辉的Android博客
- Binder IPC的权限控制 - Gityuan博客 | 袁辉辉的Android博客
- Binder死亡通知机制之linkToDeath - Gityuan博客 | 袁辉辉的Android博客
Binder驱动篇:
- Binder系列1-Binder Driver初探
- Binder系列2-Binder Driver再探
Binder使用篇:
- Binder系列8-如何使用Binder - Gityuan博客 | 袁辉辉的Android博客
- Binder系列9-如何使用AIDL - Gityuan博客 | 袁辉辉的Android博客
3.2 Socket
Socket通信方式也是C/S架构,比Binder简单很多。在Android系统中采用Socket通信方式的主要有:
- zygote:用于孵化进程,system_server创建进程是通过socket向zygote进程发起请求;
- installd:用于安装App的守护进程,上层PackageManagerService很多实现最终都是交给它来完成;
- lmkd:lowmemorykiller的守护进程,Java层的LowMemoryKiller最终都是由lmkd来完成;
- adbd:这个也不用说,用于服务adb;
- logcatd:这个不用说,用于服务logcat;
- vold:即volume Daemon,是存储类的守护进程,用于负责如USB、Sdcard等存储设备的事件处理。
等等还有很多,这里不一一列举,Socket方式更多的用于Android framework层与native层之间的通信。Socket通信方式相对于binder比较简单,这里省略。
3.3 Handler
Binder/Socket用于进程间通信,而Handler消息机制用于同进程的线程间通信,Handler消息机制是由一组MessageQueue、Message、Looper、Handler共同组成的,为了方便且称之为Handler消息机制。
有人可能会疑惑,为何Binder/Socket用于进程间通信,能否用于线程间通信呢?答案是肯定,对于两个具有独立地址空间的进程通信都可以,当然也能用于共享内存空间的两个线程间通信,这就好比杀鸡用牛刀。接着可能还有人会疑惑,那handler消息机制能否用于进程间通信?答案是不能,Handler只能用于共享内存地址空间的两个线程间通信,即同进程的两个线程间通信。很多时候,Handler是工作线程向UI主线程发送消息,即App应用中只有主线程能更新UI,其他工作线程往往是完成相应工作后,通过Handler告知主线程需要做出相应地UI更新操作,Handler分发相应的消息给UI主线程去完成,如下图:
由于工作线程与主线程共享地址空间,即Handler实例对象mHandler位于线程间共享的内存堆上,工作线程与主线程都能直接使用该对象,只需要注意多线程的同步问题。工作线程通过mHandler向其成员变量MessageQueue中添加新Message,主线程一直处于loop()方法内,当收到新的Message时按照一定规则分发给相应的handleMessage()方法来处理。所以说,Handler消息机制用于同进程的线程间通信,其核心是线程间共享内存空间,而不同进程拥有不同的地址空间,也就不能用handler来实现进程间通信。
上图只是Handler消息机制的一种处理流程,是不是只能工作线程向UI主线程发消息呢,其实不然,可以是UI线程向工作线程发送消息,也可以是多个工作线程之间通过handler发送消息。更多关于Handler消息机制文章:
- Android消息机制-Handler(framework篇)
- Android消息机制-Handler(native篇)
- Android消息机制3-Handler(实战)
要理解framework层源码,掌握这3种基本的进程/线程间通信方式是非常有必要,当然Linux还有不少其他的IPC机制,比如共享内存、信号、信号量,在源码中也有体现,如果想全面彻底地掌握Android系统,还是需要对每一种IPC机制都有所了解。
四、核心提纲
博主对于Android从系统底层一路到上层都有自己的理解和沉淀,通过前面对系统启动的介绍,相信大家对Android系统有了一个整体观。接下来需抓核心、理思路,争取各个击破。后续将持续更新和完善整个大纲,不限于进程、内存、IO、系统服务架构以及分析实战等文章。
当然本站有一些文章没来得及进一步加工,有时间根据大家的反馈,不断修正和完善所有文章,争取给文章,再进一步精简非核心代码,增加可视化图表以及文字的结论性分析。基于Android 6.0的源码,专注于分享Android系统原理、架构分析的原创文章。
建议阅读群体: 适合于正从事或者有兴趣研究Android系统的工程师或者技术爱好者,也适合Android App高级工程师;对于尚未入门或者刚入门的App工程师阅读可能会有点困难,建议先阅读更基础的资料,再来阅读本站博客。
看到Android整个系统架构是如此庞大的, 该问如何学习Android系统, 以下是我自己的Android的学习和研究论,仅供参考如何自学Android。
从整理上来列举一下Android系统的核心知识点概览:
4.1 系统启动系列
Android系统启动-概述: Android系统中极其重要进程:init, zygote, system_server, servicemanager 进程:
- Android系统启动-Init篇 - Gityuan博客 | 袁辉辉的Android博客
- Android系统启动-zygote篇 - Gityuan博客 | 袁辉辉的Android博客
- Android系统启动-SystemServer上篇 - Gityuan博客 | 袁辉辉的Android博客
- Android系统启动-SystemServer下篇 - Gityuan博客 | 袁辉辉的Android博客
- Binder系列3-启动ServiceManager - Gityuan博客 | 袁辉辉的Android博客
- 理解Android进程创建流程 - Gityuan博客 | 袁辉辉的Android博客
再来看看守护进程(也就是进程名一般以d为后缀,比如logd,此处d是指daemon的简称), 下面介绍部分守护进程:
- debuggerd
- installd
- lmkd
- logd
4.2 系统稳定性系列
Android系统稳定性主要是异常崩溃(crash)和执行超时(timeout),:
- 理解Android ANR的触发原理
- Input系统-ANR原理分析 - Gityuan博客 | 袁辉辉的Android博客
- 理解Android ANR的信息收集过程
- 解读Java进程的Trace文件 - Gityuan博客 | 袁辉辉的Android博客
- Native进程之Trace原理 - Gityuan博客 | 袁辉辉的Android博客
- WatchDog工作原理 - Gityuan博客 | 袁辉辉的Android博客
- 理解Android Crash处理流程
- 理解Native Crash处理流程
- global reference限制策略
4.3 Android进程系列
进程/线程是操作系统的魂,各种服务、组件、子系统都是依附于具体的进程实体。深入理解进程机制对于掌握Android系统整体架构和运转机制是非常有必要的,是系统工程师的基本功,下面列举进程相关的文章:
- 理解Android进程创建流程 - Gityuan博客 | 袁辉辉的Android博客
- 理解杀进程的实现原理 - Gityuan博客 | 袁辉辉的Android博客
- Android四大组件与进程启动的关系 - Gityuan博客 | 袁辉辉的Android博客
- Android进程绝杀技--forceStop - Gityuan博客 | 袁辉辉的Android博客
- 理解Android线程创建流程 - Gityuan博客 | 袁辉辉的Android博客
- 彻底理解Android Binder通信架构
- 进程的Binder线程池工作过程 - Gityuan博客 | 袁辉辉的Android博客
- Android进程生命周期与ADJ - Gityuan博客 | 袁辉辉的Android博客
- Android LowMemoryKiller原理分析
- 进程优先级 - Gityuan博客 | 袁辉辉的Android博客
- Android进程调度之adj算法 - Gityuan博客 | 袁辉辉的Android博客
- 进程篇-进程整理 - Gityuan博客 | 袁辉辉的Android博客
4.4 四大组件系列
对于App来说,Android应用的四大组件Activity,Service,Broadcast Receiver, Content Provider最为核心,接下分别展开介绍:
- startActivity启动过程分析 - Gityuan博客 | 袁辉辉的Android博客
- 简述Activity生命周期 - Gityuan博客 | 袁辉辉的Android博客
- startService启动过程分析 - Gityuan博客 | 袁辉辉的Android博客
- bindService启动过程分析 - Gityuan博客 | 袁辉辉的Android博客
- 彻底理解Android Binder通信架构
- Android Broadcast广播机制分析
- 理解ContentProvider原理 - Gityuan博客 | 袁辉辉的Android博客
- ContentProvider引用计数 - Gityuan博客 | 袁辉辉的Android博客
- Activity与Service生命周期 - Gityuan博客 | 袁辉辉的Android博客
- 简述Activity与Window关系 - Gityuan博客 | 袁辉辉的Android博客
- 四大组件之综述 - Gityuan博客 | 袁辉辉的Android博客
- 四大组件之ServiceRecord - Gityuan博客 | 袁辉辉的Android博客
- 四大组件之BroadcastRecord - Gityuan博客 | 袁辉辉的Android博客
- 四大组件之ContentProviderRecord - Gityuan博客 | 袁辉辉的Android博客
- 理解Android Context
- 理解Application创建过程 - Gityuan博客 | 袁辉辉的Android博客
- unbindService流程分析 - Gityuan博客 | 袁辉辉的Android博客
- 四大组件之ActivityRecord - Gityuan博客 | 袁辉辉的Android博客
- AMS总结(一) - Gityuan博客 | 袁辉辉的Android博客
4.5 图形系统系列
图形也是整个系统非常复杂且重要的一个系列,涉及WindowManager,SurfaceFlinger服务。
- WMS-启动过程 - Gityuan博客 | 袁辉辉的Android博客
- WMS-启动窗口(StartingWindow) - Gityuan博客 | 袁辉辉的Android博客
- 以Window视角来看startActivity - Gityuan博客 | 袁辉辉的Android博客
- Android图形系统概述 - Gityuan博客 | 袁辉辉的Android博客
- SurfaceFlinger启动篇 - Gityuan博客 | 袁辉辉的Android博客
- SurfaceFlinger绘图篇 - Gityuan博客 | 袁辉辉的Android博客
- Choreographer原理 - Gityuan博客 | 袁辉辉的Android博客
4.6 系统服务篇
再则就是在整个架构中有大量的服务,都是基于Binder来交互的,Android系统服务的注册过程也是在此之上的构建的。计划针对部分核心服务来重点分析:
- ActivityManagerService启动过程 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-启动篇 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-InputReader线程 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-InputDispatcher线程 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-UI线程 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-事件处理全过程 - Gityuan博客 | 袁辉辉的Android博客
- Input系统-ANR原理分析 - Gityuan博客 | 袁辉辉的Android博客
- http://gityuan.com/2016/11/06/packagemanagerservice)
- Installd守护进程 - Gityuan博客 | 袁辉辉的Android博客)
- 理解AlarmManager机制 - Gityuan博客 | 袁辉辉的Android博客
- 理解JobScheduler机制 - Gityuan博客 | 袁辉辉的Android博客
- Android耗电统计算法 - Gityuan博客 | 袁辉辉的Android博客
- DropBoxManager启动篇 - Gityuan博客 | 袁辉辉的Android博客
- 多用户管理UserManager - Gityuan博客 | 袁辉辉的Android博客
- 更多系统服务,敬请期待
4.7 内存&&存储篇
- 内存篇
- Android LowMemoryKiller原理分析
- Linux内存管理
- Android内存分析命令
- 存储篇
- Android存储系统之源码篇
- Android存储系统之架构篇
- Linux驱动篇
- 敬请期待
- dalvik/art
- 解读Java进程的Trace文件
4.8 工具篇
再来说说Android相关的一些常用命令和工具以及调试手段.
- 理解Android编译命令 - Gityuan博客 | 袁辉辉的Android博客
- 理解Android.bp - Gityuan博客 | 袁辉辉的Android博客
- 性能工具Systrace - Gityuan博客 | 袁辉辉的Android博客
- Android内存分析命令 - Gityuan博客 | 袁辉辉的Android博客
- ps进程命令 - Gityuan博客 | 袁辉辉的Android博客
- Am命令用法 - Gityuan博客 | 袁辉辉的Android博客
- Pm命令用法 - Gityuan博客 | 袁辉辉的Android博客
- 调试系列1:bugreport源码篇 - Gityuan博客 | 袁辉辉的Android博客
- 调试系列2:bugreport实战篇 - Gityuan博客 | 袁辉辉的Android博客
- dumpsys命令用法 - Gityuan博客 | 袁辉辉的Android博客
- Android logd日志原理
- 介绍gdb调试工具 - Gityuan博客 | 袁辉辉的Android博客
- 介绍addr2line调试命令 - Gityuan博客 | 袁辉辉的Android博客
4.9 实战篇
下面列举处理过的部分较为典型的案例,供大家参考
- Binder Driver缺陷导致定屏的案例
- 深度解读ArrayMap优势与缺陷 - Gityuan博客 | 袁辉辉的Android博客
- 数组越界导致系统重启的案例 - Gityuan博客 | 袁辉辉的Android博客
- 一行Log引发多线程并发问题的案例 - Gityuan博客 | 袁辉辉的Android博客
- 跑monkey压力测试过程的冻屏案例 - Gityuan博客 | 袁辉辉的Android博客
- 深度剖析APP保活案例 - Gityuan博客 | 袁辉辉的Android博客
五、结束语
Android系统之博大精深,包括Linux内核、Native、虚拟机、Framework,通过系统调用连通内核与用户空间,通过JNI打通用户空间的Java层和Native层,通过Binder、Socket、Handler等打通跨进程、跨线程的信息交换。只有真正阅读并理解系统核心架构的设计,解决问题和设计方案才能做到心中无剑胜有剑,才能做到知其然知其所以然。当修炼到此,恭喜你对系统有了更高一个层次的理解,正如太极剑法,忘记了所有招式,也就练成了太极剑法。
再回过头去看看那些API,看到的将不再是一行行代码、一个个接口的调用,而是各种信息的传递与交互工作,而是背后成千上万个小蝌蚪的动态执行流。记得《侠客行》里面的龙木二岛主终其一生也无法参透太玄经,石破天却短短数日练成绝世神功,究其根源是龙木二岛主以静态视角去解读太玄经,而石破天把墙壁的图案想象成无数游动的蝌蚪,最终成就绝世神功。一言以蔽之,程序代码是死的,系统运转是活的,要以动态视角去理解系统架构。
相关推荐
- 谷歌正在为Play商店进行Material Design改造
-
谷歌最近一直忙于在其应用程序中完成MaterialDesign风格的改造,而Play商店似乎是接下来的一个。9to5Google网站报道,有用户在Play商店的最新版本中发现了新界面,暗示该应用和网...
- 企业网站免费搭建,定制化建站CMS系统
-
科腾软件企业网站CMS管理系统已完成开发工作,首次开源(全部源码)发布。开发工具:VisualStudioEnterprise2022数据库:SQLite(零配置,跨平台,嵌入式)开发...
- 您需要的 11 个免费 Chrome 扩展程序
-
来源:SEO_SEM营销顾问大师Chrome扩展程序是SEO的无名英雄,他们在幕后默默工作,使您的策略脱颖而出并提高您的努力效率。从竞争对手研究到审核您的网站,速度比您说“元描述”还快,这些小工具发...
- 户外便携设备抗干扰困境如何破局?CMS-160925-078S-67给出答案
-
在户外复杂的电磁环境中,便携式设备中的扬声器需具备出色抗干扰能力,CUID的CMS-160925-078S-67在这方面表现突出。 从其结构设计来看,矩形框架虽主要为适配紧凑空...
- 一个基于NetCore开发的前后端分离CMS系统
-
今天给大家推荐一个开源的前后端分离架构的CMS建站系统。项目简介这是一个基于.Net3构建的简单、跨平台、模块化建站系统。系统业务简单、代码清晰、层级分明、全新架构便于二次扩展开发。支持多种数据库,...
- 本地Docker部署ZFile网盘打造个人云存储
-
前言本文主要介绍如何在LinuxUbuntu系统使用Docker本地部署ZFile文件管理系统,并结合cpolar内网穿透工具实现远程访问本地服务器上的ZFile传输与备份文件,轻松搭建个人网盘,无...
- pcfcms企业建站系统 免费+开源的企业内容管理系统
-
项目介绍pcfcms是基于TP6.0框架为核心开发的免费+开源的企业内容管理系统,专注企业建站用户需求提供海量各行业模板,降低中小企业网站建设、网络营销成本,致力于打造用户舒适的建站体验。演示站...
- 【推荐】一个高颜值且功能强大的 Vue3 后台管理系统框架
-
如果您对源码&技术感兴趣,请点赞+收藏+转发+关注,大家的支持是我分享最大的动力!!!项目介绍SnowAdmin是一款基于Vue3、TypeScript、Vite5、Pinia、Arco-Desi...
- java开源cms管理系统框架PublicCMS后台管理系统
-
一款使用Java语言开发的CMS,提供文章发布,图片展示,文件下载,用户权限、站点模块,内容管理、分类等功能。可免费用于商业用途maven工程数据库脚本在工程中database文件夹下代码结构:效果...
- 一定要大量读书:当我问Deepseek,它给出的高效阅读方法厉害了!
-
一年一度的世界读书日,总该写点什么。于是,我去问Deepseek给我推荐人生破局必读的10本书,结果它给了我回复,竟然10本推荐的书籍里,我都曾经浏览过,同时还给出破局关键。而说浏览过,不是读过,是因...
- 《搜神札记》:不应磨灭的惊奇(小说《搜神记》)
-
□黄勃志怪传说的书写一直是文人墨客的后花园,晚近尤盛,从张岱到袁枚到纪昀,收集那些或阴森或吊诡的行状故事,遂成一类,到民国年间,周作人挟此遗传,捋袖子拿希腊神话动刀,乃兄鲁迅不甘其后,《故事新编》虎...
- 《如何构建金字塔》之第三章总结(构建金字塔结构的方法有)
-
“没有什么比一套好理论更有用了。”——库尔特.勒温这篇读后感依然引用了这句库尔特.勒温名言,这句话也是我读芭芭拉.明托这本书的初衷。今天就“如何构建金字塔”,我来谈谈我的读后心得。我热爱写作,但是写...
- 《助人技术》第一章助人引论内容框架
-
第一章内容基本呈现如何成为助人者(心理咨询师)以及一些相关基础知识,对于进入这个行业有兴趣以及希望通过心理咨询寻求帮助但存有疑虑的当事人,都值得一读。心理咨询的三个阶段(不是说严格的三个阶段,而是广义...
- AI助手重构读后感写作流程:从提纲到完整性思考的转换
-
大家好!你有没有遇到过读完一本书,想要写读后感,却不知道从何下手的情况呢?今天我们要来探讨一下如何利用稿见AI助手来重构读后感写作流程,从提纲到完整性思考的转换。让我们一起来看看这个全新而又实用的方法...
- 图解用思维导图做读书笔记技巧(图解用思维导图做读书笔记技巧视频)
-
做阅读笔记非常有利于读后进行有效的深入思考,而思维导图这一强大的工具其最大的特点就是架构清晰,在阅读过程中对文章的分析、总结、分类起着很大的辅助作用。思维导图读书笔记步骤:1、阅读大纲。首先要快速浏览...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- MVC框架 (46)
- spring框架 (46)
- 框架图 (58)
- flask框架 (53)
- quartz框架 (51)
- abp框架 (47)
- jpa框架 (47)
- laravel框架 (46)
- springmvc框架 (49)
- 分布式事务框架 (65)
- scrapy框架 (56)
- shiro框架 (61)
- 定时任务框架 (56)
- java日志框架 (61)
- JAVA集合框架 (47)
- mfc框架 (52)
- abb框架断路器 (48)
- grpc框架 (55)
- ppt框架 (48)
- 内联框架 (52)
- cad怎么画框架 (58)
- ps怎么画框架 (47)
- ssm框架实现登录注册 (49)
- oracle字符串长度 (48)
- oracle提交事务 (47)