百篇博客系列篇.本篇为:

内存管理相关篇为:

初始化整个内存

从main()跟踪可看内存部分初始化是在OsSysMemInit()中完成的。

  1. UINT32 OsSysMemInit(VOID)
  2. {
  3.     STATUS_T ret;
  4.     OsKSpaceInit();//内核空间初始化
  5.     ret = OsKHeapInit(OS_KHEAP_BLOCK_SIZE);// 内核动态内存初始化 512K   
  6.     if (ret != LOS_OK) {
  7.         VM_ERR("OsKHeapInit fail");
  8.         return LOS_NOK;
  9.     }
  10.     OsVmPageStartup();// page初始化
  11.     OsInitMappingStartUp();// 映射初始化
  12.     ret = ShmInit();// 共享内存初始化
  13.     if (ret < 0) {
  14.         VM_ERR("ShmInit fail");  
  15.         return LOS_NOK;
  16.     }
  17.     return LOS_OK;
  18. }

鸿蒙虚拟内存整体布局图

  1. // HarmonyOS 内核空间包含以下各段:
  2. extern CHAR __int_stack_start; // 运行系统函数栈的开始地址
  3. extern CHAR __rodata_start; // ROM开始地址 只读
  4. extern CHAR __rodata_end; // ROM结束地址
  5. extern CHAR __bss_start; // bss开始地址
  6. extern CHAR __bss_end; // bss结束地址
  7. extern CHAR __text_start; // 代码区开始地址
  8. extern CHAR __text_end; // 代码区结束地址
  9. extern CHAR __ram_data_start; // RAM开始地址 可读可写
  10. extern CHAR __ram_data_end; // RAM结束地址
  11. extern UINT32 __heap_start; // 堆区开始地址
  12. extern UINT32 __heap_end; // 堆区结束地址

内存一开始一张白纸,这些extern就是给它画大界线的,从哪到哪是属于什么段。这些值大小取决实际项目内存条的大小,不同的内存条,地址肯定会不一样,所以必须由外部提供,鸿蒙内核采用了Linux的段管理方式。结合上图对比以下的解释自行理解下位置。

BSS段 (bss segment)通常是指用来存放程序中未初始化的全局变量的一块内存区域。BSS是英文Block Started by Symbol的简称。BSS段属于静态内存分配。该段用于存储未初始化的全局变量或者是默认初始化为0的全局变量,它不占用程序文件的大小,但是占用程序运行时的内存空间。

data段 该段用于存储初始化的全局变量,初始化为0的全局变量出于编译优化的策略还是被保存在BSS段。

细心的读者可能发现了,鸿蒙内核几乎所有的全局变量都没有赋初始化值或NULL,这些变量经过编译后是放在了BSS段的,运行时占用内存空间,如此编译出来的ELF包就变小了。

.rodata段,该段也叫常量区,用于存放常量数据,ro就是Read Only之意。

text段 是用于存放程序代码的,编译时确定,只读。更进一步讲是存放处理器的机器指令,当各个源文件单独编译之后生成目标文件,经连接器链接各个目标文件并解决各个源文件之间函数的引用,与此同时,还得将所有目标文件中的.text段合在一起。

stack栈段,是由系统负责申请释放,用于存储参数变量及局部变量以及函数的执行。

heap段 它由用户申请和释放,申请时至少分配虚存,当真正存储数据时才分配相应的实存,释放时也并非立即释放实存,而是可能被重复利用。

内核空间是怎么初始化的?

  1. LosMux g_vmSpaceListMux;//虚拟空间互斥锁,一般和g_vmSpaceList配套使用
  2. LOS_DL_LIST_HEAD(g_vmSpaceList);//g_vmSpaceList把所有虚拟空间挂在一起,
  3. LosVmSpace g_kVmSpace;    //内核空间地址
  4. LosVmSpace g_vMallocSpace;//虚拟分配空间地址
  5. //鸿蒙内核空间有两个(内核进程空间和内核动态分配空间),共用一张L1页表
  6. VOID OsKSpaceInit(VOID)
  7. {
  8.     OsVmMapInit();// 初始化互斥量
  9.     OsKernVmSpaceInit(&g_kVmSpace OsGFirstTableGet());// 初始化内核虚拟空间,OsGFirstTableGet 为L1表基地址
  10.     OsVMallocSpaceInit(&g_vMallocSpace OsGFirstTableGet());// 初始化动态分配区虚拟空间,OsGFirstTableGet 为L1表基地址
  11. }//g_kVmSpace g_vMallocSpace 共用一个L1页表
  12. //初始化内核堆空间
  13. STATUS_T OsKHeapInit(size_t size)
  14. {
  15.     STATUS_T ret;
  16.     VOID *ptr = NULL;
  17.     /*
  18.      * roundup to MB aligned in order to set kernel attributes. kernel text/code/data attributes
  19.      * should page mapping, remaining region should section mapping. so the boundary should be
  20.      * MB aligned.
  21.      */
  22.      //向上舍入到MB对齐是为了设置内核属性。内核文本/代码/数据属性应该是页映射,其余区域应该是段映射,所以边界应该对齐。
  23.     UINTPTR end = ROUNDUP(g_vmBootMemBase + size MB);//用M是因为采用section mapping 鸿蒙内核源码分析(内存映射篇)有阐述
  24.     size = end - g_vmBootMemBase;
  25.     //ROUNDUP(0x00000200+512,1024) = 1024  ROUNDUP(0x00000201+512,1024) = 2048 此处需细品! 
  26.     ptr = OsVmBootMemAlloc(size);//因刚开机,使用引导分配器分配
  27.     if (!ptr) {
  28.         PRINT_ERR("vmm_kheap_init boot_alloc_mem failed! %d\n" size);
  29.         return -1;
  30.     }
  31.     m_aucSysMem0 = m_aucSysMem1 = ptr;//内存池基地址,取名auc还用0和1来标识有何深意,一直没整明白, 哪位大神能告诉下?
  32.     ret = LOS_MemInit(m_aucSysMem0 size);//初始化内存池
  33.     if (ret != LOS_OK) {
  34.         PRINT_ERR("vmm_kheap_init LOS_MemInit failed!\n");
  35.         g_vmBootMemBase -= size;//分配失败时需归还size, g_vmBootMemBase是很野蛮粗暴的
  36.         return ret;
  37.     }
  38.     LOS_MemExpandEnable(OS_SYS_MEM_ADDR);//地址可扩展
  39.     return LOS_OK;
  40. }

内核空间用了三个全局变量,其中一个是互斥LosMux,IPC部分会详细讲,这里先不展开。 比较有意思的是LOS_DL_LIST_HEAD,看内核源码过程中经常会为这样的代码点头称赞,会心一笑。点赞!

  1. #define LOS_DL_LIST_HEAD(list) LOS_DL_LIST list = { &(list), &(list) }

Page是如何初始化的?

page是映射的最小单位,是物理地址<--->虚拟地址映射的数据结构的基础

  1. // page初始化
  2. VOID OsVmPageStartup(VOID)
  3. {
  4.     struct VmPhysSeg *seg = NULL;
  5.     LosVmPage *page = NULL;
  6.     paddr_t pa;
  7.     UINT32 nPage;
  8.     INT32 segID;
  9.     OsVmPhysAreaSizeAdjust(ROUNDUP((g_vmBootMemBase - KERNEL_ASPACE_BASE), PAGE_SIZE));//校正 g_physArea size
  10.     nPage = OsVmPhysPageNumGet();//得到 g_physArea 总页数
  11.     g_vmPageArraySize = nPage * sizeof(LosVmPage);//页表总大小
  12.     g_vmPageArray = (LosVmPage *)OsVmBootMemAlloc(g_vmPageArraySize);//申请页表存放区域
  13.     OsVmPhysAreaSizeAdjust(ROUNDUP(g_vmPageArraySize PAGE_SIZE));// g_physArea 变小
  14.     OsVmPhysSegAdd();// 段页绑定
  15.     OsVmPhysInit();// 加入空闲链表和设置置换算法,LRU(最近最久未使用)算法
  16.     for (segID = 0; segID < g_vmPhysSegNum; segID++) {
  17.         seg = &g_vmPhysSeg[segID];
  18.         nPage = seg->size >> PAGE_SHIFT;
  19.         for (page = seg->pageBase pa = seg->start; page <= seg->pageBase + nPage;
  20.              page++, pa += PAGE_SIZE) {
  21.             OsVmPageInit(page pa segID);//page初始化
  22.         }
  23.         OsVmPageOrderListInit(seg->pageBase nPage);// 页面分配的排序
  24.     }
  25. }

进程是如何申请内存的?

进程的主体是来自进程池,进程池是统一分配的,怎么创建进程池的去翻系列篇里的文章,所以创建一个进程的时候只需要分配虚拟内存LosVmSpace,这里要分内核模式和用户模式下的申请。

  1. //初始化进程的 用户空间 或 内核空间
  2. //初始化PCB块
  3. STATIC UINT32 OsInitPCB(LosProcessCB *processCB UINT32 mode UINT16 priority UINT16 policy const CHAR *name)
  4. {
  5.     UINT32 count;
  6.     LosVmSpace *space = NULL;
  7.     LosVmPage *vmPage = NULL;
  8.     status_t status;
  9.     BOOL retVal = FALSE;
  10.     processCB->processMode = mode;//用户态进程还是内核态进程
  11.     processCB->processStatus = OS_PROCESS_STATUS_INIT;//进程初始状态
  12.     processCB->parentProcessID = OS_INVALID_VALUE;//爸爸进程,外面指定
  13.     processCB->threadGroupID = OS_INVALID_VALUE;//所属线程组
  14.     processCB->priority = priority;//优先级
  15.     processCB->policy = policy;//调度算法 LOS_SCHED_RR
  16.     processCB->umask = OS_PROCESS_DEFAULT_UMASK;//掩码
  17.     processCB->timerID = (timer_t)(UINTPTR)MAX_INVALID_TIMER_VID;
  18.     LOS_ListInit(&processCB->threadSiblingList);//初始化任务/线程链表
  19.     LOS_ListInit(&processCB->childrenList);        //初始化孩子链表
  20.     LOS_ListInit(&processCB->exitChildList);    //初始化记录哪些孩子退出了的链表    
  21.     LOS_ListInit(&(processCB->waitList));        //初始化等待链表
  22.     for (count = 0; count < OS_PRIORITY_QUEUE_NUM; ++count) { //根据 priority数 创建对应个数的队列
  23.         LOS_ListInit(&processCB->threadPriQueueList[count]);  
  24.     }
  25.     if (OsProcessIsUserMode(processCB)) {// 是否为用户态进程
  26.         space = LOS_MemAlloc(m_aucSysMem0 sizeof(LosVmSpace));
  27.         if (space == NULL) {
  28.             PRINT_ERR("%s %d, alloc space failed\n" __FUNCTION__ __LINE__);
  29.             return LOS_ENOMEM;
  30.         }
  31.         VADDR_T *ttb = LOS_PhysPagesAllocContiguous(1);//分配一个物理页用于存储L1页表 4G虚拟内存分成 (4096*1M)
  32.         if (ttb == NULL) {//这里直接获取物理页ttb
  33.             PRINT_ERR("%s %d, alloc ttb or space failed\n" __FUNCTION__ __LINE__);
  34.             (VOID)LOS_MemFree(m_aucSysMem0 space);
  35.             return LOS_ENOMEM;
  36.         }
  37.         (VOID)memset_s(ttb PAGE_SIZE 0 PAGE_SIZE);
  38.         retVal = OsUserVmSpaceInit(space ttb);//初始化虚拟空间和本进程 mmu
  39.         vmPage = OsVmVaddrToPage(ttb);//通过虚拟地址拿到page
  40.         if ((retVal == FALSE) || (vmPage == NULL)) {//异常处理
  41.             PRINT_ERR("create space failed! ret: %d, vmPage: %#x\n" retVal vmPage);
  42.             processCB->processStatus = OS_PROCESS_FLAG_UNUSED;//进程未使用,干净
  43.             (VOID)LOS_MemFree(m_aucSysMem0 space);//释放虚拟空间
  44.             LOS_PhysPagesFreeContiguous(ttb 1);//释放物理页,4K
  45.             return LOS_EAGAIN;
  46.         }
  47.         processCB->vmSpace = space;//设为进程虚拟空间
  48.         LOS_ListAdd(&processCB->vmSpace->archMmu.ptList &(vmPage->node));//将空间映射页表挂在 空间的mmu L1页表, L1为表头
  49.     } else {
  50.         processCB->vmSpace = LOS_GetKVmSpace();//内核共用一个虚拟空间,内核进程 常驻内存
  51.     }
  52. #ifdef LOSCFG_SECURITY_VID
  53.     status = VidMapListInit(processCB);
  54.     if (status != LOS_OK) {
  55.         PRINT_ERR("VidMapListInit failed!\n");
  56.         return LOS_ENOMEM;
  57.     }
  58. #endif
  59. #ifdef LOSCFG_SECURITY_CAPABILITY
  60.     OsInitCapability(processCB);
  61. #endif
  62.     if (OsSetProcessName(processCB name) != LOS_OK) {
  63.         return LOS_ENOMEM;
  64.     }
  65.     return LOS_OK;
  66. }
  67. LosVmSpace *LOS_GetKVmSpace(VOID)
  68. {
  69. return &g_kVmSpace;
  70. }

从代码可以看出,内核空间固定只有一个g_kVmSpace,而每个用户进程的虚拟内存空间都是独立的。请细品!

task是如何申请内存的?

task的主体是来自进程池,task池是统一分配的,怎么创建task池的去翻系列篇里的文章。这里task只需要申请stack空间,还是直接上看源码吧,用OsUserInitProcess函数看应用程序的main() 是如何被内核创建任务和运行的。

  1. //所有的用户进程都是使用同一个用户代码段描述符和用户数据段描述符,它们是__USER_CS和__USER_DS,也就是每个进程处于用户态时,它们的CS寄存器和DS寄存器中的值是相同的。当任何进程或者中断异常进入内核后,都是使用相同的内核代码段描述符和内核数据段描述符,它们是__KERNEL_CS和__KERNEL_DS。这里要明确记得,内核数据段实际上就是内核态堆栈段。
  2. LITE_OS_SEC_TEXT_INIT UINT32 OsUserInitProcess(VOID)
  3. {
  4.     INT32 ret;
  5.     UINT32 size;
  6.     TSK_INIT_PARAM_S param = { 0 };
  7.     VOID *stack = NULL;
  8.     VOID *userText = NULL;
  9.     CHAR *userInitTextStart = (CHAR *)&__user_init_entry;//代码区开始位置 ,所有进程
  10.     CHAR *userInitBssStart = (CHAR *)&__user_init_bss;// 未初始化数据区(BSS)。在运行时改变其值
  11.     CHAR *userInitEnd = (CHAR *)&__user_init_end;// 结束地址
  12.     UINT32 initBssSize = userInitEnd - userInitBssStart;
  13.     UINT32 initSize = userInitEnd - userInitTextStart;
  14.     LosProcessCB *processCB = OS_PCB_FROM_PID(g_userInitProcess);
  15.     ret = OsProcessCreateInit(processCB OS_USER_MODE "Init" OS_PROCESS_USERINIT_PRIORITY);// 初始化用户进程,它将是所有应用程序的父进程
  16.     if (ret != LOS_OK) {
  17.         return ret;
  18.     }
  19.     userText = LOS_PhysPagesAllocContiguous(initSize >> PAGE_SHIFT);// 分配连续的物理页
  20.     if (userText == NULL) {
  21.         ret = LOS_NOK;
  22.         goto ERROR;
  23.     }
  24.     (VOID)memcpy_s(userText initSize (VOID *)&__user_init_load_addr initSize);// 安全copy 经加载器load的结果 __user_init_load_addr -> userText
  25.     ret = LOS_VaddrToPaddrMmap(processCB->vmSpace (VADDR_T)(UINTPTR)userInitTextStart LOS_PaddrQuery(userText),
  26.                                initSize VM_MAP_REGION_FLAG_PERM_READ | VM_MAP_REGION_FLAG_PERM_WRITE |
  27.                                VM_MAP_REGION_FLAG_PERM_EXECUTE | VM_MAP_REGION_FLAG_PERM_USER);// 虚拟地址与物理地址的映射
  28.     if (ret < 0) {
  29.         goto ERROR;
  30.     }
  31.     (VOID)memset_s((VOID *)((UINTPTR)userText + userInitBssStart - userInitTextStart), initBssSize 0 initBssSize);// 除了代码段,其余都清0
  32.     stack = OsUserInitStackAlloc(g_userInitProcess &size);// 初始化堆栈区
  33.     if (stack == NULL) {
  34.         PRINTK("user init process malloc user stack failed!\n");
  35.         ret = LOS_NOK;
  36.         goto ERROR;
  37.     }
  38.     param.pfnTaskEntry = (TSK_ENTRY_FUNC)userInitTextStart;// 从代码区开始执行,也就是应用程序main 函数的位置
  39.     param.userParam.userSP = (UINTPTR)stack + size;// 指向栈底
  40.     param.userParam.userMapBase = (UINTPTR)stack;// 栈顶
  41.     param.userParam.userMapSize = size;// 栈大小
  42.     param.uwResved = OS_TASK_FLAG_PTHREAD_JOIN;// 可结合的(joinable)能够被其他线程收回其资源和杀死
  43.     ret = OsUserInitProcessStart(g_userInitProcess &param);// 创建一个任务,来运行main函数
  44.     if (ret != LOS_OK) {
  45.         (VOID)OsUnMMap(processCB->vmSpace param.userParam.userMapBase param.userParam.userMapSize);
  46.         goto ERROR;
  47.     }
  48.     return LOS_OK;
  49. ERROR:
  50.     (VOID)LOS_PhysPagesFreeContiguous(userText initSize >> PAGE_SHIFT);//释放物理内存块
  51.     OsDeInitPCB(processCB);//删除PCB块
  52.     return ret;
  53. }

所有的用户进程都是通过init进程 fork来的, 可以看到创建进程的同时创建了一个task, 入口函数就是代码区的第一条指令,也就是应用程序 main函数。这里再说下stack的大小,不同空间下的task栈空间是不一样的,鸿蒙内核中有三种栈空间size,如下

  1. #define LOSCFG_BASE_CORE_TSK_IDLE_STACK_SIZE SIZE(0x800)//内核进程,运行在内核空间2K
  2. #define OS_USER_TASK_SYSCALL_SATCK_SIZE 0x3000 //用户进程,通过系统调用创建的task运行在内核空间的 12K
  3. #define OS_USER_TASK_STACK_SIZE 0x100000//用户进程运行在用户空间的1M

鸿蒙内核源码分析.总目录

v08.xx 鸿蒙内核源码分析(总目录) | 百万汉字注解 百篇博客分析 | 51.c.h .o

百万汉字注解.百篇博客分析

百万汉字注解 >> 精读鸿蒙源码,中文注解分析, 深挖地基工程,大脑永久记忆,四大码仓每日同步更新< gitee| github| csdn| coding >

百篇博客分析 >> 故事说内核,问答式导读,生活式比喻,表格化说明,图形化展示,主流站点定期更新中< 51cto| csdn| harmony| osc >

关注不迷路.代码即人生

QQ群:790015635 | 入群密码: 666

原创不易,欢迎转载,但请注明出处.

鸿蒙内核源码分析(内存管理篇) | 虚拟内存全景图是怎样的 | 百篇博客分析OpenHarmony源码 | v12.04的更多相关文章

  1. 鸿蒙内核源码分析(任务管理篇) | 任务池是如何管理的 | 百篇博客分析OpenHarmony源码 | v5.05

    百篇博客系列篇.本篇为: v05.xx 鸿蒙内核源码分析(任务管理篇) | 任务池是如何管理的 | 51.c.h .o 任务管理相关篇为: v03.xx 鸿蒙内核源码分析(时钟任务篇) | 触发调度谁 ...

  2. 鸿蒙内核源码分析(中断管理篇) | 江湖从此不再怕中断 | 百篇博客分析OpenHarmony源码 | v44.02

    百篇博客系列篇.本篇为: v44.xx 鸿蒙内核源码分析(中断管理篇) | 江湖从此不再怕中断 | 51.c.h .o 硬件架构相关篇为: v22.xx 鸿蒙内核源码分析(汇编基础篇) | CPU在哪 ...

  3. v76.01 鸿蒙内核源码分析(共享内存) | 进程间最快通讯方式 | 百篇博客分析OpenHarmony源码

    百篇博客分析|本篇为:(共享内存篇) | 进程间最快通讯方式 进程通讯相关篇为: v26.08 鸿蒙内核源码分析(自旋锁) | 当立贞节牌坊的好同志 v27.05 鸿蒙内核源码分析(互斥锁) | 同样 ...

  4. 鸿蒙内核源码分析(内存映射篇) | 虚拟内存虚在哪里 | 百篇博客分析OpenHarmony源码 | v15.03

    百篇博客系列篇.本篇为: v15.xx 鸿蒙内核源码分析(内存映射篇) | 虚拟内存虚在哪里 | 51.c.h .o 内存管理相关篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有哪些分 ...

  5. 鸿蒙内核源码分析(内存汇编篇) | 谁是虚拟内存实现的基础 | 百篇博客分析OpenHarmony源码 | v14.14

    百篇博客系列篇.本篇为: v14.xx 鸿蒙内核源码分析(内存汇编篇) | 谁是虚拟内存实现的基础 | 51.c.h .o 内存管理相关篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有 ...

  6. 鸿蒙内核源码分析(内存规则篇) | 内存管理到底在管什么 | 百篇博客分析OpenHarmony源码 | v16.02

    百篇博客系列篇.本篇为: v16.xx 鸿蒙内核源码分析(内存规则篇) | 内存管理到底在管什么 | 51.c.h .o 内存管理相关篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有哪 ...

  7. 鸿蒙内核源码分析(内存分配篇) | 内存有哪些分配方式  | 百篇博客分析OpenHarmony源码 | v11.02

    百篇博客系列篇.本篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有哪些分配方式 | 51.c.h .o 内存管理相关篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有哪些 ...

  8. 鸿蒙内核源码分析(进程管理篇) | 谁在管理内核资源 | 百篇博客分析OpenHarmonyOS | v2.07

    百篇博客系列篇.本篇为: v02.xx 鸿蒙内核源码分析(进程管理篇) | 谁在管理内核资源 | 51.c.h .o 进程管理相关篇为: v02.xx 鸿蒙内核源码分析(进程管理篇) | 谁在管理内核 ...

  9. 鸿蒙内核源码分析(内存主奴篇) | 皇上和奴才如何相处 | 百篇博客分析OpenHarmony源码 | v10.04

    百篇博客系列篇.本篇为: v10.xx 鸿蒙内核源码分析(内存主奴篇) | 皇上和奴才如何相处 | 51.c.h .o 前因后果相关篇为: v08.xx 鸿蒙内核源码分析(总目录) | 百万汉字注解 ...

随机推荐

  1. 【ArcGIS】 设置管段的流向

    在排水管网或者燃气管网中对管段进行几何网络分析,常常用到设置管段流向,一般有三种方法: 1,有流向字段的,直接进行唯一值渲染, 2,没有流向字段的需要建立几何网络, 2.1 在几何网络存在的情况下,设 ...

  2. mysql基础操作(三):数据约束

    首先创建一个数据库 create database homework default character set utf8; use homework; 1.1 默认值约束(default) -- 数 ...

  3. jsp中核心标签使用

    <%@ page language="java" import="java.util.*, java.lang.*" pageEncoding=" ...

  4. [源码解析] 深度学习流水线并行之PopeDream(1)--- Profile阶段

    [源码解析] 深度学习流水线并行之PopeDream(1)--- Profile阶段 目录 [源码解析] 深度学习流水线并行之PopeDream(1)--- Profile阶段 0x00 摘要 0x0 ...

  5. 使用 IDEA 配合 Dockerfile 部署 SpringBoot 工程

    准备 SpringBoot 工程 新建 SpringBoot 项目,默认的端口是 8080 ,新建 Controller 和 Mapping @RestController public class ...

  6. JAVA 之 每日一记 之 算法 ( 长按键入 )

    题目详解: 你的朋友正在使用键盘输入他的名字 name.偶尔,在键入字符 c 时,按键可能会被长按,而字符可能被输入 1 次或多次. 你将会检查键盘输入的字符 typed.如果它对应的可能是你的朋友的 ...

  7. mini-ndn0.5.0 安装教程 (避免踩坑)

    写在前面 首先需要确定一些配置,因为在安装的过程中需要编译一些内容,所以需要提前准备好. 本人之前ubuntu系统可能比较乱,在尝试很多次安装后,仍然失败,所以就直接重装了一下.说一下我自己的一些配置 ...

  8. WebService学习总结(二)--使用JDK开发WebService

    一.WebService的开发方法 使用java的WebService时可以使用一下两种开发手段 使用jdk开发(1.6及以上版本) 使用CXF框架开发(工作中) 二.使用JDK开发WebServic ...

  9. shp的基本操作

    本节将介绍如何利用python完成对shp的基本操作 1.读取shp四至 import shapefile sf = shapefile.Reader(r"E:\shp\1.shp" ...

  10. Selenium系列(十八) - Web UI 自动化基础实战(5)

    如果你还想从头学起Selenium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1680176.html 其次,如果你不懂前端基础知识, ...