Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.88/8: Рейтинг темы: голосов - 8, средняя оценка - 4.88
2 / 2 / 1
Регистрация: 23.11.2011
Сообщений: 87

Связь списков

27.03.2013, 18:29. Показов 2088. Ответов 22
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго времени суток,

В процессе решения задачи, встретилась проблема:
есть структура
C++
1
2
3
4
struct TStruct {
std::string * m_String_1;
std::string * m_String_2;
}
есть массивы:
C++
1
2
3
4
std::string ** arr_Type_1_Strings;
unsigned int Type1StringsCount;
std::string ** arr_Type_2_Strings;
unsigned int Type2StringsCount;

Что я делал:
Идет запрос на добавление структуры (с указанием значений для ее элементов), тогда проверяется есть ли в
C++
1
arr_Type_1_Strings
строка соответствующая первому значению и то же самое для второго значния из запроса, если оба есть (а спец. функция, которая проверяет наличие, на самом деле, просто возвращает указатель на соответсвующий элемент соответствующего массива ), то функция добавления возвращает ложь, иначе создается необходимая структура
C++
1
 (TStruct *tmpStruct=new TStruct)
отсутствующие строки создаются или просто записываются указатели на них в случае если эти указатели есть, т.е.
C++
1
tmpStruct->m_String_X = указатель на элемент массива где находится добавляемая строка.
Создаются эти строки (если в списке строк запрашиваемая не была найдена) так:
в массиве выделяется память под указатель
C++
1
 (std::string**)malloc(...)
, потом создается новая строка
C++
1
arr_Type_X_Strings[number]=new std::string;
потом начинается самое интересное, методом деления массива пополам, находится индекс элемента (@index) на месте которого должна быть новая строка @Str2Add (дабы сохранять порядок в массиве).

Далее в цикле (если конечно индекс не больше числа элементов в массиве до дополнения, иначе после выделения памяти элемент запишется в последнюю ячейку):

C++
1
2
3
for(unsigned int i=Type1StringsCount;i>@index;i--){
     *arr_Type_X_Strings[i]=*arr_Type_X_Strings[i-1];
}
и наконец
C++
1
arr_Type_X_Strings[@index]=@Str2Add;
и возвращается указатель на новую строку.


Это лишь малая часть программы, пока я писал остальные части я проводил тестирование (совпало) на строках идущих по порядку, соответсвенно никаких повторяющихся данных (типа std::string во всяком случае) и все в нужном порядке, но когда написав много кода, я ввел строки идущие не попорядку я удивился выводу, содержимое структур не соответсвовало ожиданиям, оно и понятно вот здесь:
C++
1
*arr_Type_X_Strings[i]=*arr_Type_X_Strings[i-1];
менялись не указатели а сами строки (вроде так да?) а точнее они смещались, что вроде бы я и хотел, вот только двойной массив я использовал, как мне казалось, чтобы данные в структурах не съехали (потом у каждой структуры другие значения выводятся, т.к. сместилось соответствие), сейчас я понимаю что запутался просто в указателях, однако сейчас мне в голову приходят только такие решения - при каждом сдвиге части массива, проходит в цикле ВСЕ (много времени процессора) структуры и изменять там указатель на новое значение позиции строки в ее массиве, либо сделать подобие базы данных с тремя столбцами - первый столбец это не изменяемый идентификатор строки (указатель этого массива бд на который ссылается элемент структуры) второй столбец это адрес элемента в промежуточном массиве, третий столбец это физический адрес строки. Хотя нет опять ерунда какая то получается, может надо использовать два массива, в одном указано соответствие физического и порядкого в другом запрашиваемого и порядкового, и при добавлении строки менять указатели в промежуточном массиве, сохраняя таким образом значение для элемента структуры. Как видите я никак не осилю эта задачу, варианты с линейными наивными решениями (разумеется с теми, которые неоправданно используют память и процессорное время) не катят, если каждая строка по 1000 символов и у вас 8000 строк, то копировать их туда сюда целиком вместе со структурой не лучший выход поэтому массивы.

Что можете предложить?

Добавлено через 6 минут
P.S. доступные библиотеки :
C++
1
2
3
4
5
6
#include <iostream>
#include <iomanip>
#include <string>
#include <cstring>
#include <cstdlib>
#include <cstdio>
так что, к сожалению без векторов и других радостей STL, в задании проверяется умение решать задачи без STL.
0
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
27.03.2013, 18:29
Ответы с готовыми решениями:

Связь полей из 2(3) списков
Суть проблемы такова: структура - отделения учащиеся тренера и группы. 4 таблицы, остальные три пляшут от таблицы учащиеся. Нужно...

Связь выпадающих списков и календаря на форме
Есть форма с 2 выпадающими списками(ComboBox)- в одном-список дней месяца(1-31), в другом-названия месяцев(январь-декабрь(укр.))....

Связь выпадающих списков в одной форме
Пoмогите создать связь между двумя выпадающими списками в форме &quot;План занятий&quot;. 1ый список задаем имя преподавателя, а во втором...

22
2 / 2 / 1
Регистрация: 23.11.2011
Сообщений: 87
30.03.2013, 06:44  [ТС]
Студворк — интернет-сервис помощи студентам
Можно пример двоичного поиска для двумерного массива, т.е.

a a
b b
c b
d b
e b
h b
h g
h h
h i
j z

в этом массиве для ключей h,b поиск вернет истину (т.е. найдены) для ключей h,a вернет ложь(нет совпадений) но укажет куда вставить новый элемент чтобы не нарушить порядок, сейчас у меня 4 функции бинарного поиска, вторая зависит от первой, их рещультаты считаются в третьей (которую изначально вызывают), за несколько часов я окончательно запутался(для некоторых пар работает, для других нет (работает моя карусель только если совпадения/несовпадения расположены в середине массива, ближе к краю начинаются ошибки, не работает для минимума - поправлю и не работает для максимума)), если у кого-нибудь есть под руками подходящий пример скиньте.
0
 Аватар для lemegeton
4903 / 2696 / 921
Регистрация: 29.11.2010
Сообщений: 5,783
30.03.2013, 21:42
Да откуда у вас двумерные массивы взялись-то?! Какая логика при декомпозиции задачи может породить такие структуры?!

Ход размышлений:

Постановка:
Организовать реестр фирм, где пара имя владельца - его адрес уникальные, фирмы напротив могут иметь дубликаты - одна фирма может иметь несколько хозяев.

Разбиваем на сущности:
1. Сущность, являющаяся "парой имя владельца и его адрес".
2. Хранилище этих сущностей, поддерживающее уникальность каждой сущности.
3. Сущность "фирма", поля не заданы, уникальность не требуется.
4. Хранилище сущности "фирма" с дубликатами.
5. Сущность, хранящая связь фирма - владелец, уникальность не требуется. "Владелец фирмы".
6. Хранилище сущностей связи фирм и владельцев, поддерживающее уникальность каждой сущности.

Сущности 1, 3 и 5 по сути различные простые структуры.
Сущности 2, 4 и 6 практически одинаковы, поэтому могут быть фасадом для некой вспомогательной сущности.
Тогда:
7. Вспомогательная сущность -- обобщенное хранилище сущностей.

Реализуем:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
#include <ctime>
#include <cstdlib>
#include <iostream>
#include <string>
 
// ñóùíîñòü "âëàäåëåö"
class Owner {
 public:
  Owner(const std::string &name, const std::string &address)
    : name(name), address(address) {}
  const std::string &getName() const { return name; }
  const std::string &getAddress() const { return address; }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  Owner(const Owner&);
  Owner &operator=(const Owner&);
  const std::string name;
  const std::string address;
};
 
// ñóùíîñòü "êîìïàíèÿ"
class Company {
 public:
  Company(const std::string &name) : name(name) {}
  const std::string &getName() const { return name; }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  Company(const Company&);
  Company &operator=(const Company&);
  const std::string name;
};
 
// ñóùíîñòü "âëàäåëåö ôèðìû"
class CompanyOwner {
 public:
  CompanyOwner(const Company &company, const Owner &owner)
    : company(company), owner(owner) {}
  const Company &getCompany() const { return company; }
  const Owner &getOwner() const { return owner; }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  CompanyOwner(const CompanyOwner&);
  CompanyOwner &operator=(const CompanyOwner&);
  const Company &company;
  const Owner &owner;
};
 
// ïîíàäîáÿòñÿ îïåðàòîðû âûâîäà, ÷òîáû óäîáíî âûâîäèòü ñóùíîñòè
std::ostream &operator<<(std::ostream &stream, const Owner &owner) {
  return stream << "Owner{" <<
    "name='" << owner.getName() << "'," <<
    "address='" << owner.getAddress() << "'}";
}
std::ostream &operator<<(std::ostream &stream, const Company &company) {
  return stream << "Company{" <<
    "name='" << company.getName() << "'}";
}
std::ostream &operator<<(std::ostream &stream, const CompanyOwner &companyOwner) {
  return stream << "CompanyOwner{" <<
    "company=" << companyOwner.getCompany() << "," <<
    "owner=" << companyOwner.getOwner() << "}";
}
 
// äëÿ ïîèñêîâûõ àëãîðèòìîâ ïîíàäîáÿòñÿ îïåðàòîðû ñðàâíåíèÿ.
// äîñòàòî÷íî îïåðàöèè "ìåíüøå"
bool operator<(const Owner &a, const Owner &b) {
  return (a.getName() < b.getName()) ||
    (!(b.getName() < a.getName()) && a.getAddress() < b.getAddress());
}
bool operator<(const Company &a, const Company &b) {
  return (a.getName() < b.getName());
}
bool operator<(const CompanyOwner &a, const CompanyOwner &b) {
  if (&a.getCompany() != &b.getCompany()) {
    return true;
  }
  return (a.getCompany() < b.getCompany()) || 
    (!(b.getCompany() < a.getCompany()) && a.getOwner() < b.getOwner());
}
 
// ôóíêöèÿ äëÿ ñðàâíåíèÿ çíà÷åíèé, ïåðåäàííûõ ÷åðåç òèïèçèîâàííûé óêàçàòåëü
template<class T>
bool lesserByPointer(const T *a, const T *b) {
  return *a < *b;
}
 
// îáîáùåííûé áèíàðíûé ïîèñê
template <class ForwardIterator, class T, class Compare>
ForwardIterator lowerBound (ForwardIterator first, ForwardIterator last,
  const T& val, Compare comp) {
  size_t count = last - first;
  while (count>0)
  {
    ForwardIterator it = first;
    size_t step=count/2;
    it += step;
    if (comp(*it, val)) {
      first = ++it;
      count -= step + 1;
    } else {
      count = step;
    }
  }
  return first;
}
 
// âñïîìîãàòåëüíàÿ ñóùíîñòü, ïîçâîëÿþùàÿ õðàíèòü òèïèçèðîâàííûå óêàçàòåëè
template <class T, bool repetitionsAllowed = false>
class Storage {
 public:
  Storage() : size(0), capacity(1000), capacityIncrement(100),
    data(new T*[capacity]) {}
  virtual ~Storage() {
    clear();
    delete [] data;
  }
  const T *add(T *value) {
    checkCapacity();
    T **position = lowerBound(data, data + size, value, lesserByPointer<T>);
    if (!repetitionsAllowed && position < end() && !(*value < **position)) {
      return *position;
    } else {
      for (T **i = data + size; i > position; --i) {
        *i = *(i - 1);
      }
      *position = value;
      ++size;
    }
    return *position;
  }
  T *remove(const T *value) {
    T **position = lowerBound(data, data + size, value, lesserByPointer<T>);
    if (position < end() && !(*value < **position)) {
      T *result = *position;
      for (T **i = position; i < end() - 1; ++i) {
        *i = *(i + 1);
      }
      --size;
      return result;
    } else {
      return 0;
    }
  }
  const T *operator[](size_t i) const {
    return data[i];
  }
  T * const *find(T *value) const {
    T **position = lowerBound(data, data + size, value, lesserByPointer<T>);
    if (position < end() && !(*value < **position)) {
      return position;
    } else {
      return end();
    }
  }
  size_t getCapacityIncrement() const { return capacityIncrement; }
  void setCapacityIncrement(size_t capacityIncrement) {
    this->capacityIncrement = capacityIncrement;
  }
  size_t getSize() const { return size; }
  void clear() {
    size = 0;
  }
  T * const *begin() const { return data; }
  T * const *end() const { return data + size; }
 protected:
  void checkCapacity() {
    if (capacity <= size - 1) {
      capacity += getCapacityIncrement();
      T **newData = new T*[capacity];
      for (int i = 0; i < size; ++i) {
        newData[i] = data[i];
      }
      delete [] data;
      data = newData;
    }
  }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  Storage(const Storage&);
  Storage &operator=(const Storage&);
  size_t size;
  size_t capacity;
  size_t capacityIncrement;
  T **data;
};
 
// ñóùíîñòü, õðàíÿùàÿ âëàäåëüöåâ
class OwnerSource {
 public:
  OwnerSource() : storage() {}
  virtual ~OwnerSource() {
    for (Owner * const *i = storage.begin(); i != storage.end(); ++i) {
      delete *i;
    }
    storage.clear();
  }
  bool add(const std::string &name, const std::string &address) {
    Owner *owner = new Owner(name, address);
    const Owner *result = storage.add(owner);
    if (result != owner) {
      delete owner;
      return false;
    } else {
      return true;
    }
  }
  bool remove(const std::string &name, const std::string &address) {
    Owner owner(name, address);
    const Owner *removed = storage.remove(&owner);    
    bool result = removed != 0;
    delete removed;
    return result;
  }
  bool contains(const std::string &name, const std::string &address) const {
    Owner object(name, address);
    return storage.find(&object) != storage.end();
  }
  size_t getSize() const {
    return storage.getSize();
  }
  const Owner &operator[](size_t position) const {
    return *storage[position];
  }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  OwnerSource(const OwnerSource&);
  OwnerSource &operator=(const OwnerSource&);
  Storage<Owner, false> storage;
};
 
// ñóùíîñòü, õðàíÿùàÿ êîìïàíèè
class CompanySource {
 public:
  CompanySource() : storage() {}
  virtual ~CompanySource() {
    for (Company * const *i = storage.begin(); i != storage.end(); ++i) {
      delete *i;
    }
    storage.clear();
  }
  bool add(const std::string &name) {
    Company *object = new Company(name);
    const Company *result = storage.add(object);
    if (result != object) {
      delete object;
      return false;
    } else {
      return true;
    }
  }
  // ïîñêîëüó â êîìïàíèÿõ âîçìîæíû ïîâòîðåíèÿ,
  // ýëåìåíò óäàëÿåòñÿ ïî èíäåêñó
  void remove(int i) {
    Company *removed = storage.remove(storage[i]);
    delete removed;
  }
  size_t getSize() const {
    return storage.getSize();
  }
  const Company &operator[](size_t position) const {
    return *storage[position];
  }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  CompanySource(const CompanySource&);
  CompanySource &operator=(const CompanySource&);
  Storage<Company, true> storage;
};
 
// ñóùíîñòü, õðàíÿùàÿ ñâÿçè êîìïàíèÿ-âëàäåëåö
class CompanyOwnerSource {
 public:
  CompanyOwnerSource() : storage() {}
  virtual ~CompanyOwnerSource() {
    for (CompanyOwner * const *i = storage.begin(); i != storage.end(); ++i) {
      delete *i;
    }
    storage.clear();
  }
  bool add(const Company &company, const Owner &owner) {
    CompanyOwner *object = new CompanyOwner(company, owner);
    const CompanyOwner *result = storage.add(object);
    if (result != object) {
      delete object;
      return false;
    } else {
      return true;
    }
  }
  bool remove(const Company &company, const Owner &owner) {
    CompanyOwner object(company, owner);
    const CompanyOwner *removed = storage.remove(&object);
    bool result = removed != 0;
    delete removed;
    return result;
  }
  bool contains(const Company &company, const Owner &owner) const {
    CompanyOwner object(company, owner);
    return storage.find(&object) != storage.end();
  }
  size_t getSize() const {
    return storage.getSize();
  }
  const CompanyOwner &operator[](size_t position) const {
    return *storage[position];
  }
 private:
  // ñëåäóþùèå äâå ñòðîêè çîðêî ñëåäÿò, ÷òîáû
  // íå áûëî âûçîâîâ êîíñòðóêòîðà êîïèðîâàíèÿ è îïåðàòîðà ïðèñâàèâàíèÿ
  CompanyOwnerSource(const CompanyOwnerSource&);
  CompanyOwnerSource &operator=(const CompanyOwnerSource&);
  Storage<CompanyOwner, false> storage;
};
 
int main(int argc, char **argv) {
  srand(time(0));
  
  OwnerSource owners; // èñòî÷íèê äàííûõ î âëàäåëüöàõ
  owners.add("Ivan", "Lenina 12");
  owners.add("Petr", "Tupik 9");
  owners.add("Ivan", "Lenina 12");  // íå áóäåò äîáàâëåí
  owners.add("Vladimir", "Ilyicha 73");
  owners.add("Ivan", "Stalina 33");
  owners.add("Petr", "Tupik 33");
  owners.remove("Petr", "Tupik 9"); // óäàëåíèå ýëåìåíòà
  owners.add("Alexandr", "Soyuza 1");
  owners.add("Zinaida", "Joka 8");
  owners.add("Anna", "Mendelya 42");
  // âûâîä ýëåìåíòîâ
  for (int i = 0; i < owners.getSize(); ++i) {
    std::cout << owners[i] << std::endl;
  }
 
  CompanySource companies; // èñòî÷íèê äàííûõ î êîìïàíèÿõ
  companies.add("vector");
  companies.add("vector");
  companies.add("iwai");
  companies.add("nobie");
  companies.remove(2);     // óäàëåíèå ïî èíäåêñó
  companies.add("vector");
  
  for (int i = 0; i < companies.getSize(); ++i) {
    std::cout << companies[i] << std::endl;
  }
 
  CompanyOwnerSource companyOwners;
  for (int i = 0; i < companies.getSize(); ++i) {
    std::cout << "adding for " << companies[i] << std::endl;
    for (int j = 0; j < owners.getSize(); ++j) {
      companyOwners.add(companies[i], owners[rand() % owners.getSize()]);
    }
  }
  
  for (int i = 0; i < companyOwners.getSize(); ++i) {
    std::cout << companyOwners[i] << std::endl;
  }
  
  return 0;
}
Всё. Откуда эти чудовищные нагромождения структур и типов данных невпопад?!
1
2 / 2 / 1
Регистрация: 23.11.2011
Сообщений: 87
31.03.2013, 10:19  [ТС]
Все сделал и сдал, очень быстрая и крайне нетребовательная к ресурсам программа получилась. Если кому интересно:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
#ifndef __PROGTEST__
#include <iostream>
#include <iomanip>
#include <string>
#include <cstring>
#include <cstdlib>
#include <cstdio>
using namespace std;
#endif /* __PROGTEST__ */
 
 
class CEntity;
class CEntities;
class CCompanyIndex;
class CIterator;
 
/**
 * @CEntity
 *
 * Represents some entity with basic control over it. 
 * Has an array of links to other entities.
 * Array of links is an array of pointers.
 * @WARNING Size of pointers array only grows up!
 * 
 * @friends CEntities
 * 
 * @constructor Entity() - basic constructor
 * @destructor ~Entity() - basic destructor
 * @method void AddLink( CEntity * NewEntity ) - adds link
 * @method void DelLink( CEntity * DelEntity ) - deletes link
 * @method  int GetLinksCount()const - returns number of active links
 *
 * @param std::string m_Name name value 
 * @param std::string m_Address address value 
 * @param int iLinksCount number of active links
 * @param CEntity ** m_Links an array of pointers to other entities
 */
class CEntity {
public:
 
    /**
     * Basic Constructor
     */
    CEntity() {
        m_Name = "";
        m_Address = "";
        m_Links = NULL;
        iLinksCount = 0;
        iLinksSize = 0;
    };
 
    /**
     * Basic Destructor
     */
    ~CEntity() {
        free(m_Links);
    };
 
    /**
     * Finds if link to requested entity exists in array and returns its index
     * @WARNING uses linear search algorithm
     * @param CEntity * FindEntity - pointer to CEntity which have to be found
     * @param  int & pos - position where link is.
     */
    bool SearchLink(CEntity * FindEntity, int & pos) {
 
        for (int i = 0; i < iLinksCount; i++) {
            if (m_Links[i] == FindEntity) {
                pos = i;
                return true;
                break;
            }
        }
        return false;
    }
 
    /**
     * Adds link to some entity to m_Links array, also resizing it
     * @param CEntity * NewEntity - pointer to CEntity
     */
    void AddLink(CEntity * NewEntity) {
        if (iLinksCount >= iLinksSize) {
            ++iLinksSize *= 2;
            m_Links = (CEntity**) realloc(m_Links, sizeof (CEntity*)*(iLinksSize + 1));
        }
        m_Links[iLinksCount++] = NewEntity;
    };
 
    /**
     * Finds and deletes link to some entity from m_Links array, 
     * without resizing it!@WARNING uses linear search algorithm
     * @param CEntity * DelEntity - pointer to CEntity which have to be deleted
     */
    void DelLink(CEntity * DelEntity) {
        int pos;
        if (SearchLink(DelEntity, pos)) {
            for (int i = pos; i < iLinksCount - 1; i++) {
                m_Links[i] = m_Links[i + 1];
            }
            iLinksCount--;
            m_Links[iLinksCount] = NULL;
        }
    };
 
    /**
     * Returns number of linked entities
     * @return int iLinksCount - number of existing links
     */
    int GetLinksCount()const {
        return iLinksCount;
    };
 
    friend class CEntities;
    friend class CIterator;
    friend class CCompanyIndex;
private:
 
    std::string m_Name;
    std::string m_Address;
    CEntity ** m_Links;
    int iLinksCount;
    int iLinksSize;
};
 
/**
 * @CEntities
 *
 * Represents CEntity Array of entities with advanced control over it.
 * Stores all entities in an alphabetical order using both m_Name and m_Address strings for sorting.
 * @WARNING Size of pointers (to entities) array only grows up!
 * 
 * 
 * @constructor CEntities() - basic constructor
 * @destructor ~CEntities() - basic destructor
 * @method bool Add - adds link and returns pointer to it
 * @method bool Del - deletes enity and returns true or false if entity was not found
 * @method CEntity * Search - returns pointer to entity or NULL if don't exists
 * 
 * @param CEntity ** InstancesArray array of pointers to entities
 * @param int iInstancesCount number of active entities
 * @param int iInstancesSize physical size of pointers array, grows in a geometrical progression with q = 2.0
 */
class CEntities {
public:
 
    /**
     * Basic Constructor which alocates memory for 1000 new instances of CEntity, also calling their constructors
     */
    CEntities() {
        iInstancesSize = 100;
        iInstancesCount = 0;
        InstancesArray = NULL;
        InstancesArray = (CEntity**) realloc(InstancesArray, sizeof (CEntity*) * iInstancesSize);
        for (int i = 0; i < iInstancesSize; i++) {
            InstancesArray[i] = new CEntity;
        }
    };
 
    /**
     * Basic Destructor - calls destructor for every entity and then frees all allocated for pointers array memory
     */
    ~CEntities() {
        for (int i = 0; i < iInstancesSize; i++) {
            delete InstancesArray[i];
        }
        free(InstancesArray);
    };
 
    /**
     * @Add
     * 
     * Adds new entity to instances array, and returns true, or returns false and do nothing if such a pair exists
     * 
     * @param const std::string & oName - entity name value
     * @param const std::string & oAddr - enity address value
     */
    CEntity* Add(const std::string& Name,
            const std::string& Addr) {
        int pos = -1;
        if (FindPair(Name, Addr, pos)) {
            //        cout << "Exists" << endl;
            return NULL;
        }
        //     else {
        //        cout << "Don't exists" << endl;
        //    }
        //    cout << "Inserting to " << pos << endl;
        if (iInstancesCount >= iInstancesSize) {
            iInstancesSize *= 1.5;
            InstancesArray = (CEntity**) realloc(InstancesArray, sizeof (CEntity*) * iInstancesSize);
            for (int i = iInstancesCount; i < iInstancesSize; i++) {
                InstancesArray[i] = new CEntity;
            }
        }
        delete InstancesArray[iInstancesCount];
        for (int i = iInstancesCount; i > pos; i--) {
            InstancesArray[i] = InstancesArray[i - 1];
        }
        iInstancesCount++;
        InstancesArray[pos] = new CEntity;
        InstancesArray[pos]->m_Name = Name;
        InstancesArray[pos]->m_Address = Addr;
        return InstancesArray[pos];
    }
 
    /**
     * @Del
     * 
     * Finds by requested pair of strings an entity, deletes it and returns true or false if entity was not found
     * 
     * @param const std::string & oName - entity name value
     * @param const std::string & oAddr - enity address value
     */
    bool Del(const std::string & oName,
            const std::string & oAddr) {
        int pos = -1;
        if (!FindPair(oName, oAddr, pos)) {
            //        cout << "Exists" << endl;
            return false;
        }
        delete InstancesArray[pos];
        for (int i = pos; i < iInstancesCount - 1; i++) {
            InstancesArray[i] = InstancesArray[i + 1];
        }
        iInstancesCount--;
        InstancesArray[iInstancesCount] = new CEntity;
        return true;
    }
 
    /**
     * @Search
     * 
     * Looking for requested pair of name and adress
     * 
     * @return pointer to CEntity or NULL if don't exists
     */
    CEntity * Search(const std::string & oName,
            const std::string & oAddr)const {
        int pos = -1;
        if (!FindPair(oName, oAddr, pos)) {
            return NULL;
        }
        return InstancesArray[pos];
    }
    friend class CCompanyIndex;
private:
 
    /**
     * @GetFirstAddrPosition
     * 
     * Looking for requested Address in entities array
     * sets index of first element with requested Address on requested interval 
     * or index where it should be if there is no entity with such an address 
     * on such interval
     * 
     * @return true if requested string found or false if not
     */
    bool GetFirstAddrPosition(const std::string& Str2Find,
            int f,
            int Size,
            int& pos)const {
        int l = f;
        int r = Size;
        while (l < r) {
            int m = l + (r - l) / 2;
            if (Str2Find.compare(InstancesArray[m]->m_Address) > 0) {
                l = m + 1;
            } else {
                r = m;
            }
        }
        pos = l;
        if (Str2Find.compare(InstancesArray[l]->m_Address) == 0) {
            return true;
        }
        return false;
    }
 
    /**
     * @GetLastAddrPosition
     * 
     * Looking for requested Address in entities array
     * sets index of last element with requested Address on requested interval 
     * or index where it should be if there is no entity with such an address 
     * on such interval
     * 
     * @return true if requested string found or false if not
     */
    bool GetLastAddrPosition(const std::string& Str2Find,
            int f,
            int Size,
            int& pos)const {
        int l = f;
        int r = Size;
        while (l < r) {
            int m = l + (r - l) / 2;
            if (!(Str2Find.compare(InstancesArray[m]->m_Address) < 0)) {
                l = m + 1;
            } else {
                r = m;
            }
        }
        pos = l;
        if (Str2Find.compare(InstancesArray[l - 1]->m_Address) == 0) {
            return true;
        }
        return false;
    }
 
    /**
     * @GetFirstNamePosition
     * 
     * Looking for requested Name in entities array
     * sets index of first element with requested Name on requested interval 
     * or index where it should be if there is no entity with such a Name
     * on such interval
     * 
     * @return true if requested string found or false if not
     */
    bool GetFirstNamePosition(const std::string& Str2Find,
            int f,
            int Size,
            int& pos)const {
        int l = f;
        int r = Size;
        while (l < r) {
            int m = l + (r - l) / 2;
            if (Str2Find.compare(InstancesArray[m]->m_Name) > 0) {
                l = m + 1;
            } else {
                r = m;
            }
        }
        pos = l;
        if (Str2Find.compare(InstancesArray[l]->m_Name) == 0) {
            return true;
        }
        return false;
    }
 
    /**
     * @GetLastNamePosition
     * 
     * Looking for requested Address in entities array
     * sets index of last element with requested Address on requested interval 
     * or index where it should be if there is no entity with such a Name
     * on such interval
     * 
     * @return true if requested string found or false if not
     */
    bool GetLastNamePosition(const std::string& Str2Find,
            int f,
            int Size,
            int& pos)const {
        int l = f;
        int r = Size;
        while (l < r) {
            int m = l + (r - l) / 2;
            if (!(Str2Find.compare(InstancesArray[m]->m_Name) < 0)) {
                l = m + 1;
            } else {
                r = m;
            }
        }
        pos = l;
        if (Str2Find.compare(InstancesArray[l - 1]->m_Name) == 0) {
            return true;
        }
        return false;
    }
 
    /**
     * @FindPair
     * 
     * Looking for requested pair of name and adress setting index in array where such an entity is or should be
     * method FindPair returns true if there is CEntity with such a Str2Find1 and 
     * Str2Find2, or false if there is no such pair
     * in both cases returns position, using reference int&pos, 
     * where this entity already is or should be
     * 
     * @return true if exists or false if it doesn't
     */
    bool FindPair(const std::string& Str2Find1,
            const std::string& Str2Find2,
            int& pos)const {
 
        int InsertPositionFirstArg = 0;
        int InsertPositionSecondArg = 0;
        bool IfFoundFirstArg = false;
        bool IfFoundSecondArg = false;
        bool PairExists = true;
        std::string SearchQuery1 = Str2Find1;
        std::string SearchQuery2 = Str2Find2;
        int FirstElement = 0;
        int ElementAgainstLast = iInstancesCount;
        IfFoundFirstArg = GetFirstNamePosition(SearchQuery1, FirstElement, ElementAgainstLast, InsertPositionFirstArg);
        if (IfFoundFirstArg) {
            //cout << "First entity for String \"" << SearchQuery1 << "\" Found on index: " << InsertPositionFirstArg << endl;
            IfFoundSecondArg = GetLastNamePosition(SearchQuery1, FirstElement, ElementAgainstLast, InsertPositionSecondArg);
            if (IfFoundSecondArg) {
                //cout << "Last entity for String \"" << SearchQuery1 << "\" Found on index: " << InsertPositionSecondArg - 1 << endl;
                IfFoundFirstArg = GetFirstAddrPosition(SearchQuery2, InsertPositionFirstArg, InsertPositionSecondArg, InsertPositionFirstArg);
                if (IfFoundFirstArg) {
                    //cout << "First entity for String \"" << SearchQuery2 << "\" Found on index: " << InsertPositionFirstArg << endl;
                    IfFoundSecondArg = GetLastAddrPosition(SearchQuery2, InsertPositionFirstArg, InsertPositionSecondArg, InsertPositionSecondArg);
                    if (IfFoundSecondArg) {
                        //cout << "Last entity for String \"" << SearchQuery2 << "\" Found on index: " << InsertPositionSecondArg - 1 << endl;
                    } else {
                        PairExists = false;
                        //cout << "Last entity for String \"" << SearchQuery2 << "\" was NOT found; insert it to index: " << InsertPositionSecondArg - 1 << endl;
                    }
                } else {
                    PairExists = false;
                    //cout << "Last entity for String \"" << SearchQuery2 << "\" was NOT found; insert it to index: " << InsertPositionFirstArg << endl;
                }
            } else {
                PairExists = false;
                //cout << "Last entity for String \"" << SearchQuery1 << "\" was NOT found; insert it to index: " << InsertPositionSecondArg - 1 << endl;
            }
        } else {
            PairExists = false;
            //cout << "First entity for String \"" << SearchQuery1 << "\" was NOT found; insert it to index: " << InsertPositionFirstArg << endl;
        }
        int ins;
        if (IfFoundFirstArg && IfFoundSecondArg) {
            ins = (InsertPositionSecondArg < InsertPositionFirstArg) ? InsertPositionSecondArg : InsertPositionFirstArg;
        } else {
            ins = (IfFoundFirstArg) ? InsertPositionSecondArg : InsertPositionFirstArg;
        }
        //cout << "pair exists ? - " << PairExists << endl;
        //cout << "insert at the end = " << ins << endl;
        pos = ins;
        return PairExists;
    }
 
    /**
     * @Print
     * 
     * Prints out all current instances
     */
    void Print() {
        for (int i = 0; i < iInstancesCount; i++) {
            std::cout << "[" << i << "]\t" << InstancesArray[i]->m_Name << "\t" << InstancesArray[i]->m_Address << std::endl;
        }
    };
    CEntity ** InstancesArray;
    int iInstancesCount;
    int iInstancesSize;
 
};
 
class CIterator {
public:
 
    CIterator() {
        iEntitiesCount = 0;
        Current = 0;
        Entities = NULL;
    };
 
    ~CIterator() {
        free(Entities);
    };
 
    bool AtEnd(void) const {
        bool b = (Current == iEntitiesCount);
        return b;
    };
 
    void Next(void) {
        Current++;
    };
 
    const string & Name(void) const {
        return Entities[Current]->m_Name;
    };
 
    const string & Address(void) const {
        return Entities[Current]->m_Address;
    };
 
    void AddEntity(CEntity * NewEntity) {
        Entities = (CEntity**) realloc(Entities, sizeof (CEntity*)*(iEntitiesCount + 1));
        Entities[iEntitiesCount] = NewEntity;
        iEntitiesCount++;
    };
private:
    CEntity ** Entities;
    int iEntitiesCount;
    int Current;
 
};
 
class CCompanyIndex {
public:
 
    CCompanyIndex() {
        Owners = new CEntities;
        Companies = new CEntities;
    };
 
    ~CCompanyIndex() {
        delete Owners;
        delete Companies;
    }
    ;
 
    bool Add(const string & oName,
            const string & oAddr,
            const string & cName,
            const string & cAddr);
 
    bool Del(const string & oName,
            const string & oAddr,
            const string & cName,
            const string & cAddr);
 
    CIterator * SearchOwner(const string & oName,
            const string & oAddr) const;
 
    CIterator * SearchCompany(const string & cName,
            const string & cAddr) const;
 
    void Print() {
 
 
        Owners->Print();
 
 
 
        Companies->Print();
 
    }
private:
 
    CEntities * Owners;
    CEntities * Companies;
 
};
 
bool CCompanyIndex::Add(const std::string& oName, const std::string& oAddr, const std::string& cName, const std::string& cAddr) {
    /**Metoda Add(oName, oAddr, cName, cAddr) přidá do existující databáze další záznam. 
     * Metoda vrací hodnotu true, pokud byl do záznam přidán, nebo hodnotu false, 
     * pokud přidán nebyl 
     * (protože již v databázi existoval záznam o tom, že daný majitel vlastní podíl v zadané firmě.). 
     * Parametry oName a oAddr reprezentují jméno a adresu vlastníka, parametry cName a cAddr udávají název a adresu firmy.
     */
    CEntity * Owner = Owners->Search(oName, oAddr);
    CEntity * Company = Companies->Search(cName, cAddr);
    if ((Owner != NULL)&&(Company != NULL)) {
        //look for link between them
        int pos;
        if (Owner->SearchLink(Company, pos)) {
            return false; //link was found
        }
    }
    if ((Owner == NULL)) {
        if ((Owner = Owners->Add(oName, oAddr)) == NULL) {
            return false;
        }
 
 
    }
    if ((Company == NULL)) {
        if ((Company = Companies->Add(cName, cAddr)) == NULL) {
            return false;
        }
    }
    Owner->AddLink(Company);
    Company->AddLink(Owner);
 
 
 
    return true;
}
 
bool CCompanyIndex::Del(const string & oName,
        const string & oAddr,
        const string & cName,
        const string & cAddr) {
    /**Metoda Del (oName, oAddr, cName, cAddr) odstraní záznam z databáze. 
     * Pokud byl záznam skutečně odstraněn, vrátí metoda hodnotu true. 
     * Pokud záznam neodstraní (protože zadaný majitel neměl v zadané firmě podíl), 
     * vrátí metoda hodnotu false.
     */
    CEntity * Owner = Owners->Search(oName, oAddr);
    CEntity * Company = Companies->Search(cName, cAddr);
    if ((Owner != NULL)) {
        if (Company != NULL) {
            int pos;
            if (Owner->SearchLink(Company, pos)) {
                //link was found
                Owner->DelLink(Company);
                Company->DelLink(Owner);
                if (Owner->GetLinksCount() == 0) {
                    Owners->Del(oName, oAddr);
                }
                if (Company->GetLinksCount() == 0) {
                    Companies->Del(cName, cAddr);
                }
                return true; //deleted
            } else {
                return false; //there is no link between company and owner
            }
        } else {
            return false; //there is no company
        }
    } else {
        return false; //there is no owner
    }
    return false;
}
 
CIterator * CCompanyIndex::SearchOwner(const string & oName,
        const string & oAddr) const {
    /*Metoda SearchOwner (oName, oAddr) zjistí seznam všech firem, ve kterých má zadaný majitel podíl. 
     * Pokud zadaný majitel nemá podíl v žádné firmě, metoda vrátí návratovou hodnotu NULL. 
     * Pokud zadaný majitel má nějaký podíl v nějaké firmě, metoda vrátí dynamicky alokovanou instanci třídy CIterator, 
     * která obsahuje seznam všech firem, kde má zadaný majitel nějaký podíl. 
     * Po zpracování výsledku volající uvolní předanou instanci CIterator voláním delete.
     */
    CEntity * Owner = Owners->Search(oName, oAddr);
    if ((Owner == NULL)) {
        return NULL;
    }
    if ((Owner->GetLinksCount() == 0)) {
        return NULL;
    }
    CIterator * it = new CIterator;
    for (int i = 0; i < Owner->GetLinksCount(); i++) {
        it->AddEntity(Owner->m_Links[i]);
    }
    return it;
}
 
CIterator * CCompanyIndex::SearchCompany(const string & cName,
        const string & cAddr) const {
    /*Metoda SearchOwner (oName, oAddr) zjistí seznam všech firem, ve kterých má zadaný majitel podíl. 
     * Pokud zadaný majitel nemá podíl v žádné firmě, metoda vrátí návratovou hodnotu NULL. 
     * Pokud zadaný majitel má nějaký podíl v nějaké firmě, metoda vrátí dynamicky alokovanou instanci třídy CIterator, 
     * která obsahuje seznam všech firem, kde má zadaný majitel nějaký podíl. 
     * Po zpracování výsledku volající uvolní předanou instanci CIterator voláním delete.
     */
    CEntity * Company = Companies->Search(cName, cAddr);
    if ((Company == NULL)) {
        return NULL;
    }
    if ((Company->GetLinksCount() == 0)) {
        return NULL;
    }
    CIterator * it = new CIterator;
    for (int i = 0; i < Company->GetLinksCount(); i++) {
        it->AddEntity(Company->m_Links[i]);
    }
    return it;
}
 
 
 
 
#ifndef __PROGTEST__
 
string randomString(int seed)
{
    int len = seed % 25 + 5;
    string retValue = "";
    for (int i = 0; i < len; i++)
        retValue += (char)((i*seed + (seed+i)*(seed-i) + i) % 78 + 48);
    return retValue;        
}
 
void  showResults ( CIterator * it )
{
    if (it == NULL)
        cout << "NULL" << endl;
    else
        while ( ! it -> AtEnd () )
        {
            cout << it -> Name ();
            cout<< it -> Address () << endl;
            it -> Next ();
        }
}
 
int main()
{
 
    CIterator * it;
    bool        status=false;
    CCompanyIndex  b1;
    status = b1 . Add ( "Smith", "Oak road", "ACME, Ltd.", "One ACME road" );
    // status = true
    status = b1 . Add ( "Brown", "Second street", "ACME, Ltd.", "Mountain road" );
    // status = true
    status = b1 . Add ( "Hacker", "5-th avenue", "Forks and Knives, Ltd.", "Cutlery avenue" );
    // status = true
    status = b1 . Add ( "Hacker", "7-th avenue", "Child toys, Inc.", "Red light district" );
    // status = true
    status = b1 . Add ( "Smith", "Oak road", "ACME, Ltd.", "Mountain road" );
    // status = true
    status = b1 . Add ( "Hacker", "5-th avenue", "ACME, Ltd.", "One ACME road" );
    // status = true
    status = b1 . Add ( "Hacker", "7-th avenue", "ACME, Ltd.", "Mountain road" );
    // status = true
 
    for (int i = 0; i < 5000; i++)
        b1.Add(randomString(i+545), randomString(i+327), randomString(i+248), randomString(i+532));
 
    for (int i = 0; i < 1238; i++)
        b1.Add(randomString(i+123), randomString(i+584), randomString(i+338), randomString(i+225));
 
    for (int i = 0; i < 300; i++)
        b1.Add("Fred", "CubeGarden", randomString(i+338), randomString(i+225));
 
    it = b1 . SearchOwner ( "Fred", "CubeGarden" );
    showResults ( it );
 
    it = b1 . SearchOwner ( "Brown", "Second street" );
    showResults ( it );
    /*
    ----8<----8<----8<----8<----
    ACME, Ltd., Mountain road
    ----8<----8<----8<----8<----
    */
    delete it;
    it = b1 . SearchOwner ( "Hacker", "Oak road" );
    showResults ( it );
 
    // it = NULL
    it = b1 . SearchOwner ( "Hacker", "7-th avenue" );
    showResults ( it );
    /*
    ----8<----8<----8<----8<----
    Child toys, Inc., Red light district
    ACME, Ltd., Mountain road
    ----8<----8<----8<----8<----
    */
    delete it;
    it = b1 . SearchCompany ( "ACME, Ltd.", "Mountain road" );
    showResults ( it );
    /*
    ----8<----8<----8<----8<----
    Hacker, 7-th avenue
    Brown, Second street
    Smith, Oak road
    ----8<----8<----8<----8<----
    */
    delete it;
    it = b1 . SearchCompany ( "Child toys, Inc.", "Mountain road" );
    showResults ( it );
    // it = NULL
    status = b1 . Del ( "Smith", "Oak road", "Child toys, Inc.", "Red light district" );
    // status = false
    status = b1 . Del ( "Smith", "Oak road", "ACME, Ltd.", "Mountain road" );
    // status = true
    it = b1 . SearchOwner ( "Smith", "Oak road" );
    showResults ( it );
    /*
    ----8<----8<----8<----8<----
    ACME, Ltd., One ACME road
    ----8<----8<----8<----8<----
    */
    delete it;
    status = b1 . Add ( "Smith", "Oak road", "ACME, Ltd.", "One ACME road" );
    // status = false
 
    return 0;
}
 
#endif /*__PROGTEST__*/
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
31.03.2013, 10:19
Помогаю со студенческими работами здесь

Связь таблиц управляемых списков по различным полям
Доброго всем дня! Начал работать с управляемыми списками в bitrix. Создал несколько таблиц. Собственно, появился вопрос: как сделать...

Как сделать связь выпадающих списков страна, город?
Как сделать связь выпадающих списков страна, город? Город сначала не показывается и заполняется автоматически при выборе страны. В MS...

Объединение 2 и более списков в список списков по индексу без использования циклов
Уважаемые товарищи, подскажите есть ли способ объединения 2 и более списков в список списков по индексу без использования циклов. К примеру...

Как сложить сумму из чисел сотен списков и узнать количество списков?
Доброго времени суток! Не могу понять как сложить сумму из несколько тысяч списков. Подскажите пожалуйста? В коде удалил из списков все...

Перестановка списков заданных уровней, учитывая промежуточное состояние списков
Написать программу перестановки списков заданных уровней, учитывая промежуточное состояние списков. Поиск соответствующих элементов...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
23
Ответ Создать тему
Новые блоги и статьи
Debian 13: Установка Lazarus QT5
ВитГо 09.05.2026
Эта инструкция моя компиляция инструкций volvo https:/ / www. cyberforum. ru/ blogs/ 203668/ 10753. html и его же старой инструкции по установке Lazarus с gtk2. . .
Нейросеть на алгоритме "эстафета хвоста" как перспектива.
Hrethgir 06.05.2026
На десерт, когда запущу сервер. Статья тут https:/ / habr. com/ ru/ articles/ 1030914/ . Автор я сам, нейросеть только помогает в вопросах которые мне не известны - не знаю людей которые знали-бы. . .
Асинхронный приём данных из COM-порта
Argus19 01.05.2026
Асинхронный приём данных из COM-порта Купил на aliexpress термопринтер QR701. Он оказался странным. Поключил к Arduino Nano. Был очень удивлён. Наотрез отказывается печатать русские буквы. Чтобы. . .
попытка написать игровой сервер на C++
pyirrlicht 29.04.2026
попытка написать игровой сервер на плюсах с открытым бесконечным миром. возможно получится прикрутить интерпретатор питон для кастомизации игровой логики. что есть на текущий момент:. . .
Контроль уникальности выбранного документа-основания при изменении реквизита
Maks 28.04.2026
Алгоритм из решения ниже разработан на примере нетипового документа "ЗаявкаНаРемонтСпецтехники", разработанного в КА2. Задача: уведомлять пользователя, если указанная заявка (документ-основание). . .
Благородство как наказание
Maks 24.04.2026
У хорошего человека отношения с женщинами всегда складываются трудно. А я человек хороший. Заявляю без тени смущения, потому что гордиться тут нечем. От хорошего человека ждут соответствующего. . .
Валидация и контроль данных табличной части документа перед записью
Maks 22.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа, разработанного в КА2. Задача: контроль и валидация данных табличной части документа перед записью с учетом регламента компании. . .
Отчёт о затраченных материалах за определенный период с макетом печатной формы
Maks 21.04.2026
Отчёт из решения ниже размещён в конфигурации КА2. Задача: разработка отчёта по затраченным материалам за определённый период, с возможностью вывода печатной формы отчёта с шапкой и подвалом. В. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru