Skip to content

Commit e58c817

Browse files
author
caiyongmin
committed
feat(dsa): tow sum/three sum/dp/number/string
1 parent 91d38e4 commit e58c817

15 files changed

+414
-45
lines changed

README.md

+43-45
Original file line numberDiff line numberDiff line change
@@ -1,59 +1,57 @@
1-
# Codes
1+
# awesome-coding-javascript
22

3-
📌 some codes, verify code logic with unit tests.
3+
📌 coding about JavaScript basic principles and data structures and algorithms, verify logic with unit test.
44

5-
[![codecov](https://codecov.io/gh/caiyongmin/codes/branch/master/graph/badge.svg)](https://codecov.io/gh/caiyongmin/codes)
5+
[![codecov](https://codecov.io/gh/caiyongmin/codes/branch/master/graph/badge.svg)](https://codecov.io/gh/caiyongmin/codes) [![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](README.md)
66

77
## JavaScript
88

9-
- [x] [bind](./src/javascript/bind/bind.test.js)
10-
- [x] [call](./src/javascript/call/call.js)
11-
- [x] [apply](./src/javascript/apply/apply.js)
12-
- [x] [new](./src/javascript/new/new.test.js)
13-
- [x] [deepclone](./src/javascript/deepclone/deepclone.js)
14-
- [x] [throttle](./src/javascript/throttle/throttle.js)
15-
- [x] [debounce](./src/javascript/debounce/debounce.js)
16-
- [x] [event-emitter](./src/javascript/event-emitter/event-emitter.js)
17-
- [x] [Promise](./src/javascript/promise/promise.html)
9+
- [x] [bind](./src/javascript/bind)
10+
- [x] [call](./src/javascript/call)
11+
- [x] [apply](./src/javascript/apply)
12+
- [x] [new](./src/javascript/new)
13+
- [x] [deepclone](./src/javascript/deepclone)
14+
- [x] [throttle](./src/javascript/throttle)
15+
- [x] [debounce](./src/javascript/debounce)
16+
- [x] [EventEmitter](./src/javascript/event-emitter)
17+
- [x] [Promise](./src/javascript/promise)
1818

1919
## Framework & Library & Plugin
2020

21-
- [x] [Babel](./src/bundler/babel/babel.js)
21+
- [x] [Babel](./src/bundler/babel)
2222
- input => [tokenizer](./src/bundler/babel/lib/tokenizer.js) => tokens
23-
- tokens => [parser](./src/bundler/babel/lib/parser.js) => ast
24-
- ast => [transformer](./src/bundler/babel/lib/transformer.js) => newAst
25-
- newAst => [codeGenerator](./src/bundler/babel/lib/codeGenerator.js) => output
26-
- [x] [Webpack](./src/bundler/webpack/minipack.js)
27-
- [x] [Webpack-Plugin](./src/bundler/webpack-plugin/OpenBrowserPlugin.js)
28-
- [x] [Babel-Plugin](./src/bundler/babel-plugin/babel-plugin-simple-import.js)
29-
- [x] [Redux](./src/bundler/redux/redux.js)
30-
- [x] [Router](./src/bundler/router/hashRouter.js)
23+
- tokens => [parser](./src/bundler/babel/lib/parser.js) => AST
24+
- AST => [transformer](./src/bundler/babel/lib/transformer.js) => newAST
25+
- newAST => [codeGenerator](./src/bundler/babel/lib/codeGenerator.js) => output
26+
- [x] [Webpack](./src/bundler/webpack)
27+
- [x] [Webpack-Plugin](./src/bundler/webpack-plugin)
28+
- [x] [Babel-Plugin](./src/bundler/babel-plugin)
29+
- [x] [Redux](./src/bundler/redux)
30+
- [x] [Router](./src/bundler/router)
3131
- [hashRouter](./src/bundler/router/hashRouter.js)
3232
- [historyRouter](./src/bundler/router/historyRouter.js)
3333

3434
## DSA
3535

36-
- array
37-
- [] two sum
38-
- [] three sum
39-
- [] [remove duplicate](./src/dsa/array/unique.js)
40-
- sort
41-
- [] [bubble sort](./src/dsa/sort/bubbleSort.js)
42-
- [] [selection sort](./src/dsa/sort/selectionSort.js)
43-
- [] [insert sort](./src/dsa/sort/insertSort.js)
44-
- [] [merge sort](./src/dsa/sort/mergeSort.js)
45-
- [] [quick sort](./src/dsa/sort/quickSort.js)
46-
- dynamic programming
47-
- [] min distance
48-
- [] min path sum
49-
- tree
50-
- [] binary tree
51-
- [] binary search tree
52-
- [] tree symmetry
53-
- [] tree traversal
54-
- [] tree balanced
55-
- number
56-
- [] thousands format
57-
- string
58-
- [] longest common subsquence
59-
- [] KMP
36+
- [Array](./src/dsa/array)
37+
- [x] [two sum](./src/dsa/array/twoSum.js)
38+
- [x] [three sum](./src/dsa/array/threeSum.js)
39+
- [x] [remove duplicate](./src/dsa/array/unique.js)
40+
- [Sort](./src/dsa/sort)
41+
- [x] [bubble sort](./src/dsa/sort/bubbleSort.js)
42+
- [x] [selection sort](./src/dsa/sort/selectionSort.js)
43+
- [x] [insert sort](./src/dsa/sort/insertSort.js)
44+
- [x] [merge sort](./src/dsa/sort/mergeSort.js)
45+
- [x] [quick sort](./src/dsa/sort/quickSort.js)
46+
- [Tree](./src/dsa/tree)
47+
- [x] [binary tree](./src/dsa/tree/binaryTree.js)
48+
- [x] [binary search tree](./src/dsa/tree/binarySearchTree.js)
49+
- [x] [tree symmetry](./src/dsa/tree/treeSymmetry.js)
50+
- [x] [tree traversal](./src/dsa/tree/treeTraversal.js)
51+
- [Dynamic Programming](./src/dsa/dynamicProgramming)
52+
- [x] [min edit distance](./src/dsa/dynamicProgramming/minEditDistance.js)
53+
- [Number](./src/dsa/number)
54+
- [x] [thousands format](./src/dsa/number/thousands.js)
55+
- [String](./src/dsa/string)
56+
- [x] [longest common subsquence](./src/dsa/string/longestCommonSub.js)
57+
- [x] [KMP](./src/dsa/string/kmp.js)

src/dsa/array/test/threeSum.test.js

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import { threeSum } from '../threeSum';
2+
3+
describe('three sum', () => {
4+
it('no result', () => {
5+
const nums = [1, 2, 3];
6+
const result = [];
7+
expect(threeSum(nums)).toEqual(result);
8+
});
9+
10+
it('has result', () => {
11+
const nums = [1, 2, 3, 0 , -1, -2];
12+
const target = 1;
13+
const result = [ [ -1, 3, -2 ], [ 0, 2, -2 ], [ 0, 1, -1 ] ];
14+
expect(threeSum(nums, target)).toEqual(result);
15+
});
16+
});
17+
18+
describe('three sum, throw error', () => {
19+
it('threeSum, throw error', () => {
20+
const err1 = 'nums must be an array';
21+
expect(() => threeSum()).toThrow(err1);
22+
});
23+
});

src/dsa/array/test/twoSum.test.js

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import { twoSum1, twoSum2 } from './../twoSum';
2+
3+
describe('two sum', () => {
4+
it('has result', () => {
5+
const arr = [1, 2, 3];
6+
const target = 5;
7+
const result = [1, 2];
8+
expect(twoSum1(arr, target)).toEqual(result);
9+
expect(twoSum2(arr, target)).toEqual(result);
10+
});
11+
12+
it('no result', () => {
13+
const arr = [1, 2, 3];
14+
const target = 1;
15+
const result = [];
16+
expect(twoSum1(arr, target)).toEqual(result);
17+
expect(twoSum2(arr, target)).toEqual(result);
18+
});
19+
});
20+
21+
describe('two sum, throw error', () => {
22+
it('twoSum1 and twoSum2, throw error', () => {
23+
const err1 = 'target must be a number';
24+
const err2 = 'arr must be an array';
25+
const err3 = 'arr length must max or equal two';
26+
expect(() => twoSum1()).toThrow(err1);
27+
expect(() => twoSum1(1, 1)).toThrow(err2);
28+
expect(() => twoSum1([], 1)).toThrow(err3);
29+
30+
expect(() => twoSum2()).toThrow(err1);
31+
expect(() => twoSum2(1, 1)).toThrow(err2);
32+
expect(() => twoSum2([], 1)).toThrow(err3);
33+
});
34+
});

src/dsa/array/threeSum.js

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
function threeEqualZero(nums, third, results) {
2+
if (nums.length < 2) {
3+
return;
4+
}
5+
6+
const record = new Set();
7+
let i = 0;
8+
let j = nums.length - 1;
9+
10+
while (i < j) {
11+
const first = nums[i];
12+
const second = nums[j];
13+
14+
if (first + second + third > 0) {
15+
j = j - 1;
16+
}
17+
else if (first + second + third < 0) {
18+
i = i + 1;
19+
}
20+
else {
21+
if (!record.has(first)) {
22+
record.add(first);
23+
results.push([first, second, third]);
24+
}
25+
i = i + 1;
26+
}
27+
}
28+
}
29+
30+
export function threeSum(nums) {
31+
if (!Array.isArray(nums)) {
32+
throw new Error('nums must be an array');
33+
}
34+
35+
if (nums.length < 3) {
36+
return [];
37+
}
38+
39+
nums.sort((a, b) => a - b);
40+
const results = [];
41+
const record = new Set();
42+
43+
for (let i = 0; i < nums.length; i++) {
44+
const third = nums[i];
45+
if (record.has(third)) {
46+
continue;
47+
}
48+
record.add(third);
49+
threeEqualZero(nums.slice(i + 1), third, results);
50+
}
51+
52+
return results;
53+
}

src/dsa/array/twoSum.js

+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
export function twoSum1(arr, target) {
2+
if (typeof target !== 'number') {
3+
throw new Error('target must be a number');
4+
}
5+
6+
if (!Array.isArray(arr)) {
7+
throw new Error('arr must be an array');
8+
}
9+
10+
if (arr.length < 2) {
11+
throw new Error('arr length must max or equal two');
12+
}
13+
14+
const set = new Set(arr);
15+
let index1;
16+
let index2;
17+
18+
for (let i = 0; i < arr.length; i++) {
19+
const number1 = arr[i];
20+
const number2 = target - number1;
21+
22+
if (set.has(number2)) {
23+
const index = arr.indexOf(number2);
24+
if (i === index) {
25+
continue;
26+
}
27+
index1 = i;
28+
index2 = index;
29+
break;
30+
}
31+
}
32+
33+
if (index1 === undefined) {
34+
return [];
35+
}
36+
37+
return [index1, index2];
38+
}
39+
40+
export function twoSum2(arr, target) {
41+
if (typeof target !== 'number') {
42+
throw new Error('target must be a number');
43+
}
44+
45+
if (!Array.isArray(arr)) {
46+
throw new Error('arr must be an array');
47+
}
48+
49+
if (arr.length < 2) {
50+
throw new Error('arr length must max or equal two');
51+
}
52+
53+
const data = {};
54+
for (let i = 0; i < arr.length; i++) {
55+
const num = arr[i];
56+
if (data[target - num] !== undefined) {
57+
return [data[target - num], i];
58+
}
59+
else {
60+
data[num] = i;
61+
}
62+
}
63+
return [];
64+
}
65+

src/dsa/dynamicProgramming/minDistance.js

Whitespace-only changes.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
export default function minEditDistance(str1, str2) {
2+
if (!str1 || !str2) {
3+
return Math.max(str1.length, str2.length);
4+
}
5+
6+
const len1 = str1.length;
7+
const len2 = str2.length;
8+
const matrix = [];
9+
10+
for (let i = 0; i <= len1; i++) {
11+
matrix[i] = [];
12+
13+
for (let j = 0; j <= len2; j++) {
14+
if (i === 0) {
15+
matrix[i][j] = j;
16+
}
17+
else if (j === 0) {
18+
matrix[i][j] = i;
19+
}
20+
else {
21+
matrix[i][j] = Math.min(
22+
matrix[i - 1][j] + 1,
23+
matrix[i][j - 1] + 1,
24+
matrix[i - 1][j - 1] + (str1[i - 1] === str2[j - 1] ? 0 : 1)
25+
);
26+
}
27+
}
28+
}
29+
30+
return matrix[len1][len2];
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import minEditDistance from './../minEditDistance';
2+
3+
describe('minEditDistance', () => {
4+
it('min edit distance, str1 is empty', () => {
5+
const str1 = '';
6+
const str2 = 'abd';
7+
const result = 3;
8+
expect(minEditDistance(str1, str2)).toBe(result);
9+
});
10+
11+
it('min edit distance, str2 is empty', () => {
12+
const str1 = 'abc';
13+
const str2 = '';
14+
const result = 3;
15+
expect(minEditDistance(str1, str2)).toBe(result);
16+
});
17+
18+
it('min edit distance', () => {
19+
const str1 = 'abc';
20+
const str2 = 'cbd';
21+
const result = 2;
22+
expect(minEditDistance(str1, str2)).toBe(result);
23+
});
24+
});

src/dsa/number/test/thousands.test.js

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import { thousands1, thousands2 } from '../thousands';
2+
3+
describe('three sum', () => {
4+
it('no result', () => {
5+
const num = {};
6+
const result = '';
7+
expect(thousands1(num)).toEqual(result);
8+
expect(thousands2(num)).toEqual(result);
9+
});
10+
11+
it('has result', () => {
12+
const num = 1234;
13+
const result = '1,234';
14+
expect(thousands1(num)).toEqual(result);
15+
expect(thousands2(num)).toEqual(result);
16+
});
17+
});

src/dsa/number/thousands.js

+21
Original file line numberDiff line numberDiff line change
@@ -3,3 +3,24 @@
33
* https://www.html.cn/archives/7324
44
* https://www.cnblogs.com/wangmeijian/p/4163936.html
55
*/
6+
7+
export function thousands1(num) {
8+
num = Number(num);
9+
10+
if (!num) {
11+
return '';
12+
}
13+
14+
return num.toLocaleString('zh');
15+
}
16+
17+
export function thousands2(num) {
18+
num = Number(num);
19+
20+
if (!num) {
21+
return '';
22+
}
23+
24+
num = String(num);
25+
return num.replace(/(\d+?)(?=(\d{3})+$)/g, '$1,');
26+
}

0 commit comments

Comments
 (0)