БлогNot. Массив случайных целых чисел без повторений

Массив случайных целых чисел без повторений

Желательно, чтобы числа эти ещё и лежали в заданном диапазоне значений [min,max].

Если нас не особо волнует скорость алгоритма, поступить можно по-простому, как показано в листинге ниже.

Функция getRandomArrayUnique вернёт 0 если массив удалось заполнить или 1, если переданы ошибочные параметры (например, мы требуем 100 различных целых значений, лежащих в диапазоне от 0 до 9).

#include <iostream>
using namespace std;

int getRandomArrayUnique (int size, int *arr, int min, int max) {
 //Генерация массива arr из size случайных целочисленых уникальных значений,
 //принадлежащих интервалу [min,max]
 //return 0 = OK, 1 = error
 int i, j;
 int duplicate;
 int value, diapazone = max-min+1;
 if (diapazone<1 || size<0 || size>diapazone) {
    return 1;
 }
 for ( i= 0; i<size; i++ ) {
  for ( ; ; ) {
   value= min+rand()%diapazone;
   duplicate= 0;
   for ( j= 0; j<i; j++ ) {
    if ( value == arr[j] ) { duplicate= 1; break; }
   }
   if ( !duplicate ) { break; }
  }
  arr[i]= value;
 }
 return 0;
}

int main(void) {
 int size = 100, min = 0, max = 99;
 int *arr = new int [size];
 if (!arr) {
  cout << "No memory!"; return 1;
 }
 int res = getRandomArrayUnique (size,arr,min,max);
 if (res) cout << "Bad data!";
 else for (int i=0; i<size; i++) cout << arr[i] << " ";
 cin.sync(); cin.get(); return 0;
}

Все 100 выведенных программой чисел будут разными, так что ей придётся использовать весь диапазон значений от 0 до 99 включительно.

Если критична скорость выполнения, начинается программистская эзотерика и поджирание дополнительной оперативной памяти. Например, код мог бы быть таким:

#include <iostream>
using namespace std;

int getRandomArrayUnique (int size, int *arr, int min, int max) {
 //Генерация массива arr из size случайных целочисленых уникальных значений,
 //принадлежащих интервалу [min,max]
 //return 0 = OK, 1 = Bad data, 2 = No memory
 int i, size2, index, diapazone;
 diapazone = max-min+1;
 if ( diapazone<1 || size<0 || size>diapazone ) {
  return 1;
 }
 if ( size == 0 ) { return 0; }
 int *arr2 = new int [diapazone];
 if (!arr2) {
  return 2;
 }
 for ( i= 0; i<diapazone; i++ ) {
  arr2[i]= min+i;
 }
 size2= diapazone;
 for ( i= 0; i<size; i++ ) {
  index= rand()%size2;
  arr[i]= arr2[index];
  size2--;
  arr2[index]= arr2[size2];
 }
 if (arr2) delete arr2;
 return 0;
}

int main(void) {
 int size = 100, min = 0, max = 99;
 int *arr = new int [size];
 if (!arr) {
  cout << "No memory!"; return 1;
 }
 int res = getRandomArrayUnique (size,arr,min,max);
 if (res==1) cout << "Bad data!";
 else if (res==2) cout << "No memory!";
 else for (int i=0; i<size; i++) cout << arr[i] << " ";
 cin.sync(); cin.get(); return 0;
}

Следует иметь в виду, что стандартная функция rand() генерирует машинно-зависимую последовательность чисел, то есть, при каждом запуске на конкретном компьютере может генерироваться одна и та же цепочка значений. Поэтому часто применяют функцию srand(), позволяющую получить последовательность на основе некоторого начального значения (ключа). В качестве такого ключа может служить, например, текущее системное время: srand(time(NULL));

Коды из заметки проверялись на оказавшемся под рукой Visual C++ 2010 Express, в 15-м может понадобиться, как всегда, волшебная директива.

08.09.2016, 17:59 [7174 просмотра]


теги: c++ алгоритм random

К этой статье пока нет комментариев, Ваш будет первым