IPB
ЛогинПароль:

> Прочтите прежде чем задавать вопрос!

1. Заголовок темы должен быть информативным. В противном случае тема удаляется ...
2. Все тексты программ должны помещаться в теги [code=pas] ... [/code], либо быть опубликованы на нашем PasteBin в режиме вечного хранения.
3. Прежде чем задавать вопрос, см. "FAQ", если там не нашли ответа, воспользуйтесь ПОИСКОМ, возможно такую задачу уже решали!
4. Не предлагайте свои решения на других языках, кроме Паскаля (исключение - только с согласия модератора).
5. НЕ используйте форум для личного общения, все что не относится к обсуждению темы - на PM!
6. Одна тема - один вопрос (задача)
7. Проверяйте программы перед тем, как разместить их на форуме!!!
8. Спрашивайте и отвечайте четко и по существу!!!

 
 Ответить  Открыть новую тему 
> Проблемы с Procedure, проблема с передачей чисел из процедуры в программу
сообщение
Сообщение #1





Группа: Пользователи
Сообщений: 2
Пол: Мужской
Реальное имя: Саша

Репутация: -  0  +


Имеется задача: создать 3 двумерных массива и найти произведение столбцов массивов. Произведение записать в одномерный массив и вывести результат под соответсвующими столбцами. Решить с помощью Procedure.
Program Laba_14;
const
n=20;
type  z=array [1..n,1..n] of integer;
      u=array [1..n] of integer;
Var   a,b,c,i,j,k:integer;
      H:z;
      V:u;
Procedure Up(B:z;n:integer; var A:u);
var i,j,k:integer;
begin
k:=1;
for i:=1 to n do begin
                 for j:=1 to n do begin
                                  k:=k*B[i,j];
                                  end;
                 A[i]:=k;
                 k:=1;
                 end;
end;
Procedure Down(n:integer; Var b:z);
var i,j:integer;
begin
for i:=1 to n do begin
                 for j:=1 to n do begin
                                  B[j,k]:=random(9);
                                  end;
                 end;
end;
begin
randomize;
Write('n=');
readln(a);
for i:=1 to 3 do begin
                 Down(a,H);
                 For j:=1 to a do begin
                                  for k:=1 to a do begin
                                                   Write(H[k,j]:2);
                                                   end;
                              writeln;
                              end;
                 writeln;
                 Up(H,a,V);
                 for c:=1 to a do begin
                                  write(V[c]:2);
                                  end;
                 writeln;
                 writeln('-------------------');
                 end;
readln;
end.


Программа выполняется правильно, но проблема в том, числа массивов, созданные рандомно, неправильно передаются в основную программу, хотя в самой процедуре они нормальные. Под "неправильно" передаются я имею ввиду то, что:
1) числа в программе отличаются от чисел в процедуре
2) при передаче чисел большинство строк остаются заполненные нулями ( во все 3 массивах только одна из строк не равна нулю), хотя при проверке в процедуре они таковыми не являются.
Прошу помощи smile.gif
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
сообщение
Сообщение #2


Гость






Цитата
проблема в том, числа массивов, созданные рандомно, неправильно передаются в основную программу, хотя в самой процедуре они нормальные.
А ты попробуй в опциях компилятора включить "Range Checking", увидишь, какие они нормальные smile.gif У тебя ж вылет за границы массива происходит.

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


Уникум
*******

Группа: Пользователи
Сообщений: 6 823
Пол: Мужской
Реальное имя: Лопáрь (Андрей)

Репутация: -  159  +


volvo сказал самое важное, я добавлю немного..

1. ошибка - в индексах массива при заполнении Random;
2. кошмарное форматирование - кто сказал, что сдвигать надо под слово begin?? Сдвигать надо на фиксированное число позиций (выбери себе - 4 или 2 обычно рулят). То же самое с var, type и const.
3. единичный оператор совершенно необязательно (и часто вредно) заключать в begin/end.
4. сделай печать по формату :5 или еще больше, иначе все налезает друг на дружку..
5. процедуру Randomize добавляй только после отладки.


--------------------
я - ветер, я северный холодный ветер
я час расставанья, я год возвращенья домой
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
сообщение
Сообщение #4


Профи
****

Группа: Пользователи
Сообщений: 652
Пол: Мужской
Реальное имя: Алексей

Репутация: -  20  +


Скажу немного попроще,наверно.
 
Procedure Down(n:integer; Var b:z);
var i,j:integer;
begin
for i:=1 to n do begin
                 for j:=1 to n do begin
                                  B[j,k]:=random(9);
                                  end;
                 end;
end;


Скажем так ты задумайсяу тебя счетчик по i,j,а "K" не меняется,зачем оно тебе вообще нужно.Более того у тебя не обьявленно начальное значиние "к",паскаль любит запихивать туда всякий мусор который тебя потом можеш удивить.Насчет построения бы бы это сделал что то вроде:
 
Procedure Down(n:integer; Var b:z);
var i,j:integer;
 begin
    for i:=1 to n do 
      for j:=1 to n do 
         B[i,j]:=random(9);
  end;


В процедуре Up ты повторно обьявляеш переменную "K",зачем? Она же у тебя обьявлена в самой программе.Еще как совет,обычно S -сумирование,а P-произведение.
 
Procedure Up(B:z;n:integer; var A:u);
var i,j,P:integer;
begin
for i:=1 to n do 
 begin
  k:=1;
  for j:=1 to n do 
     begin
     P:=P*B[j,i];{как я понимаю тебе надо все элементы в столбше перемножить,а ты меремножаеш все элементы строки}
     end;
   A[i]:=k;
   end;
end;


А вообще перед тем, как прогу отлаживай проверь сначала математическую часть.И еще процедуры должны быть говорящие,а то хрен поймеш что за апы и дауны,когщда до списков дойдеш будеш так голову вскрывать... wacko.gif yes2.gif
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
сообщение
Сообщение #5





Группа: Пользователи
Сообщений: 2
Пол: Мужской
Реальное имя: Саша

Репутация: -  0  +


Цитата

2. кошмарное форматирование - кто сказал, что сдвигать надо под слово begin?? Сдвигать надо на фиксированное число позиций (выбери себе - 4 или 2 обычно рулят). То же самое с var, type и const.
3. единичный оператор совершенно необязательно (и часто вредно) заключать в begin/end.

Так форматирую, потому что привык когда после begin начинается "программа", сразу видно что находится между. Учту smile.gif
А пишу везде begin и end, потому что часто туда приходится чего-нибудь добавлять.

Цитата

Procedure Up(B:z;n:integer; var A:u);
var i,j,P:integer;
begin
for i:=1 to n do 
 begin
  k:=1;
  for j:=1 to n do 
     begin
     P:=P*B[j,i];{как я понимаю тебе надо все элементы в столбше перемножить,а ты меремножаеш все элементы строки}
     end;
   A[i]:=k;
   end;
end;



К - перемножает элементы столбца и передает их массиву, после чего его нужно снова его сделать равным 1. Тут без разницы какой буквой писать. Зачем тогда вообще заменять К на Р лишь в цикле? Нужно заменить либо везде, либо вообще не заменять blink.gif А иначе массив А всегда будет равен 1, а Р будет равно произведению всех элементом.
А то, что я потворно объявил К ни на что не влияет, это переменная процедуры, а не всей программы. Такой логикой можно было не объявлять i и j. Так что здесь все правильно.

Цитата

Скажем так ты задумайсяу тебя счетчик по i,j,а "K" не меняется,зачем оно тебе вообще нужно.Более того у тебя не обьявленно начальное значиние "к",паскаль любит запихивать туда всякий мусор который тебя потом можеш удивить.


Описался - вот где проблема была. Да и с индексами промахнулся smile.gif Спасибо за помощь
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 
сообщение
Сообщение #6


Профи
****

Группа: Пользователи
Сообщений: 652
Пол: Мужской
Реальное имя: Алексей

Репутация: -  20  +


Ну насчет,того, что я заменил k на p,там уже я описался, там везде надо было заменить на р ,да и ты прикрасно понял,плюс, если ты посмотриш я на 1 присваивание совершаю меньше(убралось присваивание вне цикла).Насчет begin-end просто иногда может случиться что у тебя в конце процедуры соберется 5 а то и более end,поэтому,если поправила не забывать,то можно и в некоторых местах begin-end убирать.

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

Сообщение отредактировано: Krjuger -
 Оффлайн  Профиль  PM 
 К началу страницы 
+ Ответить 

 Ответить  Открыть новую тему 
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 



- Текстовая версия 16.04.2025 15:38
500Gb HDD, 6Gb RAM, 2 Cores, 7 EUR в месяц — такие хостинги правда бывают
Связь с администрацией: bu_gen в домене octagram.name