Skip to content

Latest commit

 

History

History
199 lines (158 loc) · 5.39 KB

template-typescript.md

File metadata and controls

199 lines (158 loc) · 5.39 KB

算法模板(JS/TS 版本)

为了训练自己的 JavaScript/TypeScript 编程能力,能用 TS 写的地方尽量用,不行再换 C++

输入

JS 的输入输出是一个大坑,远不像 C++ 那么方便,Node.js 环境下常用的有 readlineprocess.stdin.on 两种方法。V8 则是 readlineprint

最大的缺点是,一行数据读进来之后,需要自己进行 split 并转换成自己想要的数据类型。代码写起来不够优雅。

推荐使用 process.stdin.on 这种输入方法。

readline 输入

  • 缺点: 输入是被动的(回调函数)。不能主动控制当前的输入进度,比如第一行要输入数字,第二行要输入字符串,那么就需要判断当前输入时第几行,处理起来较为麻烦。
import readline = require('readline')

let countLine = 0
const rl = readline.createInterface({
  input: process.stdin,
  output: process.stdout
})
rl.on('line', function (line) {
  const tokens = line.split(' ')
  console.log(parseInt(tokens[0]) + parseInt(tokens[1]))
  if (++countLine === 10) {
    console.log('当前输入的是第 10 行')
  }
})

process.stdin.on 输入

  • 优点: 通过 data 事件逐渐累积数据,最后到 end 的时候统一处理,看上去像是主动处理所有的数据(但本质上还是回调函数)。
  • 缺点: end 在手动输入的情况下无法触发,因此需要加 SIGINT,利用 Ctrl+C 来触发结束命令。
let input = ''
process.stdin.on('data', function (data) {
  input += String(data)
})

process.stdin.on('end', function () {
  main()
})
process.on('SIGINT', function () {
  main()
})

function main () {
  const inputArray = input.split('\n')
  console.log(inputArray)
}

Update: 有些 OJ (AcWing) 不支持 '/dev/stdin',最简洁的办法就是:

let input = ''
let lines
process.stdin.on('data', data => input += data)
process.stdin.on('end', () => {
  lines = input.split('\n')
  main()
})

function main () {
  console.log(lines)
}

fs 输入

参考 node-scanfnode.js: readSync from stdin?

  • 优点: 输入是主动的(不是回调函数)
  • 缺点: 还是得全部输入之后再做处理(因为不能要求 readFileSync 只读一行,readFileSync 有自己的缓冲机制,你最多只能传一个 buf 给他,但是每次 readFileSync 得到的不一定是一行输入,在文件输入的情况下常常可能是多行,手动输入才可能是一行),使用上和 process.stdin.on 是等价的

node-scanf 是自己实现了一套缓冲和读取逻辑,比较复杂,代码太长了,不适合复制过来使用

import fs = require('fs')
const input = fs.readFileSync('/dev/stdin').toString()
const inputArray = input.split('\n')
console.log(inputArray)

常用函数缩写

const _max = Math.max.bind(Math)
const _min = Math.min.bind(Math)
const _pow = Math.pow.bind(Math)
const _floor = Math.floor.bind(Math)
const _round = Math.round.bind(Math)
const _ceil = Math.ceil.bind(Math)
const log = console.log.bind(console)
// const log = (...args: any[]) => {}

作者:我不吃饼干呀
链接:https://juejin.cn/post/6844903876206805005
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

快速幂

function pow (a: number, b: number, p: number) {
  let ans = 1 % p
  while (b) {
    if (b & 1) ans = Number(BigInt(ans) * BigInt(a) % BigInt(p))
    a = Number(BigInt(a) * BigInt(a) % BigInt(p))
    b >>= 1
  }
  return ans
}

function pow (a: bigint, b: bigint, p: bigint) {
  let ans = 1n % p
  while (b) {
    if (b & 1n) ans = ans * a % p
    a = a * a % p
    b >>= 1n
  }
  return ans
}

邻接表

因为 JS 里面没有静态数组,所以用数组模拟邻接表没有意义,这个方法显得有些鸡肋

const head = new Array<number>(n + 1).fill(0)
const ver = new Array<number>(m + 1).fill(0)
const edge = new Array<number>(m + 1).fill(0)
const next = new Array<number>(m + 1).fill(0)
let tot = 0
function add (x: number, y: number, z: number) {
  ver[++tot] = y
  edge[tot] = z
  next[tot] = head[x]
  head[x] = tot
}

for (const e of edges) {
  add(e[0], e[1], e[2])
  add(e[1], e[0], e[2])
}
for (let i = head[x]; i; i = next[x]) {
  const [y, z]  = [ver[i], edge[i]]
}

快速幂

function qpow (a: bigint, b: bigint, p: bigint) {
  let ans = 1n % p
  while (b) {
    if (b & 1n) ans = ans * a % p
    a = a * a % p
    b >>= 1n
  }
  return ans
}

并查集

const fa = [...Array<number>(n)].map((_, idx) => idx)
function get (x: number): number {
  if (x === fa[x]) return x
  return fa[x] = get(fa[x])
}
function merge (x: number, y: number) {
  const fx = get(x); const fy = get(y)
  fa[fx] = fy
}