Проверка вывода "Найти числа от 1 до 1000, у которых сумма простых множителей сама по себе является простым числом" из "Прыжка Аллена в C++" (глава 7)
Вопрос:
Разработайте программу, которая находит все числа от 1 до 1000, чьи главные факторы, при сложении вместе, суммируют до простого числа (например, 12 имеет простые множители 2, 2 и 3, которые в сумме равны 7, что является простым), Реализуйте код для этого алгоритма.
Я изменил задачу, чтобы суммировать только уникальные факторы, потому что я не понимаю, почему вы учитываете фактор дважды, как в его примере с использованием 12.
Мое решение. Есть ли какой-нибудь хороший (читай: автоматизированный) способ проверить вывод моей программы?
Пример вывода от 1 до 1000:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
17
19
20
22
23
24
25
26
28
29
30
31
34
37
40
41
43
44
46
47
48
49
52
53
58
59
60
61
63
67
68
70
71
73
76
79
80
82
83
88
89
92
94
96
97
99
101
103
107
109
113
116
117
118
120
121
124
127
131
136
137
139
140
142
147
148
149
151
153
157
160
163
164
167
169
171
172
173
176
179
181
184
188
189
191
192
193
197
198
199
202
207
210
211
212
214
223
227
229
232
233
239
240
241
244
251
252
257
261
263
268
269
271
272
273
274
275
277
279
280
281
283
286
289
292
293
294
297
298
306
307
311
313
317
320
325
331
332
333
334
337
347
349
351
352
353
358
359
361
367
368
369
373
376
379
382
383
384
388
389
394
396
397
399
401
404
409
412
414
419
421
423
424
425
428
431
433
439
443
449
454
457
459
461
462
463
464
467
468
472
475
478
479
480
487
491
495
499
503
509
513
521
522
523
524
529
531
538
539
541
544
546
547
548
549
550
557
560
561
562
563
567
569
571
572
575
577
587
588
593
594
599
601
603
604
605
607
612
613
617
619
621
622
628
631
639
640
641
643
646
647
651
652
653
659
661
664
668
673
677
683
684
691
692
694
701
704
709
712
714
718
719
725
726
727
733
736
738
739
741
743
751
752
756
757
759
761
764
765
768
769
772
773
775
777
783
787
792
797
798
801
809
811
821
823
825
827
828
829
833
837
838
839
841
846
847
848
850
853
856
857
859
862
863
873
877
881
883
887
891
892
903
904
907
908
909
911
918
919
922
925
928
929
932
937
941
944
947
953
954
957
960
961
966
967
971
975
977
981
983
991
997
999
Обновление: я решил свою проблему и проверил вывод моей программы, используя серию OEIS, как предложено @MVW (показано в источнике, указанном в моем новом решении github). В будущем я буду стремиться протестировать свои программы, выполнив ноль или более следующих действий (в зависимости от масштаба / важности проблемы):
- Google ключевые слова для существующего решения проблемы, сравнивая его с моим решением, если я найду его
- Компоненты модульного теста на правильность, как они построены и интегрированы, сравнивая эти тесты с известными правильными результатами
2 ответа
Некоторые предложения:
Вам необходимо проверить свойства ваших рассчитанных чисел. Вот что значит
- расчет основных факторов и
- рассчитывая их сумму и
- проверка, является ли эта сумма простым числом.
Кстати, именно это и должна делать ваша программа.
Так что один хороший вариант для проверки - это сравнение вашего вывода с известным решением или выходом другой программы, которая, как известно, работает. Хитрость в том, чтобы иметь такое решение или программу доступной. И я пренебрегаю тем, что ваше сравнение может быть также изведено ошибками:-)
Если вы просто сравните его с другими реализациями, например, с программами других людей, то получится больше голосования, это не будет доказательством. Это только увеличит вероятность того, что ваша программа верна, если несколько независимых реализаций получат одинаковый результат. Конечно, все реализации могут ошибаться:-)Чем больше согласен, тем лучше. И чем разнообразнее реализации, тем лучше. Например, вы можете использовать разные языки программирования, алгебраические системы или друга со временем, бумагой, карандашом и Википедией.:-)
Другой способ - добавить проверки к вашим промежуточным шагам, чтобы получить больше уверенности в своем результате. Вид построения цепочки доверия.
Вы можете вывести основные факторы, которые вы определили, и сравнить их с результатами основной программы факторизации, которая, как известно, работает.
Затем вы проверяете, работает ли ваше суммирование.
Наконец, вы можете проверить, правильно ли работает тест на простоту, который вы применяете к суммам кандидатов, введя в него известные простые числа и не простые числа и так далее.
Это то, что люди делают, например, с модульным тестированием. Попытка охватить большинство частей кода как работающие, надеясь, что эти части работают, что все будет работать.
Или вы можете формально доказать свою программу шаг за шагом, например, используя Hoare Calculus или другой формальный метод. Но это сложно, и вы можете в конечном итоге перевести программные ошибки на ошибки в доказательстве.
И сегодня, в эпоху Интернета, вы, конечно, можете найти решение в Интернете:
Попробуйте найти сумму простых множителей в онлайн-энциклопедии целочисленных последовательностей, которая должна дать вам серию A100118.:-) Это проблема множественности, но она показывает вам, что делают профессионалы в теории чисел, с Mathematica и фрагментами программы для вычисления ряда, аргументом для случая 1 и литературой. Довольно впечатляющий.
Вот ответ, который я получаю. Я исключаю 1, поскольку у него нет простых делителей, поэтому их сумма равна 0, а не простому числу.
Haskell> filter (isPrime . sum . map fst . primePowers) [2..1000]
[2,3,4,5,6,7,8,9,10,11,12,13,16,17,18,19,20,22,23,24,25,27,29,31,32,34,36,37,40,
41,43,44,47,48,49,50,53,54,58,59,61,64,67,68,71,72,73,79,80,81,82,83,88,89,96,97
,100,101,103,107,108,109,113,116,118,121,125,127,128,131,136,137,139,142,144,149
,151,157,160,162,163,164,165,167,169,173,176,179,181,191,192,193,197,199,200,202
,210,211,214,216,223,227,229,232,233,236,239,241,242,243,250,251,256,257,263,269
,271,272,273,274,277,281,283,284,288,289,293,298,307,311,313,317,320,324,328,331
,337,343,345,347,349,352,353,358,359,361,367,373,379,382,383,384,385,389,390,394
,397,399,400,401,404,409,419,420,421,428,431,432,433,435,439,443,449,454,457,461
,462,463,464,467,472,478,479,484,486,487,491,495,499,500,503,509,512,521,523,529
,538,541,544,547,548,557,561,562,563,568,569,570,571,576,577,578,587,593,595,596
,599,601,607,613,617,619,622,625,630,631,640,641,643,647,648,651,653,656,659,661
,665,673,677,683,691,694,701,704,709,714,715,716,719,727,729,733,739,743,751,757
,759,761,764,768,769,773,777,780,787,788,795,797,798,800,808,809,811,819,821,823
,825,827,829,838,839,840,841,853,856,857,858,859,862,863,864,877,881,883,885,887
,903,907,908,911,919,922,924,928,929,930,937,941,944,947,953,956,957,961,967,968
,971,972,977,983,991,997,1000]
Haskell> primePowers 12
[(2,2),(3,1)]
Haskell> primePowers 14
[(2,1),(7,1)]
Вы можете жестко закодировать этот список и проверить его. Я уверен, что эти результаты без ошибок.
(читать .
имеет").