-
Notifications
You must be signed in to change notification settings - Fork 140
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Adding the Max-k-Cut application #299
Changes from 1 commit
3bb2d53
57208a7
4db521a
11d6f96
5fa9126
6cb7536
9fb9458
d457732
6408781
afdabab
2ccae04
63ba6b1
c42e061
82c67bd
6e64655
6bec1f9
4aa90a8
c8a8107
a74c686
323ec1e
980df66
e696d23
647028f
c869735
5cf0eee
2997c45
91f390c
2d1e302
d19b1d1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
* Add new Maxkcut class in qiskit_optimization/applications/max_k_cut.py * Add tests for new Maxkcut class in test/applications/test_max_k_cut.py * Add reference to Maxkcut class in qiskit_optimization/applications/__init__.py file * Mention Maxkcut application in docs/tutorials/09_application_classes.ipynb * Add necessary words in .pylintdict * Add release notes regarding new Maxkcut class in releasenotes/notes/add-max-k-cut-app-7e451a5993171175.yaml Co-authored-by: fedonman <[email protected]>
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,177 @@ | ||
# This code is part of Qiskit. | ||
# | ||
# (C) Copyright IBM 2021. | ||
# | ||
# This code is licensed under the Apache License, Version 2.0. You may | ||
# obtain a copy of this license in the LICENSE.txt file in the root directory | ||
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0. | ||
# | ||
# Any modifications or derivative works of this code must retain this | ||
# copyright notice, and modified files need to carry a notice indicating | ||
# that they have been altered from the originals. | ||
|
||
|
||
"""An application class for the Max-k-cut.""" | ||
|
||
from typing import List, Dict, Tuple, Optional, Union | ||
import networkx as nx | ||
import numpy as np | ||
from matplotlib.pyplot import cm | ||
from matplotlib.colors import to_rgba | ||
from docplex.mp.model import Model | ||
|
||
from qiskit_optimization.algorithms import OptimizationResult | ||
from qiskit_optimization.problems.quadratic_program import QuadraticProgram | ||
from qiskit_optimization.translators import from_docplex_mp | ||
from .graph_optimization_application import GraphOptimizationApplication | ||
|
||
|
||
class Maxkcut(GraphOptimizationApplication): | ||
"""Optimization application for the "max-k-cut" [1] problem based on a NetworkX graph. | ||
|
||
References: | ||
[1]: Z. Tabi et al., | ||
"Quantum Optimization for the Graph Coloring Problem with Space-Efficient Embedding" | ||
2020 IEEE International Conference on Quantum Computing and Engineering (QCE), | ||
2020, pp. 56-62, doi: 10.1109/QCE49297.2020.00018., | ||
https://ieeexplore.ieee.org/document/9259934 | ||
""" | ||
|
||
def __init__( | ||
self, | ||
graph: Union[nx.Graph, np.ndarray, List], | ||
k: int, | ||
colors: Optional[Union[List[str], List[List[int]]]] = None, | ||
) -> None: | ||
""" | ||
Args: | ||
graph: A graph representing a problem. It can be specified directly as a | ||
`NetworkX <https://networkx.org/>`_ graph, | ||
or as an array or list format suitable to build out a NetworkX graph. | ||
k: The number of colors | ||
colors: List of strings or list of colors in rgba lists to be assigned to each | ||
resulting subset, there must be as many colors as the number k | ||
""" | ||
super().__init__(graph=graph) | ||
self._subsets_num = k | ||
self._colors = colors if colors and len(colors) >= k else None | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. For colors I see that more are allowed to be defined and presumably ignored; however if colors were given but less than k then they are discarded. In looking further, from what can see, these are just used if/when the user the draws the result. I am wondering whether it might be better to move this off the constructor and just say have a method call something like In terms of the colors I think it should define what the string and int formats are - if its simply matplot color format then I think that can be stated perhaps with a link to matplot colot doc that has the info plus perhaps an example in the code to help people. Also I wonder since this is a set of colors should they be passed in as a set so that it contains unique items - otherwise a user can specify the same color again. In regards of the behavior or more or less colors given than k maybe you prefer the less severe mechanism of discarding all or part of them depending on size. I think at least the code should emit a warning to alert the user - ideally they should be passing in k colors right not some other sized set. If this is a separate method just to set draw colors then maybe allowing only None (to reset to internal default) or a k sized set and raising a ValueError if the size if not equal to k might be more acceptable? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thank you for your comments, they have already been addressed:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @woodsp-ibm, what do you think of the changes? |
||
|
||
def to_quadratic_program(self) -> QuadraticProgram: | ||
"""Convert a Max-k-cut problem instance into a | ||
:class:`~qiskit_optimization.problems.QuadraticProgram` | ||
|
||
Returns: | ||
The :class:`~qiskit_optimization.problems.QuadraticProgram` created | ||
from the Max-k-cut problem instance. | ||
""" | ||
for w, v in self._graph.edges: | ||
self._graph.edges[w, v].setdefault("weight", 1) | ||
|
||
mdl = Model(name="Max-k-cut") | ||
n = self._graph.number_of_nodes() | ||
k = self._subsets_num | ||
x = {(v, i): mdl.binary_var(name=f"x_{v}_{i}") for v in range(n) for i in range(k)} | ||
first_penalty = mdl.sum_squares((1 - mdl.sum(x[v, i] for i in range(k)) for v in range(n))) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could you make a model with a standard formulation of the problem? Please do not penalize constraints by yourself. |
||
second_penalty = mdl.sum( | ||
mdl.sum(self._graph.edges[v, w]["weight"] * x[v, i] * x[w, i] for i in range(k)) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think this takes sum of edges whose end nodes are same color ( |
||
for v, w in self._graph.edges | ||
) | ||
objective = first_penalty + second_penalty | ||
mdl.minimize(objective) | ||
|
||
op = from_docplex_mp(mdl) | ||
return op | ||
|
||
def interpret(self, result: Union[OptimizationResult, np.ndarray]) -> List[List[int]]: | ||
"""Interpret a result as k lists of node indices | ||
|
||
Args: | ||
result : The calculated result of the problem | ||
|
||
Returns: | ||
k lists of node indices correspond to k node sets for the Max-k-cut | ||
""" | ||
x = self._result_to_x(result) | ||
n = self._graph.number_of_nodes() | ||
cut = [[] for i in range(self._subsets_num)] # type: List[List[int]] | ||
|
||
n_selected = x.reshape((n, self._subsets_num)) | ||
for i in range(n): | ||
node_in_subset = np.where(n_selected[i] == 1)[0] # one-hot encoding | ||
if len(node_in_subset) != 0: | ||
cut[node_in_subset[0]].append(i) | ||
|
||
return cut | ||
|
||
def _draw_result( | ||
self, | ||
result: Union[OptimizationResult, np.ndarray], | ||
pos: Optional[Dict[int, np.ndarray]] = None, | ||
) -> None: | ||
"""Draw the result with colors | ||
|
||
Args: | ||
result : The calculated result for the problem | ||
pos: The positions of nodes | ||
""" | ||
x = self._result_to_x(result) | ||
nx.draw(self._graph, node_color=self._node_color(x), pos=pos, with_labels=True) | ||
|
||
def _node_color(self, x: np.ndarray) -> List[Tuple[float, float, float, float]]: | ||
# Return a list of colors for draw. | ||
|
||
n = self._graph.number_of_nodes() | ||
|
||
# k colors chosen from cm.rainbow, or from given color list | ||
colors = ( | ||
cm.rainbow(np.linspace(0, 1, self._subsets_num)) | ||
if self._colors is None | ||
else self._colors | ||
) | ||
gray = to_rgba("lightgray") | ||
node_colors = [gray for _ in range(n)] | ||
|
||
n_selected = x.reshape((n, self._subsets_num)) | ||
for i in range(n): | ||
node_in_subset = np.where(n_selected[i] == 1) # one-hot encoding | ||
if len(node_in_subset[0]) != 0: | ||
node_colors[i] = to_rgba(colors[node_in_subset[0][0]]) | ||
|
||
return node_colors | ||
|
||
@property | ||
def k(self) -> int: | ||
"""Getter of k | ||
|
||
Returns: | ||
The number of colors | ||
""" | ||
return self._subsets_num | ||
|
||
@k.setter | ||
def k(self, k: int) -> None: | ||
"""Setter of k | ||
|
||
Args: | ||
k: The number of colors | ||
""" | ||
self._subsets_num = k | ||
self._colors = self._colors if self._colors and len(self._colors) >= k else None | ||
|
||
@property | ||
def colors(self) -> Union[List[str], List[List[int]]]: | ||
"""Getter of colors list | ||
|
||
Returns: | ||
The k size color list | ||
""" | ||
return self._colors | ||
|
||
@colors.setter | ||
def colors(self, colors: Union[List[str], List[List[int]]]) -> None: | ||
"""Setter of colors list | ||
|
||
Args: | ||
colors: The k size color list | ||
""" | ||
self._colors = colors if colors and len(colors) >= self._subsets_num else None |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
--- | ||
features: | ||
- | | ||
Adding the Max-k-Cut application :class:`qiskit_optimization.applications.Maxkcut`. | ||
|
||
Problem: Given an undirected graph, find a partition of nodes into at most k subsets such | ||
that the total weight of the edges between the k subsets is maximized. | ||
|
||
To solve this problem, the space-efficient quantum optimization representation (or encoding) | ||
for the graph coloring problem proposed in [1] is used. | ||
|
||
[1]: Z. Tabi et al., "Quantum Optimization for the Graph Coloring Problem with Space-Efficient | ||
Embedding," 2020 IEEE International Conference on Quantum Computing and Engineering (QCE), | ||
2020, pp. 56-62, doi: 10.1109/QCE49297.2020.00018., https://ieeexplore.ieee.org/document/9259934 |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,162 @@ | ||
# This code is part of Qiskit. | ||
# | ||
# (C) Copyright IBM 2021. | ||
# | ||
# This code is licensed under the Apache License, Version 2.0. You may | ||
# obtain a copy of this license in the LICENSE.txt file in the root directory | ||
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0. | ||
# | ||
# Any modifications or derivative works of this code must retain this | ||
# copyright notice, and modified files need to carry a notice indicating | ||
# that they have been altered from the originals. | ||
|
||
""" Test Maxkcut class""" | ||
|
||
from test.optimization_test_case import QiskitOptimizationTestCase | ||
|
||
import networkx as nx | ||
|
||
from qiskit.exceptions import MissingOptionalLibraryError | ||
from qiskit_optimization import QuadraticProgram | ||
from qiskit_optimization.algorithms import OptimizationResult, OptimizationResultStatus | ||
from qiskit_optimization.applications.max_k_cut import Maxkcut | ||
from qiskit_optimization.problems import QuadraticObjective, VarType | ||
|
||
|
||
class TestMaxkcut(QiskitOptimizationTestCase): | ||
"""Test Maxkcut class""" | ||
|
||
def setUp(self): | ||
super().setUp() | ||
self.graph = nx.gnm_random_graph(4, 5, 123) | ||
self.k = 3 | ||
op = QuadraticProgram() | ||
for _ in range(12): | ||
op.binary_var() | ||
self.result = OptimizationResult( | ||
x=[0, 1, 0, 1, 0, 0, 0, 0, 1, 1, 0, 0], | ||
fval=0, | ||
variables=op.variables, | ||
status=OptimizationResultStatus.SUCCESS, | ||
) | ||
|
||
def test_to_quadratic_program(self): | ||
"""Test to_quadratic_program""" | ||
maxkcut = Maxkcut(self.graph, self.k) | ||
op = maxkcut.to_quadratic_program() | ||
# Test name | ||
self.assertEqual(op.name, "Max-k-cut") | ||
# Test variables | ||
self.assertEqual(op.get_num_vars(), 12) | ||
for var in op.variables: | ||
self.assertEqual(var.vartype, VarType.BINARY) | ||
# Test objective | ||
obj = op.objective | ||
self.assertEqual(obj.sense, QuadraticObjective.Sense.MINIMIZE) | ||
self.assertEqual(obj.constant, 4) | ||
self.assertDictEqual( | ||
obj.linear.to_dict(), | ||
{ | ||
0: -2.0, | ||
1: -2.0, | ||
2: -2.0, | ||
3: -2.0, | ||
4: -2.0, | ||
5: -2.0, | ||
6: -2.0, | ||
7: -2.0, | ||
8: -2.0, | ||
9: -2.0, | ||
10: -2.0, | ||
11: -2.0, | ||
}, | ||
) | ||
self.assertDictEqual( | ||
obj.quadratic.to_dict(), | ||
{ | ||
(0, 0): 1.0, | ||
(0, 1): 2.0, | ||
(1, 1): 1.0, | ||
(0, 2): 2.0, | ||
(1, 2): 2.0, | ||
(2, 2): 1.0, | ||
(0, 3): 1.0, | ||
(3, 3): 1.0, | ||
(1, 4): 1.0, | ||
(3, 4): 2.0, | ||
(4, 4): 1.0, | ||
(2, 5): 1.0, | ||
(3, 5): 2.0, | ||
(4, 5): 2.0, | ||
(5, 5): 1.0, | ||
(0, 6): 1.0, | ||
(3, 6): 1.0, | ||
(6, 6): 1.0, | ||
(1, 7): 1.0, | ||
(4, 7): 1.0, | ||
(6, 7): 2.0, | ||
(7, 7): 1.0, | ||
(2, 8): 1.0, | ||
(5, 8): 1.0, | ||
(6, 8): 2.0, | ||
(7, 8): 2.0, | ||
(8, 8): 1.0, | ||
(0, 9): 1.0, | ||
(6, 9): 1.0, | ||
(9, 9): 1.0, | ||
(1, 10): 1.0, | ||
(7, 10): 1.0, | ||
(9, 10): 2.0, | ||
(10, 10): 1.0, | ||
(2, 11): 1.0, | ||
(8, 11): 1.0, | ||
(9, 11): 2.0, | ||
(10, 11): 2.0, | ||
(11, 11): 1.0, | ||
}, | ||
) | ||
# Test constraint | ||
lin = op.linear_constraints | ||
self.assertEqual(len(lin), 0) | ||
|
||
def test_interpret(self): | ||
"""Test interpret""" | ||
maxkcut = Maxkcut(self.graph, self.k) | ||
self.assertEqual(maxkcut.interpret(self.result), [[1, 3], [0], [2]]) | ||
|
||
def test_node_color(self): | ||
"""Test _node_color""" | ||
# default colors | ||
maxkcut = Maxkcut(self.graph, self.k) | ||
self.assertEqual( | ||
[[round(num, 2) for num in i] for i in maxkcut._node_color(self.result.x)], | ||
[ | ||
[0.5, 1.0, 0.7, 1.0], | ||
[0.5, 0.0, 1.0, 1.0], | ||
[1.0, 0.0, 0.0, 1.0], | ||
[0.5, 0.0, 1.0, 1.0], | ||
], | ||
) | ||
# given colors | ||
maxkcut = Maxkcut(self.graph, self.k, colors=["r", "g", "b"]) | ||
self.assertEqual( | ||
[[round(num, 2) for num in i] for i in maxkcut._node_color(self.result.x)], | ||
[ | ||
[0.0, 0.5, 0.0, 1.0], | ||
[1.0, 0.0, 0.0, 1.0], | ||
[0.0, 0.0, 1.0, 1.0], | ||
[1.0, 0.0, 0.0, 1.0], | ||
], | ||
) | ||
|
||
def test_draw(self): | ||
"""Test whether draw raises an error if matplotlib is not installed""" | ||
maxkcut = Maxkcut(self.graph, self.k) | ||
try: | ||
import matplotlib as _ | ||
|
||
maxkcut.draw() | ||
|
||
except ImportError: | ||
with self.assertRaises(MissingOptionalLibraryError): | ||
maxkcut.draw() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This paper addresses a graph coloring problem and does not mention max-k-cut problem.
Could you refer an appropriate paper or page?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @t-imamichi, graph colouring problem is a decision variant of max-k-cut. Graph coloring problem is about checking if there "exists colouring of the nodes which make every edge connecting nodes with a different colour." Contrary, Max-K-Cut is about finding the colouring for which the number of such edges is maximized. You could say that graph colouring to max-K-Cut shows similar relation as SAT to MAX-SAT. For this reason, and the fact max cut is already used in the package we decided to choose max-k-cut name
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As far as the paper mentions only coloring problem, I don't think it's an appropriate citation of max-k-cut. It would confuse users. Why not rename the class "k-coloring problem" as the paper addresses?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Moreover, there are many papers directly addressing max-k-cut problem. Why do you need to refer that paper (Tabi et al.) for max-k-cut class?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The notion of feasibility changes. For Max-Cut every solution is feasible, for graph colouring only those with all edges connecting different nodes are feasible. This is obviously correct but inconsistent with max-cut used in the qiskit-optimization.
Since this is for optimization purposes mostly we found Max-K-Cut more accurate. But we can also move it to k-graph-coloring
We are not aware of older papers that propose the given formulation. If there is one, we're open to replacing the reference.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You mentioned
But, coloring does not take care of edge weights. So, I don't think the optimal solution of coloring does not corresponds to the optimal solution to max-k-cut problem.
You must make a mixed integer programming model of max-k-cut. Here is the definition of k-cut value (though this page addresses minimization)
https://en.wikipedia.org/wiki/Minimum_k-cut
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You may perhaps try to use an solution of coloring problem as a heuristic solution of max-k-cut problem?
If so, I ask you to make a proper mixed integer programming model of max-k-cut problem.
The model of application classes should be exact. Otherwise, even if we apply CPLEX or Gurobi to the model, we may not be able to obtain the optimal solution.