-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrelent.js
executable file
·97 lines (87 loc) · 2.6 KB
/
relent.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
#!/usr/bin/env node
// emacs mode -*-JavaScript-*-
const fs = require('fs'),
getopt = require('node-getopt')
// parse command-line options
const opt = getopt.create([
['n' , 'normalize' , 'normalize distributions'],
['h' , 'help' , 'display this help message']
]) // create Getopt instance
.bindHelp() // bind option 'help' to default action
.parseSystem() // parse command line
const files = opt.argv
if (files.length < 2)
throw new Error ("Please specify at least two distribution files")
const normalize = opt.options.normalize
const distrib = files.map ((filename) => {
let d = fs.readFileSync(filename).toString().split("\n").map ((line) => line.split(" ").map ((x) => parseFloat(x)).filter ((p) => !isNaN(p))).filter ((line) => line.length)
if (normalize) {
const norm = d.reduce ((sum, row) => row.reduce ((sum, p) => sum + p, sum), 0)
d = d.map ((row) => row.map ((p) => p / norm))
}
return d
})
const entropy = (dist) => {
let s = 0
dist.forEach ((row) => {
row.forEach ((p) => {
if (p > 0)
s -= p * Math.log (p)
})
})
return s / Math.log(2)
}
const relent = (dist1, dist2) => {
let d = 0, norm = 0
dist2.forEach ((row, i) => {
row.forEach ((p, j) => {
if (i < dist1.length && j < dist1[i].length)
norm += dist1[i][j]
})
})
dist1.forEach ((row, i) => {
row.forEach ((p, j) => {
if (normalize)
p /= norm
const inRange = i < dist2.length && j < dist2[i].length
if (p > 0 && (inRange || !normalize))
d += p * Math.log (p / (inRange ? dist2[i][j] : 0))
})
})
return d / Math.log(2)
}
const moments = (dist) => {
let ei = 0, ed = 0, ei2 = 0, ed2 = 0, eid = 0, norm = 0
dist.forEach ((row, i) => {
row.forEach ((p, d) => {
ei += i*p
ed += d*p
ei2 += i*i*p
ed2 += d*d*p
eid += i*d*p
})
})
return { ei, ed, ei2, ed2, eid,
vi: ei2 - ei*ei,
vd: ed2 - ed*ed,
cid: eid - ei*ed,
p0: dist[0][0] }
}
files.forEach ((file, i) => {
const m = moments (distrib[i])
console.log ("P0 " + file + " " + m.p0)
console.log ("Ei " + file + " " + m.ei)
console.log ("Ed " + file + " " + m.ed)
console.log ("Vi " + file + " " + m.vi)
console.log ("Vd " + file + " " + m.vd)
console.log ("Cid " + file + " " + m.cid)
})
files.forEach ((file, i) => {
console.log ("S " + file + " " + entropy (distrib[i]))
})
files.forEach ((file1, i) => {
files.forEach ((file2, j) => {
if (i != j)
console.log ("D " + file1 + " " + file2 + " " + relent (distrib[i], distrib[j]))
})
})