原文:从一道面试题谈起,作者:360奇舞团 刘观宇

题目:

创建一个函数来判断给定的表达式中的大括号是否闭合,返回 true/false,对于空字符串,返回 true

var expression = "{{}}{}{}"
var expressionFalse = "{}{{}" function isBalanced (exp) {}

题目本身比较简单。看完文章实现,感觉实现思路很重要,更要能举一反三。

虽然大学的时候也学了数据结构,栈,但应用的很少,看完本文,对栈在实际工作中的应用,也有了全新的思路与看法。

实现

以下是应用数据结构栈的特点,先进先出,在 js 中用数组模拟实现的代码。

function isBalanced (exp) {
if (!(exp + '').trim()) {
return true;
} let stack = [];
let arr = exp.trim().split('');
for (let i = 0; i < arr.length; i++) {
let item = arr[i];
if (item === '{') {
stack.push(item);
} else if (item === '}') {
if (!stack.length) {
return false;
}
stack.pop();
}
} return stack.length === 0
} // test
var expT = '{{}}{}{}';
var expF = '{}{{}' console.log(isBalanced('')); // true
console.log(isBalanced(expT)); // true
console.log(isBalanced(expF)); // false

利用栈先进先出的特点,找到一个 { ,就压入栈中;而找到一个 } ,如栈中是空的,则直接返回false,否则就从栈中弹出一个 { 。循环结束后,通过判断栈是否为空,判断字符中中 {} 是否闭合。

举一反三

题目一:实现函数 isBalanced, 用 true/false 表示给定的字符串的括号是否平衡(一一对应)。注意要支持三种类型的括号,带有交错括号的字符串应该返回false

isBalanced('(foo { bar (baz) [boo] })') // true
isBalanced('foo { bar { baz }') // false
isBalanced('foo { (bar [baz] } )') // true

实现思路和上面的是一致的,也是利用栈的特性。

过滤无效字符,每一种有括号有一种唯一的左括号与之对应。

实现:

function isBalanced (exp) {
if (!(exp + '').trim()) {
return true;
} let stack = [];
let signObj = {
'{': '}',
'(': ')',
'[': ']'
}; let arr = exp.trim().split('');
for (let i = 0; i < arr.length; i++) {
let item = arr[i];
if (signObj.hasOwnProperty(item)) { // stack 中只压入指定符号
stack.push(item);
} else if (Object.values(signObj).includes(item)) { // 针对指定符号的值比对,得到对象key
let leftSign;
for(let key in signObj) {
if (item === signObj[key]) {
leftSign = key;
}
}
if (stack[stack.length - 1] !== leftSign) {
return false;
}
stack.pop();
}
} return stack.length === 0;
} // test
console.log(isBalanced('')); // true
console.log(isBalanced('(foo { bar (baz) [boo] })')); // true
console.log(isBalanced('foo { bar { baz }')); // false
console.log(isBalanced('foo { (bar [baz] } )')); // false

通过对象的键值对过滤掉无效字符,再通过值找到对应的key,和栈中最后压入的进行比较,不相等就是不匹配直接false,相等就表示匹配,然后把栈中的最后的弹出。

通过map和es6的语法,可以把代码写的一气呵成

function isBalanced (exp) {
if (!(exp + '').trim()) {
return true;
} const map = new Map([
['{', '}'],
['(', ')'],
['[', ']']
]); let stack = [];
let arr = exp.trim().split('');
for (let i = 0; i < arr.length; i++) {
let item = arr[i];
if (map.has(item)) {
stack.push(item);
} else if ([...map.values()].includes(item)) {
if (stack[stack.length - 1] !== [...map.entries()].filter(el => el[1] === item).pop().shift()) {
return false;
}
stack.splice(stack.length - 1, 1);
}
} return stack.length === 0
}

说明:[...map.entries()]的结果是二维数组,这个看懂了,就比较容易理解了

[
['{', '}'],
['(', ')'],
['[', ']']
]

题目二:要求严格限制括号的顺序,即中括号外围只能是大括号,内部只能是小括号。也即:括号只能以大括号、中括号、小括号的顺序只能前面的包含后面的,不能后面的包含前面的,用代码来表示一下

isStrictBalanced('foo { bar (baz) [boo] }') // true

isStrictBalanced('(foo { bar (baz) [boo] })') // false

实现思路,在入栈的时候判断优先级。

怎么判断优先级,利用字符比较。这是我没想到的,都不知道这三个的具体charCodeAt值。(_)

"{".charCodeAt() === 123,"[".charCodeAt() === 91,"(".charCodeAt() === 40

实现:

function isBalanced (exp) {
if (!(exp + '').trim()) {
return true;
} let stack = [];
let signObj = {
'{': '}',
'[': ']',
'(': ')'
};
let keys = Object.keys(signObj); let arr = exp.trim().split('');
for (let i = 0; i < arr.length; i++) {
let item = arr[i];
if (signObj.hasOwnProperty(item)) {
// stack 中只压入指定符号,并且判断优先级
// "{".charCodeAt() === 123,"[".charCodeAt() === 91,"(".charCodeAt() === 40
if (stack.length) {
let pop = stack.slice().pop()
if (pop < item) {
return false
}
}
stack.push(item);
} else if (Object.values(signObj).includes(item)) { // 针对指定符号的值比对
let leftSign;
for(let key in signObj) {
if (item === signObj[key]) {
leftSign = key;
}
}
if (stack[stack.length - 1] !== leftSign) {
return false;
}
stack.pop();
}
} return stack.length === 0;
} // test
console.log(isBalanced('')); // true
console.log(isBalanced('foo { bar (baz) [boo] }')); // true
console.log(isBalanced('(foo { bar (baz) [boo] })')); // false
console.log(isBalanced('(foo [bar])')); // false
console.log(isBalanced('[foo (bar)]')); // true
console.log(isBalanced('[foo (bar)] {bar [boo] }')); // true

思路通了,用map和es6实现其实也很容易,就不浪费笔墨了。

isBalanced函数实现的更多相关文章

  1. Python 小而美的函数

    python提供了一些有趣且实用的函数,如any all zip,这些函数能够大幅简化我们得代码,可以更优雅的处理可迭代的对象,同时使用的时候也得注意一些情况   any any(iterable) ...

  2. 探究javascript对象和数组的异同,及函数变量缓存技巧

    javascript中最经典也最受非议的一句话就是:javascript中一切皆是对象.这篇重点要提到的,就是任何jser都不陌生的Object和Array. 有段时间曾经很诧异,到底两种数据类型用来 ...

  3. JavaScript权威指南 - 函数

    函数本身就是一段JavaScript代码,定义一次但可能被调用任意次.如果函数挂载在一个对象上,作为对象的一个属性,通常这种函数被称作对象的方法.用于初始化一个新创建的对象的函数被称作构造函数. 相对 ...

  4. C++对C的函数拓展

    一,内联函数 1.内联函数的概念 C++中的const常量可以用来代替宏常数的定义,例如:用const int a = 10来替换# define a 10.那么C++中是否有什么解决方案来替代宏代码 ...

  5. 菜鸟Python学习笔记第一天:关于一些函数库的使用

    2017年1月3日 星期二 大一学习一门新的计算机语言真的很难,有时候连函数拼写出错查错都能查半天,没办法,谁让我英语太渣. 关于计算机语言的学习我想还是从C语言学习开始为好,Python有很多语言的 ...

  6. javascript中的this与函数讲解

    前言 javascript中没有块级作用域(es6以前),javascript中作用域分为函数作用域和全局作用域.并且,大家可以认为全局作用域其实就是Window函数的函数作用域,我们编写的js代码, ...

  7. 复杂的 Hash 函数组合有意义吗?

    很久以前看到一篇文章,讲某个大网站储存用户口令时,会经过十分复杂的处理.怎么个复杂记不得了,大概就是先 Hash,结果加上一些特殊字符再 Hash,结果再加上些字符.再倒序.再怎么怎么的.再 Hash ...

  8. JS核心系列:浅谈函数的作用域

    一.作用域(scope) 所谓作用域就是:变量在声明它们的函数体以及这个函数体嵌套的任意函数体内都是有定义的. function scope(){ var foo = "global&quo ...

  9. C++中的时间函数

    C++获取时间函数众多,何时该用什么函数,拿到的是什么时间?该怎么用?很多人都会混淆. 本文是本人经历了几款游戏客户端和服务器开发后,对游戏中时间获取的一点总结. 最早学习游戏客户端时,为了获取最精确 ...

随机推荐

  1. 2018.11.07 codeforces559C. Gerald and Giant Chess(dp+组合数学)

    传送门 令f[i]f[i]f[i]表示对于第iii个棋子,从(1,1)(1,1)(1,1)出发到它不经过其它棋子的方案数. 于是我们假设(h,w)(h,w)(h,w)有一个棋子,求出它的fff值就可以 ...

  2. Java基础-时间类

    关于java中六个时间类的使用和区别 java.util.Date java.sql.Date ,java.sql.Time , java.sql.Timestamp java.text.Simple ...

  3. 4. Father's Impact on a Child's Language Development 父亲对孩子语言发展的影响

    4. Father's Impact on a Child's Language Development 父亲对孩子语言发展的影响 (1)Im families with two working pa ...

  4. python3.4对已经存在的excel写入数据

    #!/usr/bin/env python # -*- coding:utf-8 -*- # __author__ = "blzhu" """ pyt ...

  5. Oracle数据库使用mybatis的时候,实体类日期为Date类型,mybatis里面定义的是Date类型,插入的时候,时分秒全部是12:00:00问题

    实体类中日期定义的是Date类型的,没毛病: 我在mybatis里面定义的是Date类型的,进行测试的时候发现,数据库插入的日期的时分秒全部都是一样的,都是12:00:00,很郁闷: 后来把mybat ...

  6. IntelliJ IDEA 2017版 spring-boot2.0.4的yml配置使用

    一.必须配置字端两个 server: port: 8080 servlet: context-path: /demo 二.两种mvc转换springboot,一种是注解,一种就是.yml或proper ...

  7. Eclipse错误: 找不到或无法加载主类或项目无法编译10种解决大法

    1.在src文件夹上点右键-Build Path-Use as Source Folder,重新进行编译,一切正常了.2.在Eclipse工程文件夹上点右键-Refresh,重新编译,一功OK(这个方 ...

  8. BZOJ 2440 [中山市选2011]完全平方数 (二分 + 莫比乌斯函数)

    2440: [中山市选2011]完全平方数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 4805  Solved: 2325[Submit][Sta ...

  9. Hadoop-2.8.0分布式安装手册

    目录 目录 1 1. 前言 3 2. 特性介绍 3 3. 部署 5 3.1. 机器列表 5 3.2. 主机名 5 3.2.1. 临时修改主机名 6 3.2.2. 永久修改主机名 6 3.3. 免密码登 ...

  10. (转)Memcache内存分配策略

    转自:http://hi.baidu.com/software_one/item/0a0a6712dc7a319899ce33e0 一.Memcache内存分配机制 关于这个机制网上有很多解释的,我个 ...