当前位置: 首页 > news >正文

LeetCode 第71题 简化路径(繁琐)

给你一个字符串path,表示指向某一文件或目录的Unix风格 绝对路径(以‘/’开头),请你将其转化为更加简洁的规范路径。

在Unix风格的文件系统中规则如下:

  • 一个点‘.’表示当前目录本身。
  • 此外,两个点‘..’表示将目录切换到上一级(指向父目录)
  • 任意多个连续的斜杠(即‘//’或‘///’)都被视为单个斜杠‘/’。
  • 任何其他格式的点(例如,‘...’或‘....’)均被视为有效的文件/目录名称。

返回的简化路径必须遵循下述格式:

  • 始终以斜杠‘/’开头
  • 两个目录名之间必须只有一个斜杠‘/’。
  • 最后一个目录名(如果存在)不能以‘/’结尾。
  • 此外,路径仅包含从根目录到目标文件或目录的路径上的目录(即不含‘.’或‘..’)。

返回简化后得到的规范路径。

示例1:

输入:path = "/home/"输出:"/home"解释:应删除尾随斜杠。

示例2:

输入:path = "/home//foo/"输出:"/home/foo"解释:多个连续的斜杠被单个斜杠替换。

示例3:

输入:path = "/home/user/Documents/../Pictures"输出:"/home/user/Pictures"解释:两个点 ".." 表示上一级目录(父目录)。

示例4:

输入:path = "/../"输出:"/"解释:不可能从根目录上升一级目录。

示例5:

输入:path = "/.../a/../b/c/../d/./"输出:"/.../b/d"解释:"..." 在这个问题中是一个合法的目录名。

提示:

1 <= path.length <= 3000
path 由英文字母,数字,'.','/' 或 '_' 组成。
path 是一个有效的 Unix 风格绝对路径。

题解1:

 栈:首先将给定的字符串path根据 / 分割成一个由若干字符串组成的列表,记为names。根据题目中规定的【规范格式的下述格式】,names中包含的字符串只能为以下几种:

  • 空字符串,例如当出现多个连续的 / ,就会分割出空字符串。
  • 一个点 . 
  • 两个点 ..
  • 只包含英文字母、数字或  _ 的目录名

对于【空字符串】以及【一个点】,实际上无需对他们进行处理,因为【空字符串】没有任何含义,而【一个点】表示当前目录本身,无需切换目录。

对于【两个点】或者【目录名】,我们则可以用一个栈来维护路径中的每一个目录名。当我们遇到【两个点】时,需要将目录切换到上一级,因此只要栈不为空,就弹出栈顶的目录。当遇到【目录名】时,就把它放入栈。

只需要遍历names中的每个字符串并进行上述操作即可。

在所有操作完成后,将栈底到栈顶的字符串用 / 进行连接,再在最前面加上  /  表示根目录。

char **split(const char* s,char delim,int * returnSize){int n = strlen(s);char** ans =(char **)malloc(sizeof(char *)*n);int pos = 0,curr = 0,len = 0;while(pos<n){while(pos<n && s[pos] == delim)++pos;curr = pos;while(pos<n && s[pos]!=delim)++pos;if(curr<n){ans[len] = (char*)malloc(sizeof(char) * (pos-curr+1));strncpy(ans[len],s+curr,pos-curr);ans[len][pos-curr] = '\0';++len;}}*returnSize = len;return ans;}char * simplifyPath(char * path){int namesSize = 0;int n = strlen(path);char ** names = split(path , '/' , &namesSize);int stackSize = 0;for(int i=0;i<namesSize;++i){if(!strcmp(names[i],"..")){if(stackSize>0)  --stackSize;}else if(strcmp(names[i],".")){stack[stackSize] = names[i];++stackSize;}}char * ans = (char *)malloc(sizeof(char) * (n + 1));int curr = 0;if (stackSize == 0) {ans[curr] = '/';++curr;} else {for (int i = 0; i < stackSize; ++i) {ans[curr] = '/';++curr;strcpy(ans + curr, stack[i]);curr += strlen(stack[i]);}}ans[curr] = '\0';for (int i = 0; i < namesSize; ++i) {free(names[i]);}free(names);free(stack);return ans;}

题解2:

先用strtok函数将/分割的分解,每次分解判断是否为‘.’、‘..’,如果为‘..’,回退至上一级目录,所以size自减1,但若是size本身是0,即在根目录是无法回退到更上一级目录的,所以做一个size值的保护,不让他成为负数值;
如果都不满足,存入stack,size++,进入下一层循环;
循环完后,如果size == 0,直接返回"/";
如果不是,将各字符串存入res,“/”隔开,返回res.

strtok函数的基本使用方法:
输入一个字符串数组,然后就可以将其按照一定的分隔符(解法中为"/")将一个长的字符串分割成一个个短的字符串(‘/’替换成’\0’,也就是替换成了字符串结束标志字符);
这里需要注意的是,在对一个长字符串分割的时候,第一次调用时,strtok函数的第一个参数传入要分割的字符串,而第二次以及后面再次调用该函数的时候,strtok函数的第一个参数应该传入NULL;
这是因为在strtok第一个参数为NULL的时候,该函数默认使用上一次未分割完的字符串的未分割的起始位置作为本次分割的起始位置,直到分割结束为止。

strcmp()函数返回一个int或整数类型。 我们可以得到以下三种返回值类型。
如果两个字符串相同,相等或相同,则返回“ 0”;
“负整数”,如果第一个不匹配字符的ASCII值小于第二个字符;
如果第一个不匹配字符的ASCII值大于第二个,则为“正整数”

char * simplifyPath(char * path){//strtok本身会舍弃空字符串,strcat来附加。char *stack[100];int size = 0;for (char *s = strtok(path, "/"); s; s = strtok(NULL, "/")) {if (strcmp(s, ".") == 0) {//do nothing} else if (strcmp(s, "..") == 0) {//back size = fmax(0, size-1);} else {stack[size++] = s;}}if (size == 0) return "/";char *res = calloc(1000, sizeof(char));for (int i=0; i<size; ++i) {strcat(res, "/");strcat(res, stack[i]);}return res;
}

相关文章:

  • 打牙祭是什么意思
  • SCADA|信创KingSCADA4.0历史报警查询的差异
  • XCTF-misc-János-the-Ripper
  • ELK日志文件分析系统——E(Elasticsearch)
  • Karate UI测试之驱动配置
  • vulnhub-Earth
  • SD和comfyui常用模型介绍和下载
  • 什么是泛型,如何使用它?
  • 【LangChain】4 基于文档的问答
  • 操作系统多级存储模型
  • Python 使用 DrissionPage 模块进行爬虫
  • pikachu靶场通关笔记30 文件包含01之本地文件包含
  • 山东大学软件学院项目实训:基于大模型的模拟面试系统项目总结(十)
  • Apache Doris FE 问题排查与故障分析全景指南
  • Vue Methods 实现原理详解
  • UGPCL
  • 手机验证码自动化处理:从原理到企业级解决方案
  • 微信小程序开发 picker选择年月日+时分秒
  • 【论文阅读】Multi-Class Cell Detection Using Spatial Context Representation
  • C# 使用 TreeView 实践 WinRiver II 的测量管理功能
  • 网站手机版怎么做/端口扫描站长工具
  • 南昌企业制作网站设计/百度手机卫士
  • 云县网站建设 云县网/江苏企业网站建设
  • 跟我一起做网站/百度云网盘网页版
  • 设计相关网站/seo推广营销靠谱
  • 简单的网站制作/友情链接交换要注意哪些问题