\documentclass{article}


% if you need to pass options to natbib, use, e.g.:
%     \PassOptionsToPackage{numbers, compress}{natbib}
% before loading maeb_2025


% ready for submission
%\usepackage[nonatbib]{maeb_2025}

% to compile a preprint version, e.g., for submission to arXiv, add add the
% [preprint] option:
%     \usepackage[nonatbib,preprint]{maeb_2025}


% to compile a camera-ready version, add the [final] option, e.g.:
    \usepackage[nonatbib,final]{maeb_2025}


% to avoid loading the natbib package, add option nonatbib:
%    \usepackage[nonatbib]{maeb_2025}


\usepackage[utf8]{inputenc} % allow utf-8 input
\usepackage[T1]{fontenc}    % use 8-bit T1 fonts
\usepackage{hyperref}       % hyperlinks
\usepackage{url}            % simple URL typesetting
\usepackage{booktabs}       % professional-quality tables
\usepackage{amsfonts}       % blackboard math symbols
\usepackage{nicefrac}       % compact symbols for 1/2, etc.
\usepackage{microtype}      % microtypography
\usepackage{xcolor}         % colors

%%%%%%%%%%%%%%%%%%%%%
% NUESTROS
%%%%%%%%%%%%%%%%%%%%%
\usepackage{biblatex}
\usepackage{subcaption}
\usepackage{graphicx}

\usepackage{algorithm}
\usepackage{algorithmic}

\addbibresource{refs.bib}

\usepackage[spanish]{babel}
\def\tablename{Tabla}
\addto\captionsspanish{\renewcommand{\tablename}{Tabla}}
\def\figurename{Figura}
\floatname{algorithm}{Algoritmo}
\spanishdecimal{.}

\usepackage{tikz} \usetikzlibrary{calc, arrows.meta, intersections, patterns, positioning, shapes.misc, fadings, through,decorations.pathreplacing}
\usepackage{amsmath,amssymb}
\usepackage{color}
\usepackage{afterpage}
\usepackage{paralist}
\usepackage{array}
\usepackage{enumerate}
\usepackage{paralist}
\usepackage{enumitem}
\usepackage{float}
\usepackage{setspace}
\usepackage{listings}
\usepackage{fancyhdr}
\usepackage{rotating}
\usepackage{multirow}
\usepackage{colortbl}
\usepackage{bm}
\usepackage{pdfpages}


% Otros paquetes

%\usepackage{quotchap}
\usepackage{lipsum}
\usepackage{booktabs}

\definecolor{BlueLink}{rgb}{0.165,0.322,0.745}
\definecolor{PinkLink}{rgb}{0.8,0.22,0.5}
\definecolor{gray}{rgb}{0.6,0.6,0.6}
\definecolor{mycian}{HTML}{139cdb} % Cian
\definecolor{mylightcian}{HTML}{a2e2ff} % Cian claro
\definecolor{mysalmon}{HTML}{ffe7c5} % Salmon
\definecolor{mylightgray}{HTML}{dedede} % Gris claro


\title{GRASP para el problema de fijación de precios basado en preferencias}


% The \author macro works with any number of authors. There are two commands
% used to separate the names and addresses of multiple authors: \And and \AND.
%
% Using \And between authors leaves it to LaTeX to determine where to break the
% lines. Using \AND forces a line break at that point. So, if LaTeX puts 3 of 4
% authors names on the first line, and the last on the second line, try using
% \AND instead of \And before the third author name.


\author{%
  Raúl Fauste-Jiménez
  %\thanks{Use footnote for providing further information about author (webpage, alternative address)---\emph{not} for acknowledging funding agencies.}
  \\
  Dpto. de Informática y Estadística\\
  Universidad Rey Juan Carlos\\
  Móstoles, España \\
  \texttt{r.fauste.2020@alumnos.urjc.es} \\
  \And
  Sergio Salazar
  %\thanks{Use footnote for providing further information about author (webpage, alternative address)---\emph{not} for acknowledging funding agencies.}
  \\
  Dpto. de Informática y Estadística\\
  Universidad Rey Juan Carlos\\
  Móstoles, España \\
  \texttt{sergio.salazar@urjc.es} \\
  \And
  Isaac Lozano-Osorio
  %\thanks{Use footnote for providing further information about author (webpage, alternative address)---\emph{not} for acknowledging funding agencies.}
  \\
  Dpto. de Informática y Estadística\\
  Universidad Rey Juan Carlos\\
  Móstoles, España \\
  \texttt{isaac.lozano@urjc.es} \\
  \And
  Jesús Sánchez-Oro
  %\thanks{Use footnote for providing further information about author (webpage, alternative address)---\emph{not} for acknowledging funding agencies.}
  \\
  Dpto. de Informática y Estadística\\
  Universidad Rey Juan Carlos\\
  Móstoles, España \\
  \texttt{jesus.sanchezoro@urjc.es} \\
}


\begin{document}


\maketitle


\begin{abstract}
%  The abstract paragraph should be indented \nicefrac{1}{2}~inch (3~picas) on  both the left- and right-hand margins. Use 10~point type, with a vertical  spacing (leading) of 11~points.  The word \textbf{Abstract} must be centered, bold, and in point size 12. Two line spaces precede the abstract. The abstract must be limited to one paragraph.

% Definir el problema en el mundo real
Uno de los problemas habituales que sufre todo tipo de comercio es establecer un precio a un producto, un aspecto clave que puede aumentar o disminuir las ventas.
% Establecer el objetivo del problema y lo que nos encontraremos (hablar de productos ilimitados, no empates, etc)
En este trabajo, abordamos el problema de fijación de precios basado en preferencias, donde, dado un conjunto de usuarios con sus preferencias de compra y presupuesto que disponen, el objetivo es establecer los precios de los productos para maximizar el beneficio de la compañía. Cabe destacar que cada usuario siempre comprará el producto de mayor preferencia que pueda permitirse, considerando que los productos son ilimitados.
% Decir que es NP dificil y las aproximaciones recientes
La literatura ha demostrado que este problema es $\mathcal{NP}$-Completo donde los modelos matemáticos actuales no se comportan bien con instancias grandes. Recientemente, se ha propuesto un enfoque metaheurístico basado en la metodología \textit{Variable Neighborhood Search} que se sitúa como el estado del arte. 
% Hablar de lo que hacemos y proponemos (mencionar instancias)
En este artículo, presentamos una metaheurística usando la metodología \textit{Greedy Randomized Adaptive Search Procedure}. Adicionalmente, uno de los principales inconvenientes es el reducido número de instancias disponibles. Por este motivo se ha generado un nuevo conjunto de datos y se han evaluado todos los métodos bajo el mismo entorno computacional. 
% Resultados
Los resultados obtenidos demuestran que nuestra propuesta supera el estado del arte y se sitúa a un 2.85\% del modelo exacto en aquellas instancias donde este logra encontrar la solución óptima.
\end{abstract}


\section{Introducción} \label{sec:intro}

En los últimos años, el rápido crecimiento del comercio electrónico ha propiciado que el problema de fijación de precios de los productos adquiera una importancia crucial por su directa repercusión en el beneficio de las empresas. De esta motivación, surge la definición de los problemas de tarificación.

Este trabajo aborda el \textit{RPP} (del inglés \textit{Rank Pricing Problem}), un problema de optimización que tiene como objetivo fijar los precios de los productos de una compañia para maximizar su beneficio. El \textit{RPP} fue definido por primera vez en \cite{rusmevichientong} donde se demostró ser un problema $\mathcal{NP}$-Completo. Dicho artículo se centra en una variación del \textit{RPP}, se añade una restricción en la escala de precios por la cual dos productos no pueden tener el mismo precio, algo que es de gran utilidad en industrias donde los productos con varias versiones de distinta gama son el modelo de negocio.

En el \textit{RPP} clásico, se conoce, como información del problema, el interés de los clientes en cada producto, de los cuales comprarán a lo sumo uno de ellos. Estos tienen un presupuesto fijo y clasifican los productos en un ranking del más al menos preferido. Una vez fijados los precios, cada cliente compra su producto preferido de entre los que se puede permitir. Se asume una cantidad ilimitada de cada producto. Sin embargo, existen variaciones del problema donde la cantidad de productos es limitada, \textit{Capacitated Rank Pricing Problem (CRPP)} \cite{preprocessTesis}, o donde un cliente puede tener varios productos con el mismo nivel de preferencia \textit{Rank Pricing Problem with Ties (RPPT)} \cite{preprocessTesis}.

La variante clásica del \textit{RPP} y que se estudia en este trabajo, se asocia con industrias donde el gasto por parte del cliente es considerable (automóviles, teléfonos móviles, electrodomésticos, ...) y cada cliente pretende, por norma general, no comprar más de una unidad. Precisamente, fue en la empresa \textit{General Motors}, \cite{rusmevichientong} donde se estudió este problema por primera vez. La empresa utilizaba una página web para recomendar vehículos a sus clientes. Para ello, los clientes escribían el presupuesto con el que contaban y ordenaban las características de los vehículos de acuerdo a la importancia que les otorgaban y sus preferencias. Una vez hecho esto, la web mostraba una lista de vehículos ordenados según sus preferencias y ajustados a su presupuesto.

Estudios posteriores se han centrado tanto en la complejidad del problema como en el desarrollo de métodos no exactos para su resolución, es decir, buscar soluciones de alta calidad que no siempre certifiquen la optimalidad.

En \cite{preprocessPaper} se propone un algoritmo de ramificación y poda, para la resolución del \textit{RPP}. En este artículo, se modela el \textit{RPP} con un problema de optimización uni-nivel, basándose en la prueba de \cite{rusmevichientong}, donde se demuestra que una solución óptima cumple que los precios de los productos pertenecen al conjunto de los presupuestos de los clientes. Posteriormente, realizan una relajación lineal. Para concluir con su propuesta, antes de ejecutar su función de ramificación y poda, ejecutan un preproceso, que consiste en aplicar a la instancia inicial la función $u$. Esta función asigna a los clientes con mayor presupuesto su producto preferido, al resto de clientes su producto preferido dentro de los que no han sido asignados ya a otro cliente con mayor presupuesto. Esto permite partir de una propuesta de solución a la hora de comenzar a evaluar con el algoritmo de ramificación y poda.

Recientemente en \cite{vnsPaper} se ha propuesto el uso de una metaheurística para la resolución del \textit{RPP}. Su propuesta consiste en un \textit{Variable Neighborhood Search (VNS)} \cite{vnsCreadores2,vnsCreadores1}, que se vuelve a aprovechar de la prueba descrita en \cite{rusmevichientong} por la que los precios,  han de pertenecer al conjunto de presupuestos. Este algoritmo necesita tres parámetros, $L_0, Q, T$, que se utilizan en la fase de exploración, para generar vectores de precios aleatorios sobre los que partir, concretamente $L_0$ vectores. Una vez definidos estos puntos de partida, se eligen los $Q$ vectores con el valor de la mejor función objetivo y de esos $Q$, se hace una selección de $T$ vectores aleatorios de los pertenecientes a sus vecindarios. En caso de no mejorar el valor de la función objetivo encontrado hasta el momento, se pasa a la fase de perturbación, donde se expande el radio del vecindario y se repite el proceso. Esto permite considerar opciones totalmente distintas, puesto que un pequeño cambio en el vector de precios, puede significar un cambio importante en el valor de la función objetivo.

Este trabajo presenta una aproximación metaheurística basada en el procedimiento de búsqueda adapativo aleatorio y voraz (\textit{Greedy Randomized Adaptive Search Procedure}) \cite{grasp2} para abordar el \textit{RPP}. En comparación con el estado del arte, la propuesta obtiene mejores resultados en las instancias disponibles, como se analizará en la Sección~\ref{sec:experimentos}.

El resto del artículo se organiza como sigue. En la Sección~\ref{sec:formal} se formaliza el problema. El algoritmo propuesto se describe con detalle en la Sección~\ref{sec:grasp}. La Sección~\ref{sec:experimentos} incluye los resultados obtenidos. Por último, las conclusiones e investigación futura se discuten en la Sección~\ref{sec:conclusion}. 

\section{Descripción formal} \label{sec:formal}

Esta sección presenta una formalización detallada del problema basada en la formulación descrita en el estado del arte \cite{vnsPaper}. Esta versión del \textit{Rank Princing Problem} asume una demanda unitaria, es decir, cada cliente no comprará más de un producto; presupuesto positivo; y acceso a un suministro ilimitado de productos (es decir, en caso de que un cliente quiera comprar un producto, este siempre podrá hacerlo).

Sea $I = \{1, ..., n\}$ el conjunto de productos ofertados y sea $K = \{1, ..., t\}$ el conjunto de clientes a estudio en la instancia del problema. Cada cliente $k \in K$ tendrá asignado por la instancia del problema un valor de preferencia para producto $i \in I$ representado por la variable $p_i^k \in \mathbb{R}$. Dados dos productos $i,i' \in I$, la condición $p_i^k > p_{i'}^k$ implica que el cliente $k$ preferirá comprar el producto $i$ al producto $i'$. Por último, el conjunto $B = \{b^1,...,b^t\}$ denota el presupuesto de los clientes, es decir, el presupuesto $b^k$ representa el precio máximo que puede pagar el cliente $k\in K$. 

El \textit{RPP} consiste en identificar el vector de valores reales $S = (s_1, ..., s_n)$ que denota el precio de cada producto $i \in I$. Cada cliente comprará aquel producto que se pueda permitir, es decir, su precio sea menor o igual que su presupuesto ($s_i \leq b^k$) que maximice su preferencia individual. La función objetivo del problema será maximizar el beneficio obtenido por las compras de los clientes. 


En la Ecuación \eqref{eq:obj} se muestra la formulación binivel del problema \textit{RPP}. Para esta formulación se necesitan definir un vector de variables binarias para cada cliente $x^k = (x^k_1 , ..., x^k_n)$ que denotarán si el cliente $k$ elige comprar el producto $i$.

\begin{equation} \label{eq:obj}
\begin{array}{lll}
\max\limits_{S}
&   \sum\limits_{k\in K} \sum\limits_{i\in I} s_ix_i^k    & \\
\mbox{s.t.}&   &       \\
    &  s_i \geq 0, & \forall i \in I ,
\end{array}
\end{equation}

donde $\forall k \in K$, $x_i^k$ es la solución óptima del problema mostrado en la Ecuación \eqref{eq:obj2}.

\begin{equation} \label{eq:obj2}
\begin{array}{lll}
\max\limits_{x^k} &   \sum\limits_{i\in I} p_i^k x_i^k    & \\
\mbox{s.t.}&   &       \\
    &  \sum\limits_{i\in I}x_i^k \leq 1, & \\
    &  \sum\limits_{i\in I}s_ix_i^k \leq b^k, & \\
    &   x_i^k\in \{0,1\}, & \forall i\in I, \forall k \in K
\end{array}
\end{equation}

La Tabla~\ref{tab:Ejemplo} muestra un ejemplo de instancia para el problema \textit{RPP}. En este caso se trabaja con dos clientes $K=\{A,B\}$ y dos productos distintos $I=\{1,2\}$, aunque el problema asume una numeración de los clientes de $1$ a $n$, para el ejemplo se denotan con letras para mejorar su comprensión. La Tabla~\ref{tab:Ejemplo} muestra en la segunda columna, Presupuesto ($b^k$), el presupuesto de cada uno de los clientes y en las columnas 3 y 4 la preferencia de cada cliente por el producto respectivo.

\begin{table}[H]    
    \caption{Ejemplo de instancia del \textit{RPP} con dos clientes y dos productos. \vspace{3mm}}  
    \label{tab:Ejemplo}
    \centering
    \begin{tabular}{cccc}
        \toprule
           &    &  \multicolumn{2}{c}{Productos ($i$)}    \\ \cmidrule(r){3-4}
         Cliente ($k$) &  Presupuesto ($b^k$) &  Producto 1 ($p_1$) &   Producto 2 ($p_2$) \\ \midrule
         $A$ & 60 & $2$ &  $1$ \\ 
         $B$ & 40 & $2$ &  $1$ \\ 
         \bottomrule
    \end{tabular}
\end{table}

La solución óptima en esta instancia se obtiene asignando al primer producto el precio de 60 unidades ($s_1 = 60$) y al segundo un precio de 40 unidades ($s_2=40$). Con esta asignación el cliente $A$ comprará el primer producto ya que su valor de preferencia es mayor y puede permitirse los dos, aumentando el valor de la función objetivo en 60. Por otro lado, el cliente $B$ no puede permitirse el producto $1$, por lo que compra el producto $2$ que si puede permitirse, aumentando en 40 unidades el valor de la función objetivo. Por lo tanto, el beneficio obtenido por la solución óptima para esta instancia es de 100 unidades.

En la instancia de ejemplo se muestra cómo la solución óptima se obtiene asignando al vector de precios valores dentro del conjunto de presupuestos de los clientes. En \cite{rusmevichientong} se demuestra que esta es una característica necesaria de la solución óptima del problema, es decir, que el dominio de los valores del vector de precios no es continuo, sino que toma valores en el conjunto de presupuestos. De esta forma, el \textit{RPP} toma la definición de problema de optimización combinatoria, y no continua. Bajo esta premisa \cite{vnsPaper} realizan una transformación a una formulación uninivel del problema, se redirige a los autores a ese trabajo para una información más detallada del mismo.


\section{Propuesta algorítmica}
\label{sec:grasp}

La metaheurística \textit{Greedy Randomized Adaptive Search Procedure} (GRASP) se propone por primera vez como solución al problema de cubrimiento de conjuntos en 1989 \cite{grasp1}. Posteriormente, en 1995 adquiere una terminología y estructura de metaheurística de propósito general y se formaliza como tal \cite{grasp2}. Trabajos recientes en problemas de optimización combinatoria han demostrado obtener resultados competitivos \cite{Casado2022,LozanoOsorio2023}.

El algoritmo GRASP es un proceso multiarranque donde en cada iteración, se realizan una  fase de construcción y una fase de mejora, \cite{grasp2, GRASP}. La primera fase, la fase de construcción, busca generar una solución diversa para posteriormente mediante un método de intensificación, la fase de mejora, obtener un máximo local. El Algoritmo~\ref{alg:GRASP} muestra el esquema completo.

\begin{algorithm}[!htp]
\caption{GRASP($\alpha$,I,B)}
\label{alg:GRASP}
\begin{algorithmic}[1]
    \STATE $S_b \gets \emptyset$ \label{alg:GRASP:ini_best}
    \FOR{$i \in 1 \ldots \textit{it}$} \label{alg:GRASP:forini}
        \STATE $S \gets \textit{Constructivo}(\alpha,I,B)$ \label{alg:GRASP:constructivo}
        \STATE $S \gets BúsquedaLocal(S,I)$ \label{alg:GRASP:busqueda}
        \IF{$RPP(S)>RPP(S_b)$} \label{alg:GRASP:ifini}
            \STATE $S_b \gets S$ \label{alg:GRASP:actSol}
        \ENDIF \label{alg:GRASP:ifend}
    \ENDFOR \label{alg:GRASP:forend}
    \RETURN $S_b$ \label{alg:GRASP:retbest}
 \end{algorithmic}
\end{algorithm}

El algoritmo, durante un número prefijado de iteraciones (pasos~\ref{alg:GRASP:forini}-\ref{alg:GRASP:forend}) aplica sucesivamente las fases de construcción (paso~\ref{alg:GRASP:constructivo}) y búsqueda local (paso~\ref{alg:GRASP:busqueda}). Por cada solución generada, si es mejor que la mejor solución encontrada hasta el momento, se actualiza (pasos~\ref{alg:GRASP:ifini}-\ref{alg:GRASP:ifend}), devolviendo la mejor de entre todas las soluciones generadas (paso~\ref{alg:GRASP:retbest}).


% Poner pseudocodigo

\subsection{Fase constructiva}

La fase de construcción en GRASP tiene como objetivo generar una solución factible de alta calidad. La característica principal de esta fase de construcción es la inclusión de aleatoriedad en el proceso para favorecer la diversidad entre las soluciones generadas. Para ello, se utiliza un parámetro $\alpha$ que será el responsable de controlar el grado de aleatoriedad del proceso y debe ajustarse de forma empírica. El Algoritmo~\ref{alg:ConstructivoGRASP} muestra el pseudocódigo del método constructivo propuesto.

\begin{algorithm}[!htp]
\caption{Constructivo($\alpha$,I,B)}
\label{alg:ConstructivoGRASP}
\begin{algorithmic}[1]
    \STATE $\textit{CL} \gets \{(i,b) : \forall i \in I, \forall b \in B\}$  \label{alg:grasp:CL}
    \STATE $S \gets \emptyset$ \label{alg:grasp:S}
    \WHILE{$\textit{CL} \neq \emptyset$} \label{alg:grasp:while}
        \STATE $g_{\max} = \max\limits_{(i,b) \in \textit{CL}} g(S,i,b)  $ \label{alg:grasp:bmax}
        \STATE $g_{\min} = \min\limits_{(i,b) \in \textit{CL}} g(S,i,b)  $ \label{alg:grasp:bmin}
        \STATE $\mu = g_{\max} - \alpha \cdot (g_{\max} - g_{\min})$ \label{alg:grasp:umbral}
        \STATE $\textit{RCL} = \{(i,b) \in \textit{CL} : g(S,i,b) \geq \mu \}$
        \label{alg:grasp:lista}
        \STATE $(i,b) \gets \textit{RND}(\textit{RCL})$        \label{alg:grasp:random}
        \STATE $S_i = b$ \label{alg:grasp:setB}
        \STATE $\textit{CL} \gets \textit{CL} \setminus \{(i,b') : \forall b' \in B\}$ \label{alg:grasp:removeCL}
    \ENDWHILE \label{alg:grasp:endwhile}
    \RETURN $S$ \label{alg:grasp:return}
    
 \end{algorithmic}
\end{algorithm}

El método comienza creando una lista de candidatos \textit{CL} que se compone de todas las posibles asignaciones de precios del producto $i$ con presupuesto $b$ (paso~\ref{alg:grasp:CL}). A continuación, se crea una solución vacía (paso~\ref{alg:grasp:S}) y comienza a iterar hasta completar la solución (pasos~\ref{alg:grasp:while}-\ref{alg:grasp:endwhile}). Cabe destacar que, aunque en la fase de construcción tradicional GRASP el primer elemento se elige de forma aleatoria, en este caso no se produce dicha elección, ya que, de forma experimental, se ha comprobado que se obtienen mejores resultados sin esta elección aleatoria. 

En cada iteración, se obtienen el mejor ($g_{\max}$) y peor ($g_{\min}$) valor de una función voraz que asigna a cada candidato la preferencia a la hora de ser elegido. En este caso, la función voraz utilizada $g$ es directamente el valor de la función objetivo, pero podría reemplazarse por cualquier otro criterio voraz. A continuación, se calcula un umbral $\mu$ (paso~\ref{alg:grasp:umbral}) que determinará qué candidatos alcanzan un mínimo de calidad para formar parte de la lista de candidatos restringida \textit{RCL} (paso~\ref{alg:grasp:lista}). Este umbral es directamente dependiente de un parámetro $\alpha$ que determinará el grado de aleatoriedad del método. En concreto, si $\alpha = 0$, entonces solo aquellos elementos con un valor de función voraz igual a $g_{\max}$ podrán entrar en la \textit{RCL}, convirtiéndose en un procedimiento completamente voraz (salvo empates). Por otra parte, si $\alpha = 1$, todos los elementos de la \textit{CL} podrán entrar a la \textit{RCL} ya que se seleccionan todos los candidatos que alcancen el valor de $g_{\min}$.

Finalmente, el siguiente elemento se elige de manera aleatoria de la \textit{RCL} (paso~\ref{alg:grasp:random}) y se asigna a la solución (paso~\ref{alg:grasp:setB}). La lista de candidatos \textit{CL} debe ser entonces actualizada eliminando todas las tuplas de elementos que contienen el candidato elegido $i$ (paso~\ref{alg:grasp:removeCL}). El método devuelve la solución construida $S$ (paso~\ref{alg:grasp:return}).

Una vez explicada y analizada la fase constructiva del algoritmo GRASP, la segunda fase consiste en la búsqueda local descrita en la Sección~\ref{subsec:localSearch}.

\subsection{Fase de mejora}
\label{subsec:localSearch}
La solución construida en la sección anterior no tiene por qué ser un óptimo local respecto a ninguna vecindad ya que, entre otros aspectos, incluye aleatoriedad en su proceso. Por este motivo, dicha solución se mejora mediante el proceso de búsqueda local descrito en esta sección.

Para definir una búsqueda local es necesario comenzar con el operador de movimiento utilizado. En este trabajo, se propone el uso del operador $\textit{Intercambiar}(S,i,j)$ que intercambiará los precios asignados a los elementos $i,j \in I$. En concreto, dada una solución $S$ compuesta por la asignación:
$$
S = (s_1, \ldots, s_{i-1}, s_{i}, s_{i+1}, \ldots ,s_{j-1}, s_j, s_{j+1}, \ldots s_n)
$$
\noindent el operador $\textit{Intercambiar}(S,i,j)$ intercambiará los valores de los elementos $i,j$, obteniendo la solución:
$$
S  = (s_1, \ldots, s_{i-1}, s_{j}, s_{i+1}, \ldots ,s_{j-1}, s_i, s_{j+1}, \ldots s_n)
$$

Una vez definido el operador de movimiento que se aplicará, es necesario definir la vecindad de soluciones que genera dicho operador. Esta vecindad, denominada $N(S)$, se define como:

$$
N(S) = \{S' : S' = \textit{Intercambiar}(S,i,j) \quad \forall i \neq j \in I\}
$$

Dada la vecindad, es necesario definir cómo se realizará la exploración de la misma, para lo que se propone el Algoritmo~\ref{alg:ls}. Cabe destacar que existen dos estrategias de exploración para la búsqueda local. La primera y conocida como \textit{Best Improvement} se queda con la mejor solución del espacio de vecindad de una solución concreta, $N(S)$, y es la aproximación descrita en el pseudocódigo \ref{alg:ls}. La segunda se conoce como \textit{First Improvement}, modifica la solución en cuanto el algoritmo encuentra una mejor, sin necesidad de explorar todo el vecindario. Su implementación es idéntica al pseudocódigo \ref{alg:ls}, con la única diferencia de que la búsqueda comienza de nuevo tras encontrar una mejora en el paso \ref{alg:ls:actMejora}. Dependiendo del problema de estudio, una aproximación puede proporcionar mejores resultados que otra \cite{localSearch}, por lo que se estudian ambas aproximaciones y se compararán los resultados obtenidos.

\begin{algorithm}[htpb]
\caption{BúsquedaLocal(S,I)}
\label{alg:ls}
\begin{algorithmic}[1]
    \STATE $S' \gets S$ \label{alg:ls:iniSol}
    \STATE $\textit{mejora} \gets \texttt{TRUE}$ \label{alg:ls:iniMejora}
    \WHILE{$\textit{mejora}$} \label{alg:ls:while}
        \STATE $\textit{mejora} \gets \texttt{FALSE}$
        \FOR{$i \in I$} \label{alg:ls:for}
            \FOR{$j \in I \setminus \{i\}$} \label{alg:ls:for2}
                \STATE $\textit{Intercambiar}(S',i,j)$ \label{alg:ls:swap}
                \IF{$\textit{RPP}(S')>\textit{RPP}(S)$}
                    \STATE $S \gets S'$ \label{alg:ls:actSol}
                    \STATE $\textit{mejora} \gets \texttt{TRUE}$ \label{alg:ls:actMejora}
                \ENDIF
                \STATE $\textit{Intercambiar}(S',j,i)$ \label{alg:ls:desSwap}
            \ENDFOR \label{alg:ls:endfor}
        \ENDFOR \label{alg:ls:endfor2}
    \ENDWHILE \label{alg:ls:endwhile}
    \RETURN $S$ \label{alg:ls:return}
 \end{algorithmic}
\end{algorithm}

Analizando el Algoritmo~\ref{alg:ls}, en primer lugar se inicializa la mejor solución, $S_b$ a la solución actual $S$ (paso~\ref{alg:ls:iniSol}). Después, $\forall i \in I$ se realiza un intercambio con otro producto $j \in I : i \neq j$ (paso~\ref{alg:ls:swap}), se evalúa la nueva solución modificada y si dicha solución es mejor que la mejor solución actual, se actualiza (paso~\ref{alg:ls:actSol}). En caso de no mejorar la mejor solución actual, se deshace el intercambio para no continuar a partir de una solución de peor calidad (paso~\ref{alg:ls:desSwap}). Finalmente, se devuelve la mejor solución encontrada durante la búsqueda, $S$ (paso~\ref{alg:ls:return}).


\section{Resultados computacionales} \label{sec:experimentos}

Para realizar los experimentos, se ha utilizado un servidor con un procesador AMD EPYC 7282 (2.8 GHz), 16 cores, 8 GB de memoria RAM, Ubuntu Server 20.04, Java 21 y Gurobi 12.0. 

En estos experimentos se han utilizado las 3 instancias del trabajo previo\footnote{\url{https://github.com/groupoasys/RPP_VNS_data}}. Debido al limitado número de instancias, el conjunto fue ampliado mediante la generación de nuevas instancias de forma aleatoria, considerando como rango de presupuestos $[1, 1000]$, partiendo de un número fijo de: número de clientes $[30, 200]$ y número de productos $[5, 150]$. De esta forma, se generaron 30 nuevas instancias teniendo un total de 33 instancias.

Para poder determinar los mejores valores de los parámetros requeridos por el algoritmo propuesto se ha utilizado \textit{irace} \cite{irace}, el cual automáticamente encuentra la mejor configuración de parámetros para nuestra propuesta. Los parámetros analizados por irace sobre un conjunto preliminar de un 25\% de instancias han sido los siguientes: como criterio de parada, un número de iteraciones entre $[1, 100]$, el valor $\alpha$ utilizado en el constructivo han sido valores en el rango $[0.00, 1.00]$, y finalmente se ha considerado como otro parámetro cada una de las posibles búsquedas locales planteadas.

Los resultados obtenidos por irace sobre un conjunto preliminar de un 20\% de instancias han mostrado que la mejor configuración consta de 87 iteraciones, $\alpha$ de 0,01 y la búsqueda local basada en \textit{Best Improvement}.

Para poder analizar el rendimiento de nuestro algoritmo, se van a mostrar tres tablas para comparar los resultados obtenidos. La Tabla~\ref{tab:FinalVNS} muestra los resultados para las 3 instancias utilizadas en el algoritmo propuesto en el estado del arte. Seguidamente, la Tabla~\ref{tab:exacto} analizará los resultados obtenidos frente a los valores conocidos por el modelo matemático y, finalmente, se mostrará la Tabla~\ref{tab:FinalNoExacto} con los mejores resultados obtenidos en las instancias donde el modelo exacto no es capaz de obtener una solución.

Las métricas que  se incluyen son: F.O., que indica el valor de la función objetivo; Tiempo (s), que indica el tiempo de ejecución en segundos; GAP (\%), que señala la desviación porcentual de la función objetivo frente al valor óptimo de la instancia y \#Óptimo, que muestra si el valor encontrado es óptimo o no. Dichas soluciones óptimas han sido certificadas gracias al modelo matemático implementado.

\begin{table}[htpb]
    \centering
    \caption{Comparativa GRASP frente a VNS.}
    \vspace{0.2 cm}
    \label{tab:FinalVNS}
    \resizebox{\textwidth}{!}{
    \begin{tabular}{l  rrrr  rrrr}
    \toprule
    \multicolumn{1}{l}{} & \multicolumn{4}{c}{\textbf{GRASP}}                               & \multicolumn{4}{c}{\textbf{VNS}}             \\ \cmidrule(r){2-5}\cmidrule(r){6-9}
\textbf{Instancias}  & \textbf{F.O}               & \textbf{Tiempo (s)}         & \textbf{GAP (\%)}            & \textbf{\#Óptimo}        & \textbf{F.O}   & \textbf{Tiempo (s)} & \textbf{GAP (\%)}   & \textbf{\#Óptimo}       \\ \midrule
    30x5                 & 807              & 0,18           & 0,00          & 1         & 807  & 600,00    & 0,00 & 1         \\
    30x25                & 1018             & 1,93           & 2,30          & 0         & 960  & 600,00   & 7,87 & 0         \\
    60x50                & 1962             & 39,30          & 2,73          & 0         & 1842 & 600,00   & 8,68 & 0  \\ \midrule
    \textbf{Total} & 1262,33 & 13,74 & 1,68 & 1 & 1203,00 & 600,00 & 5,52 & 1 \\ \bottomrule
    \end{tabular}
    }
\end{table}

Cabe destacar que en \cite{vnsPaper} no se especifica claramente cuál es el máximo valor de la función objetivo obtenido para cada instancia ni el tiempo de ejecución, ya que únicamente se reporta un informe gráfico. Los resultados que se muestran en la Tabla~\ref{tab:FinalVNS} han sido extraídos tratando de ser lo más precisos posible a partir de las gráficas que aportan en su sección de experimentos. La columna de instancias muestra la cantidad de clientes seguido del total de productos. Cabe destacar la diferencia en tiempo computal de nuestra propuesta frente al VNS (13,74 vs 600,00) con un mejor GAP (1,68 vs 5,52).

La Tabla~\ref{tab:exacto} ilustra los resultados de las instancias en las que Gurobi ha conseguido dar la solución óptima, siendo estas aquellas que cumplen $K \leq 100$ e $I \leq 75$:

\begin{table}[htpb]
    \centering
    \caption{Resultados GRASP con la mejor configuración obtenida con irace: $\alpha = 0.01$ con 87 iteraciones y la búsqueda local \textit{best improvement}.}
    \vspace{0.2 cm}
    \label{tab:exacto}
    \begin{tabular}{lrrrrrr}
    \toprule
    \multicolumn{1}{l}{} & \multicolumn{2}{c}{\textbf{Gurobi}}                                & \multicolumn{4}{c}{\textbf{GRASP}}             \\ \cmidrule(r){2-3} \cmidrule(r){4-7}
    \textbf{Instancias}      & \textbf{F.O}                & \textbf{Tiempo (s)}  & \textbf{F.O}                & \textbf{Tiempo (s)} & \textbf{GAP (\%)}            & \textbf{\#Óptimo}       \\ \midrule
    30x5              & 807 & 0,46 & 807               & 0,09 & 0,00          & 1         \\
    30x25         & 1042 & 0,41    & 1018              & 1.93           & 2,30          & 0         \\
    60x50        & 2017 & 6,71     & 1962              & 39,20 & 2,73          & 0         \\ \midrule
    100x25\_0      & 39893 & 9335,90   & 38806             & 28,66          & 2,72          & 0         \\
    100x25\_1     & 48137 & 13039,21    & 47203             & 26,15          & 1,94          & 0         \\
    100x25\_2    & 47425 & 99261,12     & 46202             & 24,95          & 2,58          & 0         \\
    100x25\_3    & 41237 & 30250,49     & 40674             & 25,93          & 1,37          & 0         \\
    100x25\_4    & 44474 & 9816,64     & 43379             & 25,92        & 2,46        & 0         \\ \midrule
    100x50\_0  & 43340 & 12454,06       & 41685             & 184,03       & 3,82          & 0         \\
    100x50\_1  & 49673 & 9176,31       & 48222             & 159,66         & 2,92          & 0         \\
    100x50\_2  & 50678 & 10658,31       & 49404             & 170,55       & 2,51          & 0         \\
    100x50\_3   & 49302 & 13558,84      & 46767             & 151,76          & 5,14          & 0         \\
    100x50\_4  & 44511 & 13371,61       & 43464             & 173,97          & 2,35         & 0         \\ \midrule
    100x75\_0  & 44462 & 401,09       & 42763             & 576,40         & 3,82          & 0         \\
    100x75\_1  & 52178 & 853,59       & 50578             & 480,49       & 3,07          & 0         \\
    100x75\_2   & 47816 & 1097,51      & 45933             & 512,38         & 3,94          & 0         \\
    100x75\_3   & 46087 & 653,59      & 44083             & 514,62        & 4,35          & 0         \\
    100x75\_4  & 45928 & 438,47       & 44442             & 529,95         & 3,24          & 0         \\ \midrule
    \textbf{Total} & 38833,72 & 12465,24 &  37632,89 & 201,59 & 2,85 & 1 \\ \bottomrule
    \end{tabular}
\end{table}

Como se observa en la Tabla~\ref{tab:exacto} el algoritmo GRASP solo encuentra una solución óptima, sin embargo, la desviación media que ofrece ($2,85\%$) ilustra que en la mayoría de instancias se queda cerca de ella. Por otro lado, el tiempo medio de ejecución del modelo matemático es, 12465,24 segundos que, comparado con los 201,59 segundos de la metaheurística propuesta, refleja la baja escalabilidad de Gurobi para la resolución del problema. En términos de promedio de la función objetivo obtenido por el algoritmo propuesto es de 37632,89 frente a 38833,72 obtenido por Gurobi. 

Para finalizar, la Tabla~\ref{tab:FinalNoExacto} muestra los valores de aquellas instancias para las que el modelo matemático no fue capaz de obtener una solución debido a limitaciones de memoria. Como era previsible, el aumento en clientes y productos se ve afectado por el tiempo medio de ejecución en comparación.


\begin{table}[htpb]
    \centering
    \caption{Resultados GRASP en instancias $K \geq 200$ e $I \geq 50$.}
    \vspace{0.2 cm}
    \label{tab:FinalNoExacto}
    \resizebox{\textwidth}{!}{
    \begin{tabular}{lrrlrrlrr}
    \cmidrule[\heavyrulewidth](r){1-3} \cmidrule[\heavyrulewidth](r){4-6} \cmidrule[\heavyrulewidth](r){7-9}
    \textbf{Instancias} & \textbf{F.O}              & \textbf{Tiempo (s)}  & \textbf{Instancias} & \textbf{F.O}              & \textbf{Tiempo (s)}  & \textbf{Instancias} & \textbf{F.O}              & \textbf{Tiempo (s)}          \\ \cmidrule(r){1-3} \cmidrule(r){4-6} \cmidrule(r){7-9}
    200x50\_0            & 85146           & 678,62     & 200x100\_0           & 90803           & 4929,35  &  200x150\_0 &  93015          &   17039,00        \\     
    200x50\_1            &    85269        &  666,88 &       200x100\_1           &    95345        &  4278,20  &  200x150\_1           &    96212        &   15124,42        \\  
    200x50\_2            &    84620        & 675,62  &       200x100\_2           &    98065        &  4493,50 &     
    200x150\_2           &  102741          &    15087,95       \\  
    200x50\_3            &    88054        &  629,00 &           200x100\_3           & 94881           & 4338,71    &
    200x150\_3           &    97056        &  16426,00          \\ 
    200x50\_4            &   88629    & 624,93  &       200x100\_4           &   85678         & 4672,00   &    
    200x150\_4           &   99797        &    14056,74      \\    \cmidrule[\heavyrulewidth](r){1-3} \cmidrule[\heavyrulewidth](r){4-6} \cmidrule[\heavyrulewidth](r){7-9} 
  
    \end{tabular}}
\end{table}

 

% Frase para acabar cerrando 

En resumen, los experimentos demuestran que el algoritmo propuesto supera al estado del arte, logrando una desviación de un 2,85\% con respecto al método exacto. Además, se presentan resultados para instancias en las que el método exacto no es competitivo, evidenciando la eficiencia y robustez de nuestra propuesta en escenarios más exigentes.

\section{Conclusión} \label{sec:conclusion}

% Resumen de la propuesta (GRASP + instancias + irace)

Este trabajo presenta una solución metaheurística basada en GRASP para el problema \textit{Rank Pricing Problem}. El algoritmo se compone de un método constructivo basado en la función objetivo y una fase de mejora basada en intercambios. Uno de los principales inconvenientes del estado del arte es la limitada cantidad de instancias. En este aspecto, se ha ampliado el conjunto con 30 nuevas instancias de diversos tamaños. Para evaluar los parámetros y las componentes algorítmicas que mejor se adaptan al problema, se ha utilizado irace. Una vez obtenida la mejor configuración, se ha comparado frente a un modelo exacto y al mejor algoritmo conocido en la literatura. 

% Resumen de los resultados
Los resultados muestran cómo frente al estado del arte, nuestra propuesta obtiene mejores resultados en 2 de las 3 instancias. Comparando frente a todas las instancias que resuelve el modelo matemático, nuestro algoritmo se sitúa a un 2,85 de desviación y adicionalmente se han mostrado los resultados para las instancias más grandes.

% Trabajos futuros (comparativa mejor, IA, métodos nuevos o nuevos problemas).

Dentro de los trabajos futuros de este problema, el principal es obtener el código del estado del arte o realizar una implementación justa para poder evaluar frente al resto de instancias. Con el objetivo de demostrar que estas contribuciones son aplicables a problemas relacionados, nace también el interés en abordar alguna otra variante del problema. Finalmente, también se plantea el uso de otras métricas voraces en la fase constructiva o el uso de métodos de aprendizaje supervisado que contribuyan a la toma de decisiones \cite{KarimiMamaghan2022,LozanoOsorio2024}.


\begin{ack}
Los autores agradecen el apoyo del ``Ministerio de Ciencia, Innovación y Universidades (MCIN/AEI/10.13039/501100011033/FEDER, UE) bajo la subvención ref. PID2021-126605NB-I00 y PID2021-125709OA-C22, del ``Ministerio para la Transformación Digital y de la Función Pública'' mediante Concesión TSI-100930-2023-3 y de la Comunidad Autónoma de Madrid, con el proyecto CIRMA-CM (referencia TEC-2024/COM-404).


%Do {\bf not} include this section in the anonymized submission, only in the final paper. You can use the \texttt{ack} environment provided in the style file to automatically hide this section in the anonymized submission.
\end{ack}


%\renewcommand{\bibfont}{\fontsize{9}{9}\selectfont}
\printbibliography



\end{document}