Source code for qrisp.qiro.qiroproblems.qiroMaxClique

"""
\********************************************************************************
* Copyright (c) 2024 the Qrisp authors
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the Eclipse
* Public License, v. 2.0 are satisfied: GNU General Public License, version 2
* with the GNU Classpath Exception which is
* available at https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
********************************************************************************/
"""

from qrisp import rz, rzz, x
import numpy as np
import copy
import networkx as nx
from qrisp.algorithms.qiro.qiroproblems.qiro_utils import * 


[docs] def create_max_clique_replacement_routine(res, problem_updated): """ Creates a replacement routine for the problem structure, i.e., defines the replacement rules. See the `original paper <https://journals.aps.org/prxquantum/abstract/10.1103/PRXQuantum.5.020327>`_ for a description of the update rules. Parameters ---------- res : dict Result dictionary of QAOA optimization procedure. problem_updated : List Updates that happened during the QIRO routine. Consits of the updated problem, a list of Qubits which were found to be positively correlated, i.e. part of the problem solution, and a list Qubits which were found to be negatively correlated, i.e. they contradict solution qubits in accordance with the update rules. Returns ------- new_graph : nx.Graph Updated graph for the problem instance. solutions : list Updated set of solutions to the problem. sign : int The sign of the correlation. exclusions : list Updated set of exclusions for the problem. """ graph = problem_updated[0] solutions = problem_updated[1] exclusions = problem_updated[2] orig_edges = [list(item) for item in graph.edges()] orig_nodes = list(graph.nodes()) #get the max_edge and eval the sum and sign max_item, sign = find_max(orig_nodes, orig_edges , res, solutions) if max_item == None: return graph, solutions, 0 ,exclusions new_graph = copy.deepcopy(graph) # we just directly remove vertices from the graph if isinstance(max_item, int): if sign < 0: border = list(graph.adj[max_item].keys()) border.append(max_item) to_remove = [int(item) for item in graph.nodes() if item not in border] new_graph.remove_nodes_from( to_remove) solutions.append(max_item) exclusions += to_remove elif sign > 0: #remove item new_graph.remove_node(max_item) exclusions.append(max_item) else: if sign > 0: #keep the two items in solution and remove all that are not adjacent to both intersect = list(set(list(graph.adj[max_item[0]].keys())) & set(list(graph.adj[max_item[0]].keys()))) intersect.append(max_item[0]) intersect.append(max_item[1]) to_remove = [int(item) for item in graph.nodes() if item not in intersect] new_graph.remove_nodes_from([item for item in graph.nodes() if item not in intersect]) solutions.append(max_item[0]) solutions.append(max_item[1]) exclusions += to_remove elif sign < 0: #remove all that do not border on either! node union = list(graph.adj[max_item[0]].keys()) union += list(graph.adj[max_item[1]].keys()) union.append(max_item[0]) union.append(max_item[1]) to_remove = [int(item) for item in graph.nodes() if item not in union] #to_delete = [item for item in graph.nodes() if item not in union] new_graph.remove_nodes_from(to_remove) exclusions += to_remove return new_graph, solutions, sign, exclusions
[docs] def create_max_clique_cost_operator_reduced(problem_updated): r""" Creates the ``cost_operator`` for the problem instance. This operator is adjusted to consider qubits that were found to be a part of the problem solution. Parameters ---------- problem_updated : List Updates that happened during the QIRO routine. Consits of the updated problem, a list of Qubits which were found to be positively correlated, i.e. part of the problem solution, and a list Qubits which were found to be negatively correlated, i.e. they contradict solution qubits in accordance with the update rules. Returns ------- cost_operator : function A function receiving a :ref:`QuantumVariable` and a real parameter $\gamma$. This function performs the application of the cost operator. """ problem = problem_updated[0] solutions = problem_updated[1] G_compl = nx.complement(problem) def cost_operator(qv, gamma): for pair in list(G_compl.edges()): rzz(3*gamma, qv[pair[0]], qv[pair[1]]) rz(-gamma, qv[pair[0]]) rz(-gamma, qv[pair[1]]) for i in problem.nodes(): if not i in solutions: rz(gamma, qv[i]) return cost_operator