Лекции.Орг


Поиск:




Категории:

Астрономия
Биология
География
Другие языки
Интернет
Информатика
История
Культура
Литература
Логика
Математика
Медицина
Механика
Охрана труда
Педагогика
Политика
Право
Психология
Религия
Риторика
Социология
Спорт
Строительство
Технология
Транспорт
Физика
Философия
Финансы
Химия
Экология
Экономика
Электроника

 

 

 

 


Министерство образования и науки Российской Федерации 3 страница




 

/*prog11.c*/

 

#include<stdio.h>

 

main()

{

char x = ‘A’, y = 97;

printf (“Численное значение переменной х равно %d, в ней хранится символ %c \n Численное значение переменной у равно %d, в ней хранится символ %с \n”, x, x, y, y);

return 0;

}

 

Откомпилировав и выполнив программу мы увидим, что буква ‘А’ имеет ASCII-код 65, а ASCII-коду 97 соответствует строчная буква ‘а’.

Таким образом, мы можем присваивать целочисленные значения переменным типов int и char, выполнять с ними обычные арифметические действия, представлять в десятичной системе счисления. Однако переменные этих типов занимают разный объем памяти. Операция

 

sizeof (имя_переменной)

 

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

 

/*prog12.c*/

 

#include<stdio.h>

 

main()

{

int x = 97;

char y = 97;

printf (“Переменной х типа int присвоено значение %d, она занимает %d байт Переменной у типа char присвоено значение %d, она занимает %d байт ”, x, sizeof(x), y, sizeof(y));

return 0;

}

 

При вызове функции printf сначала вычисляются значения всех её аргументов - x, sizeof(x), y, sizeof(y) – а затем операционная система выводит их на экран.

 

Строка в языке С представляет собой массив, образованный следующими друг за другом переменными типа char. Они нумеруются целыми числами (индексами массива), начиная с нуля. Спецификация формата ввода-вывода строки - %s. Строка не обязательно должна занимать весь массив – ее конец отмечает элемент с нулевым значением.

 

/*prog13.c*/

 

#include<stdio.h>

 

main()

{

char str[20] = “I am a student”;

str[6] = 0;

printf(“%s \n”, str);

return 0;

}

 

Прежде всего, обратим внимание на объявление массива str. Сначала идёт имя типа элемента массива – char, затем имя массива – str и модификатор типа – квадратные скобки. В языке С квадратные скобки интерпретируются как операция доступа к элементу массива: str[6] = 0 – элементу массива str с индексом 6 присвоить значение 0. Объявление массива следует понимать так: выполнив с переменной str операцию доступа к элементу массива, мы получим значение типа char. Число в квадратных скобках при объявлении массива равно количеству его элементов; так как индекс начального элемента 0, то индекс последнего элемента массива равен 19. В строке “I am a student” 11 букв и три пробела – всего 14 символов, да еще символ с нулевым значением, завершающий строку – получается, что длина строки составляет 15 символов, последний байт массива не используется. В теле функции элементу с индексом 6 присваивается значение 0, следовательно, строка сокращается. Откомпилировав и выполнив программу prog5.c, мы увидим, что она выводит на экран лишь 6 символов строки: “I am a”.

 

Есть еще один способов объявления массива. На самом деле в переменной str хранится адрес начального элемента массива, т.е. str – это ссылка на переменную str[0].

 

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

 

/*prog14.c*/

 

#include<stdio.h>

 

main()

{

char *str = “I am a student”;

str[7]=‘S’;

printf(“%s \n”, str);

return 0;

}

 

После выполнения программы на экране появляется сообщение “I am a Student”. Таким образом, объявления

 

char str[20];

 

char *str;

 

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

 

В программах prog9.c и prog10.c мы присваивали строке значение при помощи инициализации, а затем изменяли ее отдельные символы. Использовать операцию = для изменения самой строки в теле функции нельзя, это делает специальная библиотечная функция. Зато при вводе строки не надо выполнять операцию ссылки:

scanf(“%s”, str);

 

/*prog15.c*/

 

/* Программа выводит на экран строку (без пробелов), которую ввёл пользователь */

 

#include <stdio.h>

#define N 1000

 

main()

{

char str[ N ];

scanf (“%s”, str);

printf(“%s \n”, str);

return 0;

}

 

Переменная str в программе prog15 является буфером – областью памяти для временного хранения введённой информации. Обычно объём требуемой памяти для буфера известен лишь приблизительно, поэтому его выделяют с некоторым запасом. Числовые значения, которые используются в программе, следует задавать с помощью директивы препроцессору define, в этом случае уменьшается вероятность ошибок в случае их изменения.

 

/*prog16.c*/

 

/* Программа подсчитывает количество символов в строке (без пробелов), которую ввёл пользователь */

 

#include<stdio.h>

#define N 1000

 

main()

{

unsigned int i = 0, count = 0; /* переменная i будет индексом массива символов (строки), а переменная count представляет собой счётчик, в ней будет накапливаться количество символов в строке, введённой пользователем */

 

char c, str[ N ];

scanf (“%s”, str);

 

c = str[ 0 ];

while (c!= 0)

count++;

 

printf(“%u \n”, count);

return 0;

}

 

Задания

 

Составьте программы, для решения следующих задач:

 

<9.1> на вход поступают строка (без пробелов) и символ, программа подсчитывет, сколько раз символ встречается в строке;

 

<9.2> на вход поступают строка (без пробелов) и два символа, программа определяет, какой из них встречается в строке чаще;

 

 

ПЕРЕДАЧА ПАРАМЕТРОВ В КОМАНДНОЙ СТРОКЕ

 

Вызывая компилятор, мы передавали ему имя файла с исходным текстом программы в виде аргумента командной строки. Точно так же мы можем передавать строки программам, написанным на языке С. Рассмотрим пример:

 

/*prog17.c*/

 

#include<stdio.h>

main(int argc, char* argv[])

{

printf(“Количество слов в командной строке равно %d \n”, argc);

if (argc > 1)

printf(“Первый аргумент программы - %s \n”, argv[ 1 ]);

else

printf (“Вы запустили программу %s без аргументов \n”, argv[0]);

return 0;

}

 

Теперь функция main вызывается с двумя аргументами, их передает программе операционная система. Традиционно они называются argc и argv. Аргумент argc относится к типу int, это – целое число. Значением аргумента argc служит количество слов в командной строке при вызове программы, не считая приглашения командной строки. Слова отделяются друг от друга пробелами и знаками табуляции (Tab), а не запятыми, точками или другими знаками препинания. Первое слово – имя исполняемого файла, следовательно, значение argc всегда больше нуля. Команда

 

printf(“Количество слов в командной строке равно %d \n”, argc);

 

выводит на экран значение аргумента argc. Если программа запущена так:

 

prog6.c aa bb cc

 

то значение argc должно быть равно 4.

 

Аргумент argv объявлен при помощи ключевого слова char и двух модификаторов типа - * и []:

 

сhar* argv[];

 

Символ * можно отделять от слов char, argv – пробелами или писать слитно с тем или другим словом. Напомним, что модификаторы типа интерпретируются как операции, но в отличие от бинарных операций (сложение, умножение), у которых два операнда, эти операции являются унарными – у них только один аргумент. В языке С операции доступа к элементу массива ([]) и разадресации (*) выполняются справа налево. Таким образом, если мы осуществляем доступ к элементу массива argv, то получим значение типа char* - строку символов. Отсюда следует, что argv – массив строк, т.е. каждый элемент массива представляет собой цепочку символов, которая завершается нулевым значением. Количество элементов массива не указывается – его определяет операционная система. Значениями элементов массива argv служат слова командной строки: argv[0] - имя исполняемого файла, argv [1] – первый аргумент и т.д., к каждому слову операционная система добавляет завершающий нуль. Команда

 

printf(“Первый аргумент программы - %s \n”, argv[1]);

 

выводит на экран первый аргумент программы - argv[1].

 

Принцип надежности работы программы требует, чтобы при запуске программы без аргументов предыдущая команда не выполнялась. Для этого мы используем управляющую конструкцию с ключевым словом if – если (ветвление). Программа проверяет истинность условия, в нашем случае – верно ли то, что argс > 1. Если условие истинно, то выполняется следующая за ней команда, если нет, и присутствует ключевое слово else – то выполняется команда, стоящей после этого слова. Затем управление передается команде, следующей за ветвлением. Команда

 

printf(“Вы запустили программу %s

без аргументов \n”, argv[0]);

 

выводит сообщение с указанием имени исполняемого файла.

 

Задание.

 

Составьте программу, которая выводит два первых аргумента, переданных ей при запуске слитно, раздельно и в столбик (один под другим). Не забудьте корректно обработать случай запуска программы с меньшим количеством аргументов.

 

МАТРИЦЫ

 

Матрицей называется квадратная или прямоугольная числовая таблица. Элементы матрицы объединяются в горизонтальные строки и вертикальные столбцы; в языке С строки и столбцы матрицы нумеруются целыми числами 0, 1, 2,.. Порядок матрицы представляют в виде m * n, где m – количество строк, n – количество столбцов. Элемент матрицы а, стоящий на пересечении i-той строки и j-того столбца обозначается символом a [ i ] [ j ]; если а – матрица порядка m * n, то i = 0, 1,.., m - 1, j = 0, 1,…, n - 1. Значения i и j называются индексами элемента a [ i ] [ j ].

Рассмотрим следующую программу:

 

/* matrix1

программа выделяет область памяти для матрицы а порядка 1000*1000; после запуска программы пользователь вводит значение элемента a [ 0 ] [ 0 ], которое затем выводится на экран*/

 

# include<stdio.h>

# define M 1000

# define N 1000

 

main()

{

int a [ M ] [ N ]; //объявляет матрицу а

int i = 0, j = 0;

scanf (“%d”,&(a [ i ] [ j ])); //читает значение элемента а [ i ] [ j ], i = j = 0

printf (“ %d\n”, a [ i ] [ j ]); // выводит на экран значение элемента а [ i ] [ j ], i = j = 0

return 0;

}

 

Количество строк и столбцов матрицы а определены при помощи директивы препроцессора # define; нулевые значения индексам i, j присвоены в ходе их объявления (напомним, что этот механизм называется инициализацией переменных).

 

Задание. Откомпилируйте и выполните программу matrix1.

 

Большие программы очень трудно понять, поэтому в них всегда присутствуют комментарии, содержащие необходимые разъяснения. Мы уже отмечали, что комментарии располагаются между специальными символами /* и */; если комментарий занимает часть строки, то перед ним ставят символ //, конец комментария в этом случае совпадает с концом строки.

 

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

 

/* matrix2

После запуска программы пользователь вводит количество строк и столбцов матрицы, а затем ее элементы; программа выводит на экран транспонированную матрицу (напомним, что операция транспонирования матрицы заменяет её строки столбцами)*/

 

// Директивы препроцессора

 

# include<stdio.h>

# define M 1000

# define N 1000

 

main()

{

// объявление переменных

int a [ M ] [ N ];

int m, n, i, j;

// программа читает количество строк и столбцов матрицы а

scanf (“ %d %d”, &m, &n);

//читает значения элементов матрицы а

for (i = 0; i < m; i++)

for (j = 0; j < n; j++)

scanf (“ %d”, &(a [ i ] [ j ]));

// выводит элементы матрицы на экран

for (j = 0; i < n; i++)

for (i = 0; j < m; i++)

printf(“ %d ”, a [ i ] [ j ]);

return 0;

}

 

Как и в программе matrix1, программа выделяет область памяти, в которой можно разместить матрицу порядка 1000*1000. предполагается, что значения m и n, которые введет пользователь, меньше, чем 1000. Таким образом, часть выделенной памяти не используется. С помощью более сложных механизмов языка С можно каждый раз выделять такую область памяти, в которой можно разместить матрицу порядка m*n.

Оператор цикла for (i = 0; i < m; i++) сначала присваивает переменной i значение 0, затем проверяет условие i < m. Если неравенство справедливо, выполняется команда, непосредственно следующую за конструкцией for. После этого оператор цикла увеличивает значение i на 1 (выполняет операцию i++, которая называется инкрементом), снова проверяет условие i < m и т.д. Как только условие i < m перестанет быть справедливым, выполнение команды, непосредственно следующей за конструкцией for, прекращается и управление передается команде, стоящей далее.

НЕ НАДО СТАВИТЬ ТОЧКУ С ЗАПЯТОЙ ПОСЛЕ КОНСТРУКЦИИ for!!!

Команда, непосредственно следующая за оператором for, также является циклом. С ее помощью программа повторяет команды – читает элементы строки матрицы а; когда строка прочитана, то внешний оператор цикла for (i = 0; i < m; i++) осуществляет переход к чтению следующей строки. Когда оба цикла завершатся, будут прочитаны все строки исходной матрицы. Точно также организован вывод элементов матрицы на экран.

 

Отметим, что команда, следующая за оператором цикла, образует с ним единый блок, поэтому ее набирают с отступом на 2-4 интервала.

 

Задание. Откомпилируйте и выполните программу matrix2

 

Выполнив программу, мы убедимся, что результат представлен на экране в виде одной числовой строки. Для того, чтобы на экран выводилась матрица в прямоугольном виде, необходимо вывод каждой строки завершать символом переноса ‘\n’

 

for (j = 0; j < m; j++)

{

for (i = 0; i < n; i++)

printf (“%d ”, a [ i ] [ j ]);

printf (“ \n ”);

}

 

(после %d в команде printf обязательно оставьте пробел, иначе числа будут выводиться без разделителей).

 

Задание. Дополните программу и выполните ее.

 

Как говорилось выше, оператор цикла for управляет выполнением непосредственно следующей за ним команды. В нашем случае он должен обеспечить выполнение двух команд: вывод элементов строки матрицы и переход на новую строку. Для этого обе команды объединяются в одну при помощи фигурных скобок {…}.

 

Язык С предназначен в первую очередь для разработки программ, которые выполняются на компьютерах, обладающих очень высокой производительностью. Здесь машинное время очень дорого, поэтому ввод данных осуществляется не с клавиатуры, а из файла. Составьте текстовый файл mydata.txt, содержащий следующие данные

 

3 4

1 2 3 4

4 3 1 2

17 18 19 20

 

Первые два числа задают количество строк и столбцов матрицы, а далее следует сама матрица. Откомпилируйте программу matrix2; пусть исполняемый файл называется matrix2.exe. Запустите программу, набрав в командной строке

 

matrix2.exe < mydata.txt

 

Символ < указывает операционной системе, что стандартный канал ввода исполняемого файла matrix2.exe следует соединить с файлом mydata.txt

 

Задание. Выполните программу так, чтобы исходные данные были прочитаны из текстового файла.

 

 

РАЗРАЬРТКА ПРОГРАММЫ

 

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

 

Пусть наша задача формулируется следующим образом: после запуска программы пользователь вводит матрицу порядка m * n; программа изменяет порядок элементов каждого столбца на противоположный.

 

Программы matrix1, matrix2 могут служить макетами. Следующий макет – программа matrix3

 

/* matrix3

после запуска программы пользователь вводит количество строк и столбцов матрицы, а затем ее элементы;

программа меняет местами первый и последний элементы каждого столбца и результат выводит на экран*/

 

// Директивы препроцессора

# include<stdio.h>

# define M 1000

# define N 1000

 

main()

{

// объявление переменных

int a [ M ] [ N ];

int m, n, i, j;

 

int buf;

//выделяется буфер для временного хранения промежуточных результатов

 

scanf(“%d % d”,&m, &n);

//программа читает количество строк и столбцов матрицы,

 

// программа читает элементы матрицы

for (i = 0; i < m; i++)

for (j = 0; j < n; j++)

scanf (“%d”,& (a [ i ] [ j ]));

 

// программа меняет местами первый и последний элемент каждого столбца

for (j = 0; j < n; j++)

 

{ buf = a [ 0 ] [ j ];

// значение первого элемента столбца помещаем в буфер buf

 

//значения первого и последнего элемента столбца j меняются местами

a [ 0 ] [ j ] = a [ m ] [ j ];

a [ m ] [ j ] = buf;

}

 

// программа выводит матрицу-результат на экран

for (i = 0; i < m; i++)

for (j = 0; j < n; j++)

printf (“%d ”, a [ i ] [ j ]);

return 0;

}

 

Задание. Откомпилируйте и выполните программу matrix3, проверьте правильность результата и внесите в её текст необходимые изменения.

 

Теперь мы можем составить завершающую программу-макет. Она меняет местами элементы a [ i ] [ j ] и a [ к ] [ j ], сначала i равно 0, а к равно n - 1, затем каждый раз значение i увеличивается на 1, а значение к уменьшается на 1 до тех пор, пока I будет меньше к.

НЕ ЗАБУДЬТЕ ОБЪЯВИТЬ НОВУЮ ПЕРЕМЕННУЮ к

 

…………………………………………………………

 

for (j = 0; j < n; j++)

for (i = 0; к = n - 1; i < к; i++, к--)

{ buf = a [ i ] [ j ];

a [ i ] [ j ] = a [ к ] [ j ];

a [ к ] [ j ] = buf;

}

 

Напомним, что операции, которые отделяются друг от друга запятой, в языке С образуют единое целое; таким образом, в начале цикла for переменной i присваивается значение 0, а переменной к - значение n - 1. После каждого прохода цикла значение переменной i увеличивается на 1, а значение переменной к уменьшается на 1 (выполняется операция к--, которая называется декрементом).

 

 

ОПТИМИЗАЦИЯ И ДОКУМЕНТИРОВАНИЕ ПРОГРАММЫ

 

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

 

Постановка задачи. На вход поступают два натуральных числа, a и b, программа находит все совершенные числа, заключённые между a и b.

 

В качестве имени файлов будем использовать аббревиатуру (сокращение слов) perfnum: по-английски совершенное число - perfect number.

 

Составление теста программы. Перебирая натуральные числа, замечаем, что число 6 является совершенным: 6 = 1 + 2+ 3. Файл, содержащий исходные данные:

 

 

a=2 b=10

 

назовём perfnum.txt

 

Результатом выполнения программы должно стать число 6.

 

Макеты программы

 

Макет № 1

 

/*perfnum1.c*/

/*программа читает данные, содержащиеся в тестовом файле и выводит их на экран*/

 

#include <stdio.h>

 

main()

{

int a,b;

scanf (" a=%d b=%d", &a, &b);

printf (" a=%d b=%d\n");

return 0;

}

 

Программа читает из входного потока содержимое файла. Команда запуска откомпилированной программы:

 

perfnum.txt > perfnum.exe

 

Макет № 2

 

/*perfnum2.c*/

/*программа читает числа a,b, содержащиеся в тестовом файле и выводит на экран все делители чисел a и b*/

 

#include <stdio.h>

 

main()

{

int a,b;

int j;

 

scanf (" a=%d b=%d", &a, &b);

printf ("divisors of a: \n")

for (j = 1; j < a; j++)

if (a % j == 0)

printf (" %d ", j);

printf (" /n divisors of b: \n")

for (j = 1; j < b; j++)

if (b % j == 0)

printf ("%d ", j);

return 0;

}

 

Программа вычисляет остатки от деления числа a на 1, 2,..., a-1. Если остаток от деления a на j равен нулю, то число j является делителем a, оно выводится на экран. Аналогично вычисляются делители числа b.

 

 

Макет № 3

 

/*perfnum3.c*/

/*программа читает числа a,b, содержащиеся в тестовом файле и выводит на экран все совершенные числа, заключённые между a и b*/

 

#include <stdio.h>

 

 

main()

{

int a, b;

int i, j, s;

 

scanf (" a=%d b=%d", &a, &b);

 

for (i = a; i <= b; i++)

{ // перебираем числа, заключённые между a и b

 

s = 0; //инициализация сумматора s

for (j = 1; j < i; j++)

if (i % j == 0) s+=j;

/* После завершения цикла значение переменной s равно сумме всех делителей числа i */

if (i == s)

printf ("%d ", i);

}

return 0;

}

 

Эта программа находит все совершенные числа, заключённые между a и b. Однако она обладает существенным недостатком. Программа, которая выполняется на компьютере, занимает его ресурсы - процессор, оперативную память, каналы ввода-вывода, файлы, внешние устройства и т.д. Основная адача разработчика - рациональное использование ресурсов. Обычно затраты ресурсов оцениваются относительно некоторого характерного параметра программы. Будем считать, что a=1, число b будет характерным параметром, обозначим его символом n. Для того, чтобы найти все совершенные числа, не превосходящие n, программа выполняет 1+2+...+n-1 проходов цикла: она перебирает числа от 1 до n и у k-го числа проверяет k-1 чисел, выбирая из них делители k;1+2+...+n-1 = n * (n – 1) / 2, следовательно, программа выполняет порядка n * n операций.

 

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

 

Проверим, является ли число 255 совершенным. Число 1 является делителем любого числа, поэтому сразу можно поместить в переменную s значение 1; 255 - нечётное число, следовательно, оно не имеет чётных делителей, так что будем искать делители только среди нечётных чисел. 255 делится на 3, значит оно имеет ещё один делитель – 255 / 3 = 85; s = 1+ 3 + 85 +... Cледующая пара делителей - 5 и 51: s = 1 + 3 + 85 + 5 + 51 +... Точно так же находим ещё два делителя - 15 и 17. Следующий делитель 17 у нас уже есть, перебор прекращается, s = 1 + 3 + 85 + 5 + 51 + 15 + 17 = 177 < 255, число 255 не является совершенным.

 

Теперь проверим, является ли совершенным число 16: это число чётное, оно делится на 2 и 4, 16 / 2 = 8, 16 / 4 = 4, два делителя совпадают, поиск делителей прекращается.

 

Обобщая наши исследования, можно сказать, что в том случае, когда число p является делителем числа q, то q делится без остатка на q / p, один из этих делителей не превосходит корня квадратного из q, а другой не меньше этого значения. Следующий макет программы выглядит так:

 

Макет № 4

 

/*perfnum4.c*/

/*программа читает числа a,b, содержащиеся в тестовом файле и выводит на экран все совершенные числа, заключённые между a и b*/

 

#include <math.h> /*заголовочный файл math.h содержит описания математических функций */

 

#include <stdio.h> /*файл содержит описания средств ввода-вывода*/

 

main()

{

int a, b;

int i, j, k,s;

int p, q;

float m;

 

scanf (" a = %d b = %d", &a, &b);

 

if (b < 6)

{

printf ("NO PERFECT NUMBERS\n");

return 0;

}

 

if (a % 2 ==0) // Если a чётное число

{ p = a; q = a+1; }

else

{ q = a; p = a +1; }

 

 

for (i = p; i <= b; i = i+2)

{ // перебираем чётные числа, заключённые между a и b

 

s = 1; //инициализация сумматора s

m = sqrt (i) + 1; // будем искать те делители числа i, которые не превосходят корня квадратного из i

 

 

for (j = 2; j <= m; j++)

if (i % j == 0)

{

s+= j;

 

if ((k = i / j) < j) s+= k;

}

/* После завершения цикла значение переменной s равно сумме всех делителей числа i */

 

if (i == s)

printf ("%d ", i);

}

 

 

for (i = q; i <= b; i = i + 2)

{ // перебираем нечётные числа, заключённые между a и b

 

s = 1; //инициализация сумматора s

 

m = sqrt (i) + 1; // будем искать те делители числа i, которые не превосходят корня квадратного из i

 

for (j = 3; j <= m; j = j + 2) /*ищем только нечётные делители*/

 

if (i % j == 0)

{

s+= j;

if ((k = i / j) < j) s+=k;

}

/* После завершения цикла значение переменной s равно сумме всех делителей числа i */

if (i == s)

printf ("%d ",i);

}

return 0;

}

 

Вычисление корня квадратного требует значительного времени, поэтому мы введём две целых переменных - m и sqm. Когда начинается выполнение программы, их значения равны, оответственно, (int)sqrt (a) + 1 и m * m. Как только i становится больше sqm, программа прибавляет к sqm 2 * m + 1 и увеличивает m на единицу. Таким образом, m равно наименьшему натуральному числу, квадрат которого больше i.

 

Макет № 5

 

/*perfnum5.c*/

/*программа читает числа a,b, содержащиеся в тестовом файле и выводит на экран все совершенные числа, заключённые между a и b*/

 

#include <math.h> /*файл содержит описания математических функций */

#include <stdio.h> /*файл содержит описания средств ввода-вывода*/

 

 

main()

{

unsigned int a,b;

unsigned int i, j, k,s;

unsigned int p,q;

unsigned int m, sqm;

 

scanf (" a = %d b = %d", &a, &b);

 

if (b < 6)

{

printf ("NO PERFECT NUMBERS\n");

return 0;

}

if (a % 2 ==0) // Если a чётное число

{ p=a; q=a+1; }

else

{ q=a; p=a+1; }

 

m = (int) sqrt (p) + 1; sqm = m * m;

 

for (i = p; i <= b; i = i+2)

{ // перебираем чётные числа, заключённые между a и b

s = 1; //инициализация сумматора s

 

for (j = 2; j <= m; j++)

if (i % j == 0)

{

s+= j;

if ((k = i / j) < j) s+= k;

}

/* После завершения цикла значение переменной s равно сумме всех делителей числа i */





Поделиться с друзьями:


Дата добавления: 2016-12-18; Мы поможем в написании ваших работ!; просмотров: 253 | Нарушение авторских прав


Поиск на сайте:

Лучшие изречения:

Логика может привести Вас от пункта А к пункту Б, а воображение — куда угодно © Альберт Эйнштейн
==> читать все изречения...

2285 - | 2212 -


© 2015-2025 lektsii.org - Контакты - Последнее добавление

Ген: 0.013 с.