| import numpy as np |
| |
| # Model A only. |
| # Uses least squares regression to find the solution |
| # when there is one unknown variable. |
| def lstsq_solution(A, B): |
| A_inv = np.linalg.pinv(A) |
| x = np.matmul(A_inv, B) |
| return x[0][0] |
| |
| # Model B only. |
| # Uses the pseudoinverse matrix to find the solution |
| # when there are two unknown variables. |
| def pinv_solution(A, mv, B): |
| new_A = np.concatenate((A, mv), axis=1) |
| new_A_inv = np.linalg.pinv(new_A) |
| new_x = np.matmul(new_A_inv, B) |
| print("pinv solution:", new_x[0][0], new_x[1][0]) |
| return (new_x[0][0], new_x[1][0]) |
| |
| # Model A only. |
| # Finds the coefficient to multiply A by to minimize |
| # the percentage error between A and B. |
| def minimize_percentage_error_model_a(A, B): |
| R = np.divide(A, B) |
| num = 0 |
| den = 0 |
| best_x = 0 |
| best_error = 100 |
| for r_i in R: |
| num += r_i |
| den += r_i**2 |
| if den == 0: |
| return 0 |
| return (num/den)[0] |
| |
| # Model B only. |
| # Finds the coefficients to multiply to the frame bitrate |
| # and the motion vector bitrate to minimize the percent error. |
| def minimize_percentage_error_model_b(r_e, r_m, r_f): |
| r_ef = np.divide(r_e, r_f) |
| r_mf = np.divide(r_m, r_f) |
| sum_ef = np.sum(r_ef) |
| sum_ef_sq = np.sum(np.square(r_ef)) |
| sum_mf = np.sum(r_mf) |
| sum_mf_sq = np.sum(np.square(r_mf)) |
| sum_ef_mf = np.sum(np.multiply(r_ef, r_mf)) |
| # Divides x by y. If y is zero, returns 0. |
| divide = lambda x, y : 0 if y == 0 else x / y |
| # Set up and solve the matrix equation |
| A = np.array([[1, divide(sum_ef_mf, sum_ef_sq)],[divide(sum_ef_mf, sum_mf_sq), 1]]) |
| B = np.array([divide(sum_ef, sum_ef_sq), divide(sum_mf, sum_mf_sq)]) |
| A_inv = np.linalg.pinv(A) |
| x = np.matmul(A_inv, B) |
| return x |
| |
| # Model A only. |
| # Calculates the least squares error between A and B |
| # using coefficients in X. |
| def average_lstsq_error(A, B, x): |
| error = 0 |
| n = 0 |
| for i, a in enumerate(A): |
| a = a[0] |
| b = B[i][0] |
| if b == 0: |
| continue |
| n += 1 |
| error += (b - x*a)**2 |
| if n == 0: |
| return None |
| error /= n |
| return error |
| |
| # Model A only. |
| # Calculates the average percentage error between A and B. |
| def average_percent_error_model_a(A, B, x): |
| error = 0 |
| n = 0 |
| for i, a in enumerate(A): |
| a = a[0] |
| b = B[i][0] |
| if b == 0: |
| continue |
| n += 1 |
| error_i = (abs(x*a-b)/b)*100 |
| error += error_i |
| error /= n |
| return error |
| |
| # Model B only. |
| # Calculates the average percentage error between A and B. |
| def average_percent_error_model_b(A, M, B, x): |
| error = 0 |
| for i, a in enumerate(A): |
| a = a[0] |
| mv = M[i] |
| b = B[i][0] |
| if b == 0: |
| continue |
| estimate = x[0]*a |
| estimate += x[1]*mv |
| error += abs(estimate - b) / b |
| error *= 100 |
| error /= A.shape[0] |
| return error |
| |
| def average_squared_error_model_a(A, B, x): |
| error = 0 |
| n = 0 |
| for i, a in enumerate(A): |
| a = a[0] |
| b = B[i][0] |
| if b == 0: |
| continue |
| n += 1 |
| error_i = (1 - x*(a/b))**2 |
| error += error_i |
| error /= n |
| error = error**0.5 |
| return error * 100 |
| |
| def average_squared_error_model_b(A, M, B, x): |
| error = 0 |
| n = 0 |
| for i, a in enumerate(A): |
| a = a[0] |
| b = B[i][0] |
| mv = M[i] |
| if b == 0: |
| continue |
| n += 1 |
| error_i = 1 - ((x[0]*a + x[1]*mv)/b) |
| error_i = error_i**2 |
| error += error_i |
| error /= n |
| error = error**0.5 |
| return error * 100 |
| |
| # Traverses the data and prints out one value for |
| # each update type. |
| def print_solutions(file_path): |
| data = np.genfromtxt(file_path, delimiter="\t") |
| prev_update = 0 |
| split_list_indices = list() |
| for i, val in enumerate(data): |
| if prev_update != val[3]: |
| split_list_indices.append(i) |
| prev_update = val[3] |
| split = np.split(data, split_list_indices) |
| for array in split: |
| A, mv, B, update = np.hsplit(array, 4) |
| z = np.where(B == 0)[0] |
| r_e = np.delete(A, z, axis=0) |
| r_m = np.delete(mv, z, axis=0) |
| r_f = np.delete(B, z, axis=0) |
| A = r_e |
| mv = r_m |
| B = r_f |
| all_zeros = not A.any() |
| if all_zeros: |
| continue |
| print("update type:", update[0][0]) |
| x_ls = lstsq_solution(A, B) |
| x_a = minimize_percentage_error_model_a(A, B) |
| x_b = minimize_percentage_error_model_b(A, mv, B) |
| percent_error_a = average_percent_error_model_a(A, B, x_a) |
| percent_error_b = average_percent_error_model_b(A, mv, B, x_b)[0] |
| baseline_percent_error_a = average_percent_error_model_a(A, B, 1) |
| baseline_percent_error_b = average_percent_error_model_b(A, mv, B, [1, 1])[0] |
| |
| squared_error_a = average_squared_error_model_a(A, B, x_a) |
| squared_error_b = average_squared_error_model_b(A, mv, B, x_b)[0] |
| baseline_squared_error_a = average_squared_error_model_a(A, B, 1) |
| baseline_squared_error_b = average_squared_error_model_b(A, mv, B, [1, 1])[0] |
| |
| print("model,\tframe_coeff,\tmv_coeff,\terror,\tbaseline_error") |
| print("Model A %_error,\t" + str(x_a) + ",\t" + str(0) + ",\t" + str(percent_error_a) + ",\t" + str(baseline_percent_error_a)) |
| print("Model A sq_error,\t" + str(x_a) + ",\t" + str(0) + ",\t" + str(squared_error_a) + ",\t" + str(baseline_squared_error_a)) |
| print("Model B %_error,\t" + str(x_b[0]) + ",\t" + str(x_b[1]) + ",\t" + str(percent_error_b) + ",\t" + str(baseline_percent_error_b)) |
| print("Model B sq_error,\t" + str(x_b[0]) + ",\t" + str(x_b[1]) + ",\t" + str(squared_error_b) + ",\t" + str(baseline_squared_error_b)) |
| print() |
| |
| if __name__ == "__main__": |
| print_solutions("data2/all_lowres_target_lt600_data.txt") |