-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathalgorithm-bitstring-query-classical.js
77 lines (58 loc) · 1.71 KB
/
algorithm-bitstring-query-classical.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
const logger = require('../src/logger')()
const Bits = require('../src/bits')
// a classical illustration for the Bernstein-Vazirani algorithm
repeat(5, function() {
run()
})
function run() {
let host = new Host()
let oracle = new Oracle({ length: 4 })
let result = host.test(oracle)
logger.log('')
logger.log(`The host detected an oracle value of "${result}".`)
logger.log(`Does the oracle confirm this? ${oracle.confirm(result)}`)
logger.log('')
}
function Host() {
Object.assign(this, {
test: function(oracle) {
this.bits = []
repeat(oracle.length, function(index) {
let bitmask = Bits.fromNumber(0, oracle.length).flip(index).toString()
let result = oracle.query(bitmask)
logger.log(`Querying the oracle with bitmask "${bitmask}" returns: ${result}.`)
this.bits.unshift(result == 0 ? false : true)
}.bind(this))
return Bits.fromArray(this.bits).toString()
}
})
}
function Oracle(options) {
this.length = options && options.length ? options.length : 4
let random = Math.floor(Math.random() * Math.pow(2, this.length))
this.bitstring = Bits.fromNumber(random, this.length).toString()
Object.assign(this, {
query: function(value) {
let input = Bits.fromString(value).toNumber()
let target = Bits.fromString(this.bitstring).toNumber()
return this.matches(input, target)
},
matches: function(input, target) {
let result = 0
let product = input & target
while (product > 0) {
if (product % 2 === 1) result++
product >>= 1
}
return result
},
confirm: function(value) {
return this.bitstring === value ? 'yes' : 'no'
}
})
}
function repeat(number, fn) {
for (let i = 0; i < number; i++) {
fn.apply(this, [i])
}
}