Pairwise Least Squares

From electowiki
Revision as of 15:21, 21 October 2023 by Paretoman (talk | contribs) (Added ballot generation to implementation.)
(diff) ← Older revision | Latest revision (diff) | Newer revision → (diff)

Pairwise least squares is a description for a method that assigns a score to each candidate based on voters' pairwise preferences.

Summary

Least squares estimation is a technique from linear algebra that finds the closest solution to a system of equations.

A pairwise listing of preferences is the number of people who prefer a candidate to another.

Applying least squares to pairwise preferences produces a score for each candidate.

The original preferences are now approximated by taking the difference between any two candidates' scores. The approximation is a projection in the space of pairwise preferences. The projection is the result of subtracting any cyclic group preference component. The projection minimizes the sum of squares of errors in preference numbers.

These scores turn out to be essentially the same as borda scores.

Abstraction

Let the election be modeled as a network.

Let x be a list of scores. Each node in the network is a candidate with a score.

Let A be a list of preferences. Every pair of nodes is connected by two edges. The edges are directed and represent preference for one candidate over another. In network theory, this is an incidence matrix with a row for each edge and a column for each node. For each preference, put a 1 for the “for” candidate and a -1 for the “against” candidate. Leave the rest as 0.

Let b be a tally of preferences. This is how many voters support a preference.

The problem is to find scores x for some given tally of preferences b:

Ax ≟ b.

Approximation

Generally, the problem is unsolvable because there can be cyclic group preferences. Cyclic preferences don’t make sense if the preferences b are computed differences in score, Ax.

However, we can find a solution to a transformed equation:

Aᵀ A x̂ = Aᵀ b.

This new is an approximation of scores that fits the network model..

Solution

There are actually many solutions because the same score could be added to each candidate. So we set the score of the first candidate to 0 in order to solve for one particular solution.

A₀ = A[:,1:]

The solution is:

x̂ = ( A₀ᵀ A₀ )⁻¹ A₀ᵀ b.

Projection

Multiplying by Aᵀ ignores any component of b that is in the null space of Aᵀ. The null space of Aᵀ is composed entirely of cyclic group preferences. This is Aᵀ is a list of edges connected to each node and the null space N(Aᵀ) is such that Aᵀ N(Aᵀ) = 0, which is a statement of conservation of flow. Putting the solution to this transformed equation back into the original equation gives some transformed preference matrix p that has no cycles:

A x̂ = p.

The closest we can get to preferences b using our score model is to project b into the space of possible score model preferences Ax, which does not have any cyclic component. The projection can be represented by the matrix P. The below definition of P gives Pb = p.

P = A₀ ( A₀ᵀ A₀ )⁻¹ A₀ᵀ
Pb = A₀ ( A₀ᵀ A₀ )⁻¹ A₀ᵀ b
Pb = A₀ ( A₀ᵀ A₀ )⁻¹ A₀ᵀ A₀ x̂
Pb = A₀ x̂
Pb = p

Least Square Error

The projected preferences p are the closest to the actual preferences b. In other words, projecting made the difference b - p as small as possible.

p = argmin p ∈ A x̂ || b - p ||

Let’s call this difference the error e.

e = b – p

The error e is a vector. The length of a vector is the square root of the sum of squares of the components. This means projecting also minimizes the sum of the squares of the error for each preference.

p = argmin p ∈ A x̂ || e ||²

Borda

The matrix A₀ᵀ A₀ is 2 ( n I - 1 ). A₀ᵀ b is the sum of preferences for minus preferences against. So, the scores are effectively the net borda scores A₀ᵀ b.

A₀ᵀ b = 2 ( n I - 1 ) x̂

Implementation

from numpy import *

def net(n):
    m = int(n * (n-1) )
    A = mat(zeros((m,n))) # incidence matrix
    e = mat(zeros((m,2))) # edge list
    r = 0
    for i in range(n):
        for k in range(n):
            if i == k:
                continue
            A[r,i] = 1
            A[r,k] = -1
            e[r] = mat([i,k])
            r = r + 1
    A0 = A[:,1:]
    L0 = (A0.T * A0) ** -1 * A0.T
    z = zeros([1,L0.shape[1]])
    L = block([[z],[L0]])
    return [e,A,L]

def edgeFromDense(d,e):
    m = e.shape[0]
    b = mat(zeros((m,1)))
    for i in range(m):
        k0 = int(e[i,0])
        k1 = int(e[i,1])
        b[i,0] = d[k0,k1]
    return b

def denseFromEdge(e,b,n):
    m = e.shape[0]
    d = mat(zeros((n,n)))
    for i in range(m):
        k0 = int(e[i,0])
        k1 = int(e[i,1])
        d[k0,k1] = b[i,0]
    return d

def solveNetwork(A,b,L):
    x = L * b
    p = A * x
    return [x,p]

def solveNetDense(d):
    n = d.shape[0]
    [e,A,L] = net(n)
    b = edgeFromDense(d,e)
    [x,p] = solveNetwork(A,b,L)
    pd = denseFromEdge(e,p,n) 
    pd2 = pd + (d + d.T) / 2
    return [x,pd,pd2]

def denseFromRanking(r):
    n = r.shape[1]
    d = mat(zeros((n,n)))
    for i in range(n):
        for k in range(i):
            rk = r[0,k]
            ri = r[0,i]
            if (ri < rk):
                d[i,k] += 1
            elif (rk < ri):
                d[k,i] += 1
    return d

def denseFromRankingsTallies(rankings,tally):
    n = rankings.shape[1]
    d = mat(zeros((n,n)))
    for i in range(rankings.shape[0]):
        r = rankings[i]
        dense = denseFromRanking(r)
        d += dense * tally[i]
    return d

def test():
    rankings = mat([[1,2,3,4],
                   [3,4,1,2],
                   [4,1,2,3],
                   [3,2,1,4]])
    tally = [2,3,2,3]
    d = denseFromRankingsTallies(rankings,tally)
    [x,pd,pd2] = solveNetDense(d)
    print("Scores")
    print(x)
    print("Borda Scores")
    print(sum(d,1))
    print("Preferences")
    print(d)
    print("Projected Preferences")
    print(pd)
    print("Projected Preferences Plus Pairwise Loops")
    print(pd2)
    print("Higher Order (>2) Loops Removed")
    print(d - pd2)
    print("Borda Scores of Projected Preferences Plus Pairwise Loops")
    print(sum(pd2,1))

test()

Test Output

Scores
[[ 0.]
 [ 1.]
 [ 3.]
 [-1.]]
Borda Scores
[[12.]
 [16.]
 [24.]
 [ 8.]]
Preferences
[[ 0.  5.  2.  5.]
 [ 5.  0.  4.  7.]
 [ 8.  6.  0. 10.]
 [ 5.  3.  0.  0.]]
Projected Preferences
[[ 0. -1. -3.  1.]
 [ 1.  0. -2.  2.]
 [ 3.  2.  0.  4.]
 [-1. -2. -4.  0.]]
Projected Preferences Plus Pairwise Loops
[[0. 4. 2. 6.]
 [6. 0. 3. 7.]
 [8. 7. 0. 9.]
 [4. 3. 1. 0.]]
Higher Order (>2) Loops Removed
[[ 0.  1.  0. -1.]
 [-1.  0.  1.  0.]
 [ 0. -1.  0.  1.]
 [ 1.  0. -1.  0.]]
Borda Scores of Projected Preferences Plus Pairwise Loops
[[12.]
 [16.]
 [24.]
 [ 8.]]

External Links