Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Форум: "Потрепаться";
Текущий архив: 2004.05.16;
Скачать: [xml.tar.bz2];

Вниз

Задачка   Найти похожие ветки 

 
Кот-трахкун ©   (2004-04-22 20:52) [0]

Принесла одна малая (14 лет). Чешу репу над ней уже третий день.
Вот задачка:
Есть листбокс. В нем по вертикали расположено следующее:
1
1
1
0
0
0

Нужно переместить все единички по порядку вниз.
Типа "111000", "110100", "110010", "110001", "101100", "101010", "101001", "100110", "100101", "100011", "011100", "011010" и так до тех пор. пока все единички не окажутся внизу.
Решение должно быть наглядным. Т.е. с применением Sleep(100) и Application.ProcessMessages, чтоб было видно, как единички перемещаются.
Я уже перепробовал все: while, repeat, for, forfor, forforfor и сочетания вышеприведенного. Но так ничего толкового и не добился.
Пока в списке было две единички (110000), то с перемещением проблем не было. Но когда их больше - ничего не получается.
(смущенно озираясь): Ну как вам задачка?

Кот-трахкун. Затейник.
(aka AL2002)


 
Anatoly Podgoretsky ©   (2004-04-22 20:56) [1]

Только вместо ListBox для визуализации лучше применить 6 штук TLabel


 
Gero ©   (2004-04-22 20:59) [2]


> Только вместо ListBox для визуализации лучше применить 6
> штук TLabel

TForm.Canvas"a будет достаточно.

А это классическая задача на все возможные перестановки.


 
Nous Mellon ©   (2004-04-22 21:01) [3]


> TForm.Canvas"a будет достаточно.

Ты эта Подгорецкому что-ли советуешь? :)


 
Jack128 ©   (2004-04-22 21:02) [4]

for i := 0 to 2 do
 for j := 2 - i to 4 - i do
 begin
   ListBox1.Items[j] := "0";
   ListBox1.Items[j + 1] := "1";
   ListBox1.Refresh;
   Sleep(1000);
 end;


 
Gero ©   (2004-04-22 21:03) [5]


> Ты эта Подгорецкому что-ли советуешь? :)

А это разве ему надо?


 
Nous Mellon ©   (2004-04-22 21:07) [6]


> А это разве ему надо?

Так это был ответ на его пост.
Ну вы, блин, даете (с)
:)


 
Anatoly Podgoretsky ©   (2004-04-22 22:04) [7]

От автора требуется разъяснение, мне кажется что можно переставлять только пары 10 -> 01


 
Anatoly Podgoretsky ©   (2004-04-22 22:06) [8]

А про визуализацию, можно и на PaintBox отрисовывать все итерации, их все равно не более 36, в виде вертикальных столбиков.

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


 
Yanis ©   (2004-04-22 22:14) [9]

Хватит трепаться :) Это вам не ... Хотя.
Как всё - таки задачу то решить(не "с помощью чего", а "как", код всмысле :))?


 
Jack128 ©   (2004-04-22 23:05) [10]


> На мой взгляд задача визуально похожа на пузырьковую сортировку,
> крайний правый 1 падает вниз до предела.
нет, потому что после того как он упал, он вдруг снова наверх лезет :-)

> 111000", "110100", "110010", "110001", "101100", "101010",
> "101001


А полный перебор я вроде написал, но у меня перебирается не в том порядке в каком коту нужно.. :-(


 
Jack128 ©   (2004-04-22 23:14) [11]

Вот еще вариант. вроде работает нормально(правда визуально выглядит не так как дано в задании, но это мелочи)

procedure PolniyPerebor(s: string; sl: TStringList);
var
 i, j: integer;
 Temp: TStringList;
begin
 sl.Clear;
 if Length(s) = 1 then
   sl.Add(s)
 else
 begin
   Temp := TStringList.Create;
   try
     PolniyPerebor(copy(s, 1, Length(s) - 1), Temp);
     for i := 0 to Temp.Count - 1 do
     begin
       for j := 1 to Length(Temp[i]) do
         sl.add(copy(Temp[i], 1, j - 1) + s[Length(s)] + copy(Temp[i], j, Length(Temp[i])));
       sl.add(Temp[i] + s[length(s)]);
     end;
   finally
     Temp.Free;
   end;
 end;
end;

function SortProc(List: TStringList; Index1, Index2: Integer): Integer;
var
i, j: Integer;
begin
 i := StrToInt(List[Index1]);
 j := StrToInt(List[Index2]);
 Result := j - i;
end;

procedure TForm1.Button2Click(Sender: TObject);
var
 sl: TStringList;
 i, j: integer;
begin
 sl := TStringList.Create;
 try
   PolniyPerebor("111000", sl);
   // Удаляем дубликаты
   for i := 0  to sl.Count - 1 do
     for j := i + 1 to sl.Count - 1 do
       while (j < sl.Count) and (sl[i] = sl[j])  do
         sl.Delete(j);
   //--------------------------------------------------------------
   sl.CustomSort(SortProc);
   listbox1.Items.Assign(sl);
 finally
   sl.Free;
 end;
end;


 
Gero ©   (2004-04-22 23:21) [12]

Задача неясна тем, что
> Нужно переместить все единички по порядку вниз
несоответствует
> Типа "111000", "110100", "110010", "110001", "101100", "101010",
> "101001", "100110", "100101", "100011", "011100", "011010"


 
Кот-трахкун ©   (2004-04-22 23:28) [13]

Жек, пасиба, но типа немнога не таво... :(

Передвигать нуна командой Move (желательно), т.к. пример должен быть наглядным.

Мне тут Анатолий намекнул про рекурсию. Папробавать нада. Завтря, навена.

Может, завтра, со свежей башки чо-та мяукнеца.

Пасиба. :)


 
Jack128 ©   (2004-04-22 23:28) [14]


>  Gero ©   (22.04.04 23:21)
именно. Мой первый код "перемещает все единички по порядку вниз" , а второй перемещает их так как в примере написано..В общем завтра придет ал и все объяснит(я надеюсь)


 
Mihey ©   (2004-04-22 23:59) [15]

Вот мой код. Вроде всё по заданию. На форме кнопчак и 6 TEdit с названиями LabelX, чтобы было непонятней.


procedure TmainForm.btnSortClick(Sender: TObject);
label HereWeGoAgain;
const N = 6;
var a: array [1..6] of Byte;
   i, k: Integer;

 function IsDone(ar: array of Byte): Boolean;
  var t: Integer;
  begin
   Result := True;
   for t := N-1 downto 1 do
    begin
     If ((a[t] = 1) and (a[t+1] = 0)) then
      Result := False;
    end;
  end;

begin
 for i := 1 to N do
  begin
   a[i] := StrToInt(TEdit(mainForm.FindComponent("Label" + IntToStr(i))).Text);
  end;
 If IsDone(a) then
  begin
   ShowMessage("Good!");
   Exit;
  end;
 HereWeGoAgain:
 while not IsDone(a) do
  begin
    for i := N-1 downto 1 do
     begin
      If ((a[i] = 1) and (a[i+1] = 0)) then
       begin
        a[i] := 0;
        a[i+1] := 1;
        for k := 1 to N do
          begin
            TEdit(mainForm.FindComponent("Label" + IntToStr(k))).Text := IntToStr(a[k]);
          end;
        Application.ProcessMessages;
        Sleep(500);
        goto HereWeGoAgain;
       end;
     end;
  end;
 ShowMessage("Hmmm! Good!");
end;


 
Anatoly Podgoretsky ©   (2004-04-23 00:46) [16]

Jack128 ©   (22.04.04 23:28) [14]
Что я могу объяснить, это кот пусть объясняет, как именно надо перемещать, а то не все ясно с условием перемещения.
Все по порядку вниз, это обмен пар, но замечание ТИПА и пример не подверждают это.


 
Кот-трахкун ©   (2004-04-23 20:52) [17]

Надо было тему назвать не "Задачка", а "Задачка с тремя офигевшими".
Имею в виду, конечно, себя, а не вас, о Джек №128, Анатолий и всех остальных, которых я не знаю.


> а то не все ясно с условием перемещения.
> Все по порядку вниз, это обмен пар, но замечание ТИПА и
> пример не подверждают это.

Возможно, мой вопрос кажется заданным в весьма свободной литературной форме, и использует выражения, традиционно не свойственные суровому и краткому лексикону настоящего программера;)
Тем не менее, если вглядеться, можно ясно уловить смысл моего вопроса. Надеюсь, что кто-то из читающих этот топик
опытных мастеров приоткроет мне тайну личных и годами отточенных манипуляций с листбоксами и их методами, ибо скупая информация из книжки "Delphi 5 для профессионалов" ограничивается какой-то отмазкой по этому поводу.


> Mihey ©   (22.04.04 23:59) [15]

Mihey, при чем здесь шесть TEditов с кнопчаком?!
Вникнитесь в мою проблему. Есть ListBox. Есть шесть строчек "111000" (сокращенно LB111).
А теперь внимание! Мы двигаем третью по счету единичку на одну позицию вниз и получаем "110100". Не колеблясь, мы продолжаем двигать эту едииничку в самый низ списка. Вдруг - оригинальный ход - с последней строчки мы перемещаем эту единичку на четвертую позицию в списке, то есть - на одну больше, чем было изначально.
Улавливаете? Следите за моими руками.
Единичка под номером два ПЕРЕМЕЩАЕТСЯ! Следом за единичкой №3, которую мы пнули под зад на одну позицию вниз, единичка №2 тоже соскакивает под нашим чутким руководством на одну позицию ниже. Чтобы не свершилось насилие, единичка №3 спригивает вниз еще на одну позицию, дабы избежать надругательства со стороны номера второго.
Милиция, протокол, свидетели.
Как видим, жизнь в листбоксе бурлит и эволюция катится все-таки куда-то. Те, кто понял, могут остановится на этом месте и дальше не читать.

ЗЫ
Привет McSimmу. Почему он, подлец, не делает новые дайджесты?
Я хоть и не разбрасываюсь горстями лингвистического бисера по станицам этого форума, но все же почитываю его иногда. В оффлайне.

Тем, кто дочитал до этого места.
Как вы еще не поняли, единички в этом листбоксе, мягко говоря, должны перетекать в самый низ списка. Другими словами - мы, перемещая единички, факультативно перебираем все комбинации с тремя неизвестными. Брут-форс, ептыть.
ВВС, клонирование, стелсы, космические станции, которые бороздят просторы Атлантического океана... неужели в наш век не придуман алгоритм такого простого, на первый взгляд, перемещения?
Удивительно, но факт. Как по мне. А вам так не кажется?


 
Mihey ©   (2004-04-23 21:00) [18]

2 Кот-трахкун:

Бессмысленная фигня, тем не менее, офигенно простая в реализации. Цтой очередной. Счастья вам, живите хорошо.


 
Кот-трахкун ©   (2004-04-23 21:36) [19]

Бессмысленная. Согласен. Но отнюдь не фигня.
Внимательно прочитайте мой предыдущий постинг.

За пожелание спасибо.


 
Gero ©   (2004-04-23 21:43) [20]


> Кот-трахкун

Если Вы сами написали весь алгоритм, то в чем проблема?
В ListBox"e итемы перемещаются процедурой Items.Move.


 
VID ©   (2004-04-23 22:15) [21]

To Кот-трахкун: ну так я не понял, вот такой метод тебе подойдёт ?

procedure TForm1.Button1Click(Sender: TObject);
Var Steps,I,X:Integer;
   B:Boolean;
begin
 X:=0;
 For I:=0 to ListBox1.Items.Count - 1 do //Узнаём количество единичек в списке
   IF ListBox1.Items[I] = "1" then Inc(X);
 IF X = 0 then exit;
 //Узнаём количество проходов
 Steps := ListBox1.Items.Count - X;
 //Начинаем пошаговую перестановку
 For I := 1 to Steps do
 begin
   B := False; //Если B = false то значит ещё не втречен ни один 0
   For X := ListBox1.Items.Count - 1 downto 0 do
   begin
     IF ListBox1.Items[X] <> "1" then B := True;
     IF (ListBox1.Items[X] = "1") and B then
     begin
       ListBox1.Items.Move(X, X+1);
       Sleep(1000);
       Refresh;
     end;
   end;
 end;
end;

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


 
Mihey ©   (2004-04-23 23:11) [22]

2 Кот-Трахкун:

Ок, еле впетрил, что тебе надо. Дубль 2. Кидаем на форму TListBox с названием ListBox и кнопчак, которому пишем в обработку:

procedure TmainForm.btnSortClick(Sender: TObject);
var  Coord: array [1..3] of Byte;
    WorkWith: Byte;
    Stators: array [1..3] of Byte;
    Stators222: array [1..3] of Byte;
    s: Boolean;

 procedure OutArray;
 var q: Integer;
 begin
   ListBox.Items.Clear;
   for q := 1 to 6 do
    If (q <> Coord[1]) and (q <> Coord[2]) and (q <> Coord[3]) then
     ListBox.Items.Add("0") else ListBox.Items.Add("1");
 end;

 function DoMove(What: Byte; Show: Boolean): Boolean;
 begin
  // True = достигли кончиты
  Result := False;
  If Coord[What] < 6 then
   begin
    Coord[What] := Coord[What] + 1;
    If Show then
     begin
      OutArray;
      Application.ProcessMessages;
      Sleep(2000);
     end;
    end
     else Result := True;
 end;

begin
 // We have 6
 Coord[1] := 1;
 Coord[2] := 2;
 Coord[3] := 3;

 // Работаем с третьим
 WorkWith := 3;

 while (coord[1] <> 4) and (coord[2] <> 5) and (coord[3] <> 6) do
  begin
   Stators222[1] := Coord[1];
   Stators222[2] := Coord[2];
   Stators222[3] := Coord[3];
    while (coord[2] <> 5) and (coord[3] <> 6) do
     begin
      Stators[1] := Coord[1];
      Stators[2] := Coord[2];
      Stators[3] := Coord[3];
       s := False;
        While not s do
         begin
          s := DoMove(WorkWith, True);
         end;
      Coord[1] := Stators[1];
      Coord[2] := Stators[2];
      Coord[3] := Stators[3];
      DoMove(WorkWith, False);
      WorkWith := WorkWith-1;
      DoMove(WorkWith, True);
      WorkWith := WorkWith+1;
     end;
   // с первой
   Coord[1] := Stators222[1];
   Coord[2] := Stators222[2];
   Coord[3] := Stators222[3];
   DoMove(WorkWith, False);
   WorkWith := WorkWith-1;
   DoMove(WorkWith, False);
   WorkWith := WorkWith-1;
   DoMove(WorkWith, True);
   WorkWith := WorkWith+2;
  end;

 ShowMessage("I""m a big bad niggar!");
end;

Компилим, жмём буттон, всё колкой идёт. Проверь сам. За отстутсвие Items.Move чур не винить - это легко сделать в процедуре Move. Ну что, то это или не то???????


 
Mihey ©   (2004-04-23 23:13) [23]

В догонку. Sleep убавь до 500, а то заснёшь...


 
Кот-трахкун ©   (2004-04-23 23:57) [24]

VID, я проливаю водопады слез, но твой код не работает. :(

Mihey, я офигел. Твой пример заработал.
По прошествии Sleep(50000) мы с монитором на пару тацевали джигу. Танцевали на протяжении примерно 150000 миллисекунд.

Теперь мне осталось чесать репу, как теперь суда втулить метод Move вместо генерации единичек-ноликов.

А кнопку я так и назвал - Mihey (procedure TForm1.MiheyClick(Sender: TObject);)
в честь одного оченно умного чела (не буду показывать пальцем, кого).

Сенькьсь.

Є(0.0)Э



Страницы: 1 вся ветка

Форум: "Потрепаться";
Текущий архив: 2004.05.16;
Скачать: [xml.tar.bz2];

Наверх




Память: 0.53 MB
Время: 0.037 c
8-1077816064
evrey
2004-02-26 20:21
2004.05.16
Получение цвета пикселя в изображении.


11-1069436539
DDA
2003-11-21 20:42
2004.05.16
KOL SetFileTime?


14-1083141223
Andryk
2004-04-28 12:33
2004.05.16
Головоломка для автомобилистов :))))


14-1083088406
УНЯ
2004-04-27 21:53
2004.05.16
ЕСЕМЕСки


14-1083030908
Думкин
2004-04-27 05:55
2004.05.16
С днем рождения! 27 апреля.





Afrikaans Albanian Arabic Armenian Azerbaijani Basque Belarusian Bulgarian Catalan Chinese (Simplified) Chinese (Traditional) Croatian Czech Danish Dutch English Estonian Filipino Finnish French
Galician Georgian German Greek Haitian Creole Hebrew Hindi Hungarian Icelandic Indonesian Irish Italian Japanese Korean Latvian Lithuanian Macedonian Malay Maltese Norwegian
Persian Polish Portuguese Romanian Russian Serbian Slovak Slovenian Spanish Swahili Swedish Thai Turkish Ukrainian Urdu Vietnamese Welsh Yiddish Bengali Bosnian
Cebuano Esperanto Gujarati Hausa Hmong Igbo Javanese Kannada Khmer Lao Latin Maori Marathi Mongolian Nepali Punjabi Somali Tamil Telugu Yoruba
Zulu
Английский Французский Немецкий Итальянский Португальский Русский Испанский