Подтвердить что ты не робот

Члены двух групп разного размера должны встречаться друг с другом (1v1, один раз)

Недавно я боролся с написанием своего рода алгоритма "стиль знакомства с сайтом". Основная цель состоит в том, чтобы каждый член одной группы (Мужчины) встречался с каждым членом другой группы (Женщины) за их столом один раз.

Условия:

  • Количество таблиц равно количеству женщин.
  • Каждому человеку присваивается таблица, где сидит женщина (диалог 1v1).
  • В следующем раунде каждый человек переключается на другую таблицу, в которой он не был раньше.
  • Если группы имеют разный размер, ни один член (мужчина или женщина) не должен находиться в состоянии паузы (отсутствие партнера) в течение двух раундов подряд.

Сложность возникает, когда группа мужчин имеет больше членов, чем группа женщин, или наоборот.

Пример:

var men = [
      'm1', 'm2', 'm3', 'm4', 'm5',
   ],

   women = [
      'w1', 'w2', 'w3'
   ];


┃ ROUND 1                       ┃ ROUND 2
┌─────────────┬─────────────┐   ┌─────────────┬─────────────┐
│     men     │    women    │   │     men     │    women    │
├─────────────┴─────────────┤   ├─────────────┴─────────────┤
│      m1    >>>    w1      │   │      m4    >>>    w1      │
│      m2    >>>    w2      │   │      m5    >>>    w2      │
│      m3    >>>    w3      │   │      m1    >>>    w3      │
│      m4          pause    │   │      m2          pause    │
│      m5          pause    │   │      m3          pause    │
└───────────────────────────┘   └───────────────────────────┘

┃ ROUND 3
┌─────────────┬─────────────┐
│     men     │    women    │
├─────────────┴─────────────┤
│      m2    >>>    w1      │
│      m3    >>>    w2      │
│      m4    >>>    w3      │
│      m5          pause    │
│      m1          pause    │
└───────────────────────────┘

Сопоставлениями являются:

var results = [
    'm1' = [
        'w1', 'w3', null
    ],

    'm2' = [
        'w2', null, 'w1'
    ],

    'm3' = [
        'w3', null, 'w2'
    ],

    'm4' = [
        null, 'w1', 'w3'
    ],

    'm5' = [
        null, 'w2', null
    ],
];

До сих пор мои попытки решить эту проблему были безуспешными, несмотря на то, что они рассматривали подобные вопросы на этом сайте и другие (см. снимок экрана ниже). В этой попытке я побежал 15/15 раундов и все еще получал людей в паузе на 2 или более раунда и т.д.

введите описание изображения здесь * Имена на скриншоте являются фальшивыми; созданный Faker


Моя текущая (нерабочая) попытка в Javascript

В основном я имею четыре массива

  • Массив людей
  • Массив женщин
  • Массив доступных таблиц для раунда
  • Массив неудовлетворенных женщин на человека

var men_array                     = [
  'm1', 'm2', 'm3', 'm4', 'm5', 'm6', 'm7', 'm8', 'm9', 'm10'
];

var women_array                   = [
  'w1', 'w2', 'w3', 'w4', 'w5'
];

var available_tables_this_round   = [];
var unmet_women                   = [];

// Run round
function start_round(){
	console.log('START OF ROUND ----------------');

    // Set available tables this round
    // One table per woman
    available_tables_this_round = women_array;
  
    // Selected table
    var selected_table;
  
    // Finding table partner for each man
    men_array.forEach(function(item){
        var current_man = item;
        
        // Checking if this user has unmet women on record
        if(typeof unmet_women[current_man] == 'undefined'){
            // Unmet women array not set. Setting...
            unmet_women[current_man] = women_array;
        }
      
        // Loop through available tables to see which
        // of those the man can join (has not visited yet).
        for(var i = 0; i < available_tables_this_round.length; i++){
						var current_woman = available_tables_this_round[i];
            
            // If woman among the available has not been met
            if($.inArray(current_woman, available_tables_this_round) !== -1){
              	selected_table = current_woman;
                
                 // Removing table from those available this round
                 available_tables_this_round = $.grep(available_tables_this_round, function(value) {
                  return value != selected_table;  
                });
              
                // Removing woman from unmet by this man
                unmet_women[current_man] = $.grep(unmet_women[current_man], function(value) {
                  return value != selected_table;   
                });
                
                console.log(current_man, '>>>', selected_table);
              
              	// Exiting loop since we've found a match for the man (for this round!)
                break;
            }
        }        
    });
    
    console.log('END OF ROUND ----------------');
  
    
}

// Button handling
$(document).on('click', 'button#start_round_btn', start_round);
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<button id="start_round_btn">Start Round</button>
4b9b3361

Ответ 1

Основная проблема здесь заключается в том, чтобы ни мужчины, ни женщины не ставили паузу дважды подряд.

Чтобы этого не случилось, подумайте о том, чтобы поставить "стоп-столы" между таблицами с женщинами, пока у вас не будет достаточно столов для размещения мужчин. Это означает, что в нумерации таблицы на основе 0 эти "таблицы пауз" получают индекс нечетной позиции.

Когда вы затем зацикливаете мужчин из одной таблицы в другую, никогда не будет человека, который будет дважды помещаться в очередь, но они будут встречаться с каждой женщиной, прежде чем они вернутся к первому столбу присвоено.

С помощью этого метода становится очевидным, что решения проблемы не существует, когда мужчин более чем вдвое больше, чем женщин. В этом случае неизбежно, что человек дважды ставится на паузу.

Аналогичный трюк можно использовать, когда мужчин меньше: женщины никогда не могут быть одинокими в течение более чем одного раунда, поэтому в этом случае вводите манекенов ( "пауза-сиделки" ) с той же методологией: поставьте мужчин в строки и ввести в эту строку этих "манекенов", чтобы они попадали на нечетные индексы. Я называю этот потенциально расширенный ряд "сидерами".

Если вы сделаете это выше, у вас будет столько же сидений, сколько и таблиц, и становится тривиальным для выполнения заданий для каждого раунда: просто зацикливайте сидения на столы: каждый раунд сиденья перемещается к следующей таблице в последовательность, возвращаясь назад к первой таблице после последней.

Вот код:

function getRounds(men, women) {
    // Exclude the cases where a solution is not possible
    if (men.length > women.length * 2) throw "not possible: too many men"; 
    if (women.length > men.length * 2) throw "not possible: too many women"; 
    // Create the tables:
    // Women get a fixed place, so the table name is the woman name
    var tables = women.slice();
    // Create the seaters: they are the men
    var seaters = men.slice();
    // Which do we have fewer of? tables or seaters?
    var maxLen = Math.max(tables.length, seaters.length);
    var least = tables.length < maxLen ? tables : seaters;
    // The smallest of both arrays should get some dummies added to it.
    // We insert the dummies at odd indices: This is the main point of this algorithm
    for (var i = 0; least.length < maxLen; i++) least.splice(i*2+1, 0, 'pause');

    // Now everything is ready to produce the rounds. There are just as many
    // rounds as there are tables (including the "pause tables"):
    return tables.map( (_, round) =>    
        // Assign the men: each round shifted one place to the left (cycling):
        tables.map( (table, tid) => [seaters[(round+tid)%maxLen], table] )
    );
}

var result1 = getRounds(["John", "Tim", "Bob", "Alan", "Fred"], 
                        ["Lucy", "Sarah", "Helen"]);
console.log(result1);

var result2 = getRounds(["John", "Tim"],
                        ["Lucy", "Sarah", "Helen", "Joy"]);
console.log(result2);

Ответ 2

Вы можете заполнить массивы с паузой и сохранить одну и ту же длину массивов. Затем вы можете использовать кросс-произведение массивов со сдвигом по первому массиву и сохранением значения второго массива.

function getSeating(a1, a2) {
    function fill(a, l) {
        var p = l - a.length;
        a = a.slice();
        while (p) {
            a.splice(p, 0, 'pause' + p);
            p--;
        }
        return a;
    }

    var max = Math.max(a2.length, a1.length);

    a1 = fill(a1, max);
    a2 = fill(a2, max);
    return a1.map(function (_, i) {
        return a2.map(function (b, j) {
            return [a1[(i + j) % max], b];
        });
    });
}

console.log(getSeating(['m1', 'm2', 'm3', 'm4', 'm5'], ['w1', 'w2', 'w3']));
console.log(getSeating(['m1', 'm2', 'm3'], ['w1', 'w2', 'w3', 'w4', 'w5']));
console.log(getSeating(['m1', 'm2', 'm3', 'm4', 'm5'], ['w1', 'w2', 'w3', 'w4', 'w5']));

console.log(getSeating(['m1', 'm2', 'm3', 'm4'], ['w1', 'w2']));
console.log(getSeating(['m1', 'm2'], ['w1', 'w2', 'w3', 'w4']));
console.log(getSeating(['m1', 'm2', 'm3', 'm4'], ['w1', 'w2', 'w3', 'w4']));
.as-console-wrapper { max-height: 100% !important; top: 0; }

Ответ 3

Вы можете просто создать таблицы массивов следующим образом:

var tables = men.map((element, index) => index < women.length ? [element, women[index]] : [element, "pause"]);

Чтобы получить:

[ [ 'm1', 'w1' ],
  [ 'm2', 'w2' ],
  [ 'm3', 'w3' ],
  [ 'm4', 'pause' ],
  [ 'm5', 'pause' ] 
]

Следовательно, вам просто нужно перекомбинировать пары для "nouvelles rencontres"

Ответ 4

OK Мне понравился этот вопрос. Мой подход будет обрабатываться до тех пор, пока группа не будет вдвое больше другой. Никто не будет ждать дважды подряд. Я также с удовольствием воспользуюсь своим методом Array.prototype.rotate().

Хорошо, код:

Array.prototype.rotate = function(n){
	var len = this.length,
	    res = new Array(this.length);
	if (n % len === 0) return this.slice(); // no need to rotate just return a copy
	else for (var i = 0; i < len; i++) res[i] = this[(i + (len + n % len)) % len];
	return res;
};

function arrangeTables(matches){
  var meetings = [],
    tableCount = matches[0].length;
  for (var i = 0, len = matches.length; i < len; i++){
    meetings.push(matches.slice(0,tableCount).map((t,ix) => t[ix]));
    matches = matches.concat(matches.splice(0,tableCount).rotate(1));
  }
  return meetings;
}

var men = ['m1', 'm2', 'm3', 'm4', 'm5'],
  women = ['w1', 'w2', 'w3'],
    few = [],
   much = men.length > women.length ? (few = women,men) : (few = men,women), // this is needed for the nesting order of maps to get matches.
matches = much.map(m => few.map(f => [f,m]));
 result = arrangeTables(matches);
console.log(result);