2018-10-10 18:03:03 -04:00
|
|
|
|
---
|
|
|
|
|
id: 59667989bf71cf555dd5d2ff
|
2020-12-16 00:37:30 -07:00
|
|
|
|
title: S-表达式
|
2018-10-10 18:03:03 -04:00
|
|
|
|
challengeType: 5
|
|
|
|
|
videoUrl: ''
|
2021-01-13 03:31:00 +01:00
|
|
|
|
dashedName: s-expressions
|
2018-10-10 18:03:03 -04:00
|
|
|
|
---
|
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
# --description--
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
<p> <a href='https://en.wikipedia.org/wiki/S-Expression' title='wp:S表达式'>S-Expressions</a>是一种解析和存储数据的便捷方式。 </p>任务: <p>为S-Expressions编写一个简单的读取器/解析器,处理引用的和不带引号的字符串,整数和浮点数。 </p><p>该函数应从字符串中读取单个但嵌套的S-Expression,并将其作为(嵌套)数组返回。 </p><p>除非包含在带引号的字符串中,否则可以忽略换行符和其他空格。 </p><p> “ <tt>()</tt> ”内部引用的字符串不会被解释,但会被视为字符串的一部分。 </p><p>处理字符串中的转义引号是可选的;因此“ <tt>(foo”bar)</tt> “可能被视为字符串” <tt>foo“bar</tt> ”,或作为错误。 </p><p>为此,读者无需识别“ <tt>\</tt> ”以进行转义,但如果语言具有适当的数据类型,则还应识别数字。 </p><p>请注意,除了“ <tt>()”</tt> “(” <tt>\</tt> “,如果支持转义)和空格,没有特殊字符。其他任何内容都是允许的,不带引号。 </p><p>读者应该能够阅读以下输入</p><p></p><pre> ((数据“引用数据”123 4.5)
|
|
|
|
|
(数据(!@#(4.5)“(更多”“数据”))))
|
|
|
|
|
</pre><p></p><p>并将其转换为本机数据结构。 (有关本机数据结构的示例,请参阅<a href='http://rosettacode.org/wiki/#Pike' title='#Pike'>Pike</a> , <a href='http://rosettacode.org/wiki/#Python' title='#蟒蛇'>Python</a>和<a href='http://rosettacode.org/wiki/#Ruby' title='#红宝石'>Ruby</a>实现。) </p>
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
# --hints--
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
`parseSexpr`是一个函数。
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
|
|
|
|
```js
|
2020-12-16 00:37:30 -07:00
|
|
|
|
assert(typeof parseSexpr === 'function');
|
2018-10-10 18:03:03 -04:00
|
|
|
|
```
|
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
`parseSexpr("(data1 data2 data3)")`应返回[“data1”,“data2”,“data3”]“)
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
|
|
|
|
```js
|
2020-12-16 00:37:30 -07:00
|
|
|
|
assert.deepEqual(parseSexpr(simpleSExpr), simpleSolution);
|
2018-10-10 18:03:03 -04:00
|
|
|
|
```
|
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
`parseSexpr("(data1 data2 data3)")`应该返回一个包含3个元素的数组“)'
|
2018-10-10 18:03:03 -04:00
|
|
|
|
|
|
|
|
|
```js
|
2020-12-16 00:37:30 -07:00
|
|
|
|
assert.deepEqual(parseSexpr(basicSExpr), basicSolution);
|
2018-10-10 18:03:03 -04:00
|
|
|
|
```
|
2020-08-13 17:24:35 +02:00
|
|
|
|
|
2021-01-13 03:31:00 +01:00
|
|
|
|
# --seed--
|
|
|
|
|
|
|
|
|
|
## --after-user-code--
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
const simpleSExpr = '(data1 data2 data3)';
|
|
|
|
|
const simpleSolution = ['data1', 'data2', 'data3'];
|
|
|
|
|
|
|
|
|
|
const basicSExpr = '((data "quoted data" 123 4.5) (data (!@# (4.5) "(more" "data)")))';
|
|
|
|
|
const basicSolution = [["data","\"quoted data\"",123,4.5],["data",["!@#",[4.5],"\"(more\"","\"data)\""]]];
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## --seed-contents--
|
|
|
|
|
|
|
|
|
|
```js
|
|
|
|
|
function parseSexpr(str) {
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-12-16 00:37:30 -07:00
|
|
|
|
# --solutions--
|
|
|
|
|
|
2021-01-13 03:31:00 +01:00
|
|
|
|
```js
|
|
|
|
|
function parseSexpr(str) {
|
|
|
|
|
const t = str.match(/\s*("[^"]*"|\(|\)|"|[^\s()"]+)/g);
|
|
|
|
|
for (var o, c = 0, i = t.length - 1; i >= 0; i--) {
|
|
|
|
|
var n,
|
|
|
|
|
ti = t[i].trim();
|
|
|
|
|
if (ti == '"') return;
|
|
|
|
|
else if (ti == '(') t[i] = '[', c += 1;
|
|
|
|
|
else if (ti == ')') t[i] = ']', c -= 1;
|
|
|
|
|
else if ((n = +ti) == ti) t[i] = n;
|
|
|
|
|
else t[i] = `'${ti.replace('\'', '\\\'')}'`;
|
|
|
|
|
if (i > 0 && ti != ']' && t[i - 1].trim() != '(') t.splice(i, 0, ',');
|
|
|
|
|
if (!c) if (!o) o = true; else return;
|
|
|
|
|
}
|
|
|
|
|
return c ? undefined : eval(t.join(''));
|
|
|
|
|
}
|
|
|
|
|
```
|