Определение комбинаций для получения суммы (Microsoft Excel)
Предположим, у вас есть рабочий лист с тремя столбцами данных. В первом столбце последовательно расположены все буквы алфавита от A до Z.
Второй столбец содержит количество вхождений, которое соответствует букве алфавита. Третий столбец содержит количество часов, которое соответствует букве алфавита.
Что, если вы хотите как можно более равномерно распределить комбинацию букв алфавита на четыре группы на основе третьего столбца (часы)?
Например, если сумма всех часов для каждой буквы алфавита составляет 4000 часов, вы хотите придумать комбинацию, которая разделяла бы алфавит так, чтобы в каждой из четырех групп было около 1000 часов на группу.
На самом деле это хорошо известная проблема в области дискретной математики. Для обеспечения решений было разработано множество алгоритмов, и существуют определенные языки программирования (например, LISP)
которые значительно облегчают создание древовидных структур, которые могут «искать» оптимальные решения.
В этом случае, однако, лучше всего подходит простой подход, включающий использование макроса. Предположим, что у вас есть данные в столбцах от A до C. Следующий макрос проанализирует указанный диапазон и вернет комбинацию значений, которая соответствует вашим требованиям.
Function DoDist(sRaw As Range, _ iTCol As Integer, _ iBuckets As Integer, _ iWanted As Integer, _ iRetCol As Integer) As String Dim lGTotal As Long Dim lPerBucket As Long Dim lCells() As Long Dim sRet() As String Dim lBk() As Long Dim sBk() As String Dim lTemp As Long Dim sTemp As String Dim J As Integer Dim K As Integer Dim L As Integer Application.Volatile ReDim lCells(sRaw.Rows.Count) ReDim sRet(sRaw.Rows.Count) ReDim lBk(iBuckets) ReDim sBk(iBuckets) lGTotal = 0 For J = 1 To sRaw.Rows.Count lCells(J) = sRaw(J, iTCol) lGTotal = lGTotal + lCells(J) sRet(J) = sRaw(J, iRetCol) Next J For J = 1 To sRaw.Rows.Count - 1 For K = J + 1 To sRaw.Rows.Count If lCells(J) < lCells(K) Then lTemp = lCells(J) lCells(J) = lCells(K) lCells(K) = lTemp sTemp = sRet(J) sRet(J) = sRet(K) sRet(K) = sTemp End If Next K Next J lPerBucket = lGTotal / iBuckets For J = 1 To sRaw.Rows.Count L = iBuckets For K = iBuckets To 1 Step -1 If lBk(K) <= lBk(L) Then L = K Next K lBk(L) = lBk(L) + lCells(J) sBk(L) = sBk(L) & sRet(J) & ", " Next J For J = 1 To iBuckets If Right(sBk(J), 2) = ", " Then sBk(J) = Left(sBk(J), Len(sBk(J)) - 2) End If sBk(J) = sBk(J) & " (" & lBk(J) & ")" Next J DoDist = sBk(iWanted) End Function
Обратите внимание, что этой функции передается пять параметров. Первый — это диапазон, который вы хотите оценить, второй — смещение столбца в этом диапазоне, который должен быть суммирован, третий — количество «сегментов», которые вы хотите использовать при оценке, четвертый — это количество ведро, которое вы хотите вернуть, а пятое — смещение столбца (в указанном диапазоне), который содержит значения, которые вы хотите вернуть.
Что делает макрос — это захватывает все значения в столбце, который вы хотите просуммировать, а затем сортирует их в порядке убывания. Эти значения, от наибольшего к наименьшему, затем распределяются между сколь угодно большим количеством «корзин»
Вы указали, что там должно быть. Число всегда добавляется в корзину, содержащую наименьший итог. Строка, возвращаемая функцией, представляет собой возвращаемые значения (все, что находится в каждой ячейке столбца, заданного пятым параметром), и общую сумму корзины.
Например, если вы хотите оценить диапазон A1: C: 26, вы хотите, чтобы распределение основывалось на значениях в третьем столбце диапазона (столбец C), вы хотели, чтобы в анализе было четыре сегмента, вы хотите вернуть третье ведро, и вы хотите, чтобы функция возвращала то, что находится в столбце A диапазона, тогда вы должны использовать следующее для вызова функции:
=DoDist(A1:C26,3,4,3,1)
_Примечание: _
Если вы хотите узнать, как использовать макросы, описанные на этой странице (или на любой другой странице на сайтах ExcelTips), я подготовил специальную страницу, содержащую полезную информацию.
link: / excelribbon-ExcelTipsMacros [Щелкните здесь, чтобы открыть эту специальную страницу в новой вкладке браузера]
.
ExcelTips — ваш источник экономичного обучения Microsoft Excel.
Этот совет (2408) относится к Microsoft Excel 97, 2000, 2002 и 2003. Вы можете найти версию этого совета для ленточного интерфейса Excel (Excel 2007 и более поздних версий) здесь:
link: / excelribbon-Determining_Combinations_to_Make_a_Total [Определение комбинаций для получения общей суммы]
.