数据结构:力扣OJ题(每日一练)2

简介: 数据结构:力扣OJ题(每日一练)2

题一:有效的括号

给定一个只包括 '('')''{''}''['']' 的字符串 s ,判断字符串是否有效。

有效字符串需满足:

  1. 左括号必须用相同类型的右括号闭合。
  2. 左括号必须以正确的顺序闭合。
  3. 每个右括号都有一个对应的相同类型的左括号。
  4. 示例 2:

输入:s = "()[]{}"

输出:true

思路一:

      第一步:写出数组栈的结构体,然后分别写出栈的初始化,入栈,出栈,获取栈顶元素,销毁栈,检验栈是否为空的函数;

       第二步:创建一个结构体变量,初始化,while(*s)决定是否继续循环,用switch找到对应的前置符号将他们入栈,如果是后置符号,则先判断ps中是否为空,然后再判断是否有对应的前置符合,没有:直接结束,:继续循环;

       第三步:创建相应的bool型变量记录SLEmpty()函数的返回值,销毁创建空间。

注意:OJ题不会判断代码开辟动态内存空间,所以在OJ题,不销毁开辟的动态内存也正确。

//约定类型方便更改类型
typedef char STDataType;
typedef struct Stack
{
  STDataType* a;
  int top;
  int capacity;
}SL;
//初始化
void SLInit(SL* ps)
{
  assert(ps);
  ps->a = NULL;
  ps->capacity = ps->top = 0;
}
//入栈
void SLPush(SL* ps, STDataType x)
{
  assert(ps);
  //栈顶=容量说明需要扩容
  if (ps->capacity == ps->top)
  {
    int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
    STDataType* tmp = (STDataType*)realloc(ps->a,sizeof(STDataType) * newcapacity);
    if (tmp == NULL)
    {
      perror("realloc fail");
      exit(-1);
    }
    ps->capacity = newcapacity;
    ps->a = tmp;
  }
  ps->a[ps->top] = x;
  //后缀++方便下一次入栈和打印栈顶
  ps->top++;
}
//出栈
void SLPop(SL* ps)
{
  assert(ps);
  //为空情况“0”
  assert(ps->top > 0);
  //
  --ps->top;
}
//获得栈顶元素
STDataType SLTTop(SL* ps)
{
  assert(ps);
  //为空情况“0”
  assert(ps->top > 0);
  int n = (ps->top) - 1;
  return ps->a[n];
}
//获取栈中有效元素个数
int SLSize(SL* ps)
{
  assert(ps);
  return ps->top;
}
//销毁栈
void SLDestroy(SL* ps)
{
  assert(ps);
  //开辟数组优势:一次全部释放
  free(ps->a);
  ps->a = NULL;
  ps->capacity = ps->top = 0;
}
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0
bool SLEmpty(SL* ps)
{
  assert(ps);
  //为“0”说明为NULL
  if (ps->top == 0)
  {
    return true;
  }
  return false;
}
//思路一:使用switch
bool isValid(char * s)
{
  SL ps;
  char val;
  //初始化
  SLInit(&ps);
  while (*s)
  {
    switch (*s)
    {
    case '(':
    case '[':
    case '{':
        //入栈
        SLPush(&ps, *s);
        break;
    case ')':
    case '}':
    case ']':
        //判断栈是否为空
        if (SLEmpty(&ps))
        {
          //销毁开辟的动态内存
          SLDestroy(&ps);
            return false;
        }
        //栈顶值
        val = SLTTop(&ps);
        //出栈
        SLPop(&ps);
        if (*s == ')' && val != '(' ||
            *s == ']' && val != '[' ||
            *s == '}' && val != '{')
        {
            SLDestroy(&ps);
            return false;
        }
        break;
    }
    s++;
  }
  bool ret = SLEmpty(&ps);
  SLDestroy(&ps);
  return ret;
}

改进简化:

       原理同上,要实现的内容也同上,不过简化了switch()语句造成的多次调用,以及步骤上的增加,改用if----else语句后,代码更加的简洁清晰不需要考虑每种后置符号的操作,if一次性全部解决了。总体上减少了三分之一的代码量

//约定类型方便更改类型
typedef char STDataType;
typedef struct Stack
{
  STDataType* a;
  int top;
  int capacity;
}SL;
//初始化
void SLInit(SL* ps)
{
  assert(ps);
  ps->a = NULL;
  ps->capacity = ps->top = 0;
}
//入栈
void SLPush(SL* ps, STDataType x)
{
  assert(ps);
  //栈顶=容量说明需要扩容
  if (ps->capacity == ps->top)
  {
    int newcapacity = ps->capacity == 0 ? 4 : 2 * ps->capacity;
    STDataType* tmp = (STDataType*)realloc(ps->a,sizeof(STDataType) * newcapacity);
    if (tmp == NULL)
    {
      perror("realloc fail");
      exit(-1);
    }
    ps->capacity = newcapacity;
    ps->a = tmp;
  }
  ps->a[ps->top] = x;
  //后缀++方便下一次入栈和打印栈顶
  ps->top++;
}
//出栈
void SLPop(SL* ps)
{
  assert(ps);
  //为空情况“0”
  assert(ps->top > 0);
  //
  --ps->top;
}
//获得栈顶元素
STDataType SLTTop(SL* ps)
{
  assert(ps);
  //为空情况“0”
  assert(ps->top > 0);
  int n = (ps->top) - 1;
  return ps->a[n];
}
//销毁栈
void SLDestroy(SL* ps)
{
  assert(ps);
  //开辟数组优势:一次全部释放
  free(ps->a);
  ps->a = NULL;
  ps->capacity = ps->top = 0;
}
// 检测栈是否为空,如果为空返回非零结果,如果不为空返回0
bool SLEmpty(SL* ps)
{
  assert(ps);
  //为“0”说明为NULL
  if (ps->top == 0)
  {
    return true;
  }
  return false;
}
//改进简化内容if——else
bool isValid(char * s)
{
  SL ps;
  char val;
  //初始化
  SLInit(&ps);
  while (*s)
  {
       if(*s == '(' || *s == '{' || *s == '[')
    {
        //入栈
        SLPush(&ps, *s);
    }
    else
    {
          //判断栈是否为空
        if (SLEmpty(&ps))
        {
      //销毁开辟的动态内存
      SLDestroy(&ps);
      return false;
        }
      //栈顶值
      val = SLTTop(&ps);
      //出栈
      SLPop(&ps);
      if (*s == ')' && val != '(' ||
        *s == ']' && val != '[' ||
        *s == '}' && val != '{')
      {
        SLDestroy(&ps);
        return false;
      }
    }
    s++;
  }
  bool ret = SLEmpty(&ps);
  SLDestroy(&ps);
  return ret;
}

本人实力有限可能对一些地方解释的不够清晰,可以自己尝试读代码,望海涵!

目录
相关文章
|
4天前
|
DataX
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
4月前
|
存储 Java
数据结构第三篇【链表的相关知识点一及在线OJ习题】
数据结构第三篇【链表的相关知识点一及在线OJ习题】
46 7
|
4月前
|
索引
力扣(LeetCode)数据结构练习题(3)------链表
力扣(LeetCode)数据结构练习题(3)------链表
121 0
|
4月前
力扣(LeetCode)数据结构练习题(2)
力扣(LeetCode)数据结构练习题(2)
39 0
|
4月前
|
存储
力扣(LeetCode)数据结构练习题
力扣(LeetCode)数据结构练习题
76 0
|
7月前
|
存储 索引
【数据结构OJ题】设计循环队列
力扣题目——设计循环队列
49 1
【数据结构OJ题】设计循环队列
|
7月前
【数据结构OJ题】用栈实现队列
力扣题目——用栈实现队列
52 0
【数据结构OJ题】用栈实现队列
|
1月前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
142 77
|
1月前
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
38 7
|
1月前
|
存储 C++ 索引
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
44 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】

热门文章

最新文章