? ? ? ? 栈也是一种线性数据结构,最主要的特点是入栈顺序和出栈顺序是相反的,操作时只能从栈顶进行操作,在Java中给我们提供了一个泛型栈——Stack,其中最常用的方法有:
? ? ? ? 因为栈也是一种线性结构,所以这里可以利用之前我们写的数组,这里可以用接口的方式来实现我们自己的顺序栈,下面进行代码演示。
public interface Stack_i <Elem>{
public abstract boolean push(Elem e);
public abstract Elem pop();
public abstract Elem peek();
public abstract int getLength();
public abstract boolean isEmpty();
}
public class ArrStack<E> implements Stack_i<E>{
private MyArray<E> stack;
private int length;
public ArrStack() {
this.stack = new MyArray<>(16);
length = 0;
}
@Override
public boolean push(E e) {
try {
stack.add(e);
}catch (Exception exception){
return false;
}
this.length++;
return true;
}
@Override
public E pop() {
if(this.getLength()==0) return null;
E e;
try {
e = stack.removeByIndex(this.length - 1);
}catch (Exception exception){
return null;
}
this.length--;
return e;
}
@Override
public E peek() {
if(this.getLength()==0) return null;
E e;
try {
e = stack.getValueByIndex(this.length - 1);
}catch (Exception exception){
return null;
}
return e;
}
@Override
public int getLength() {
return this.length;
}
@Override
public boolean isEmpty() {
return this.length==0;
}
}
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
public class testMyStack {
public static void test(ArrStack<Integer> stack, List<Integer> list){
Random random = new Random();
for(int i = 0;i < 10;i++){
stack.push(random.nextInt(1000));
System.out.println(stack.peek()+"——现在还有"+stack.getLength()+"个元素");
}
Integer temp;
while ((temp = stack.pop())!=null){
list.add(temp);
}
System.out.println();
}
public static void main(String[] args) {
List<Integer> list = new ArrayList();
ArrStack<Integer> stack = new ArrStack();
test(stack,list);
for (Integer i:list) {
System.out.println(i);
}
}
}
输出结果?
? ? ? ? 在测试中,我们使用循环对栈先进行入栈,入栈元素分别为377、338、269、107、129、66、101、760、977、786,之后我们又循环出栈到list集合中,出栈顺序为786、977、760、101、66、129、107、268、338、377,从案例中可以看出,栈这种数据结构的特点——先进后出。?
? ? ? ? 这种类型的题就适合用栈这种数据结构,这里我用数组集合的做法和栈的做法做比较:
import java.util.ArrayList;
class Solution {
public static boolean isValid(String s) {
ArrayList<Character> list = new ArrayList<>();
int length = 0;
while(length < s.length()) {
if(list.size()==0){
list.add(s.charAt(length++));
continue;
}
if(list.get(list.size()-1).equals('(')&&s.charAt(length)==')'){
list.remove(list.size()-1);
length++;
continue;
}
if(list.get(list.size()-1).equals('[')&&s.charAt(length)==']'){
list.remove(list.size()-1);
length++;
continue;
}
if(list.get(list.size()-1).equals('{')&&s.charAt(length)=='}'){
list.remove(list.size()-1);
length++;
continue;
}
list.add(s.charAt(length++));
}
return list.isEmpty();
}
}
输出结果:
class Solution {
public boolean isValid(String s) {
Stack<Character> stack = new Stack<>();
char[] chars = s.toCharArray();
for(int i = 0;i < chars.length;i++){
if(chars[i]=='('||chars[i]=='['||chars[i]=='{') stack.push(chars[i]);
if(chars[i]==')'||chars[i]==']'||chars[i]=='}'){
if(stack.isEmpty()){
return false;
}
char char_temp = stack.pop();
if(char_temp=='('&&chars[i]!=')') return false;
if(char_temp=='['&&chars[i]!=']') return false;
if(char_temp=='{'&&chars[i]!='}') return false;
}
}
return stack.size()==0;
}
}
?输出结果:
? ? ? ? ?这里简单说明一下这道题的思路,因为要判断括号是否匹配,则左括号和右括号都是一一对应的,若出现了不对应的情况,则视为不匹配,我们可以遇到左括号时进栈而遇到右括号时出栈,此时,若括号匹配,则栈中现在只有左括号,我们只用判断出栈元素是否与当前循环元素匹配就行,循环结束后,栈应该为空。在这其中,只要有一次匹配失败,则返回false。总结一下,我们一个需要注意这几个要求: