Главная страница
    Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2006.06.18;
Скачать: CL | DM;

Вниз

Как создать алгоритм возвращения поряд. номера слова в строке.   Найти похожие ветки 

 
Dr. Andrew   (2006-05-10 15:23) [0]

Мастера, пожалуйста, подскажите алгоритм возвращения порядкового номера слова в строке (для меня желательно в WideString). Как можно создать такой алгоритм, например, function GetNumWord(WStr : WideString; Word : Widestring) : Integer, где WStr - строка в которой находится слово Word. И как быть, если это слово в одной и той же форме может повторяться в строке? Нужен последовательный порядковый номер слова в строке, например - "эта строка содержит слова и эти слова в этой строке." Здесь слова "слова" имеют порядковые номера 4 и 7. Всем спасибо, особенно за конкретный пример или ссылку на конкретный алгоритм.


 
Сергей М. ©   (2006-05-10 15:34) [1]


> Как создать алгоритм возвращения поряд. номера слова в строке


Каком кверху.

ТЗ готовь.


 
Dr. Andrew   (2006-05-10 15:42) [2]

Сергей, от первого слова в строке или тексте до последнего необходимо восстанавливать его порядковый номер.


 
Dstr ©   (2006-05-10 16:04) [3]

//Разделить строку S на слова используя разделительный символ Delimeter
//и поместить результат в TStrings
procedure StrBreakApart(const S, Delimeter: string; Parts: TStrings);
var
 CurPos: integer;
 CurStr: string;
begin
 Parts.clear;
 Parts.BeginUpdate();
 try
   CurStr := S;
   repeat
     CurPos := Pos(Delimeter, CurStr);
     if (CurPos > 0) then
     begin
       Parts.Add(Copy(CurStr, 1, Pred(CurPos)));
       CurStr := Copy(CurStr, CurPos + Length(Delimeter),
         Length(CurStr) - CurPos - Length(Delimeter) + 1);
     end
     else
       Parts.Add(CurStr);
   until CurPos = 0;
 finally
   Parts.EndUpdate();
 end;
end;

//поиск слова ета функция написана мной сейчас "на скорую руку"
procedure SelWord(Where: String;What: Integer): String;
Var Tmp: TStringList;
Begin;
Tmp := TStringList.Create;
StrBreakApart(Where, " ", Tmp);
Result := Tmp.Strings[What + 1];
End;

//Если у тебя многострочный текст - Создаешь временую строку,помешаешь туда текст и заменяешь \n\r или \r\n на пробелы и реализуешь то что я написал


 
Dr. Andrew   (2006-05-10 16:06) [4]

Спасибо Dstr! Сейчас попробую. А можно с Вами держать связь по e-mail?


 
Dstr ©   (2006-05-10 16:12) [5]

Не,лучше ICQ 209-298-286


 
Dr. Andrew   (2006-05-10 16:20) [6]

Dstr.

Функция SelWord возвращает слово по его порядковому номеру и то неверно, почему-то, а у меня совсем другая задача. Возвращать порядковый номер слова в тексте, например - "Это мой текст и я его написал.", где "Это" - 1, "мой" - 2, "текст" - 3 ... "написал" - N.


 
Dr. Andrew   (2006-05-10 16:22) [7]

Спасибо. ICQ не имею и не уважаю, как программу. Извините - у каждого свои пристрастия. :-) Может тогда посмотрите на мою задачу с нового угла зрения. Номер слова нужен по тексту, а не слово в тексте по номеру.


 
Jeer ©   (2006-05-10 16:40) [8]

Dr. Andrew   (10.05.06 16:20) [6]

определись с понятием "слово" и "разделитель(и)" слов.
Все сразу станет на место.
Позиция символа в строке - pos()


 
Dstr ©   (2006-05-10 16:48) [9]

ААааа,ну ето эшо проще:

function GetWord(Str, Smb: string; WordNmbr: Byte): string;
var
 SWord: string;
 StrLen, N: Byte;
begin

 StrLen := SizeOf(Str);
 N := 1;

 while ((WordNmbr >= N) and (StrLen <> 0)) do
 begin
   StrLen := Pos(Smb, str);
   if StrLen <> 0 then
   begin
     SWord := Copy(Str, 1, StrLen - 1);
     Delete(Str, 1, StrLen);
     Inc(N);
   end
   else
     SWord := Str;
 end;

 if WordNmbr <= N then
   Result := SWord
 else
   Result := "";
end;


//А вот так используйте   GetWord("Здесь ваш текст"," ",3); // Возвращает -> "текст"

А вот насчет ICQ ето зря,ведь RealTime куда лучше чем етот Email...Я сам в протоколе разбырался аси,он кривой конечно,но зато работает,и работает быстро.В прогах я всегда уважаю быстроту больше красотулек...


 
Dr. Andrew   (2006-05-10 17:11) [10]

Нет, Вы не совсем правильно истолковали вопрос. Да, GetWord действительно возвратит третье слово из строки или текста. Но мне неоходимо, например в обработчике события onMouseDown, чтобы при щелчке на слове "текст" я получил как результат этой функцииего порядковый номер, то есть число 3. теперь более понятнее я написал. А функция Pos мне не годится, потому что она возвращает символы слова, а не порядковый номер слова. Спасибо всем за участие и жду новых решений.


 
Jeer ©   (2006-05-10 17:14) [11]

Dr. Andrew   (10.05.06 17:11) [10]


> и жду новых решений.


А самому поработать ?


 
Dr. Andrew   (2006-05-10 17:22) [12]

С огромным удовольствием, если бы было над чем.


 
Dr. Andrew   (2006-05-10 17:22) [13]

С огромным удовольствием, если бы было над чем.


 
Dr. Andrew   (2006-05-10 17:27) [14]

Jeer! Я ведь обращаюсь к профессионалам, мастерам. Для того это и форум, чтобы найти мысль, на которую тебя натолкнут. Пока такой мысли нет. И я не профессиональный программист, а детский врач. И необходимость в коде по моей работе (это долго объяснять). У врачей теперь тоже есть компьютеры и они немного программируют для себя и своих пациентов. Так что может кто-то из мастеров все же поможет отечественной науке?


 
Dstr ©   (2006-05-10 17:31) [15]


> Нет, Вы не совсем правильно истолковали вопрос. Да, GetWord
> действительно возвратит третье слово из строки или текста.
>  Но мне неоходимо, например в обработчике события onMouseDown,
>  чтобы при щелчке на слове "текст" я получил как результат
> этой функцииего порядковый номер, то есть число 3. теперь
> более понятнее я написал. А функция Pos мне не годится,
> потому что она возвращает символы слова, а не порядковый
> номер слова. Спасибо всем за участие и жду новых решений


Ой,извини я сегодня чето тормажу,
Тогда используешь также StrBreakApart и в цикле сравнивай Строки Из TstringList С твоим словом...


 
Jeer ©   (2006-05-10 17:35) [16]

Ок.
В тексте объемом с "Война и мир" слово война встречается 515 раз и каждое из них имеет свою позицию, т.е. имее 515 чисел.
Вам в каком виде надо это представить и кому ?


 
Dstr ©   (2006-05-10 17:54) [17]

А ты блин прав,Токгда мош ето подойдет?:

function Seps(As_Arg: Char): Boolean;
begin
 Seps := As_Arg in
   [ #0..#$1F, " ", ".", ",", "?", ":", ";", "(", ")", "/", "\"];
end;

function WordCount(CText: string): Longint;
var
 Ix: Word;
 Work_Count: Longint;
begin
 Work_Count := 0;
 Ix         := 1;
 while Ix <= Length(CText) do
 begin
   while (Ix <= Length(CText)) and (Seps(CText[Ix])) do
     Inc(Ix);
   if Ix <= Length(CText) then
   begin
     Inc(Work_Count);

     while (Ix <= Length(CText)) and (not Seps(CText[Ix])) do
       Inc(Ix);
   end;
 end;
 Word_Count := Work_Count;
end;

Function NumWorld(S: String; Pos: Integer): Longint;  //Ну если ето не поможет,то не знаю,Нужно конечно указать позицию курсора в Pos
Begin;
Delete(S, Pos, Length(S) - Pos);
Result := WordCount(S);
End;


 
Dr. Andrew   (2006-05-10 18:00) [18]

Сейчас попробую. Необходимо этот алгоритм в поиской системе по историям болезни детей со злокачественными заболеваниями крови. Вот мне то и надо по аналогии с текстом  "Война и мир" все 515 порядковых номеров слова "война" при выделении его по тексту. Спасибо за помощь. Может еще какая мысль возникнет - не откажите записать ее на эту страницу.


 
Dstr ©   (2006-05-10 18:00) [19]

Хотя проще вместо Delete использовать SetLength,Я токо про нее вспомнил :)


 
Dr. Andrew   (2006-05-10 18:01) [20]

А что надо конкретно изменить и где?


 
Dstr ©   (2006-05-10 18:06) [21]

procedure TForm1.Button1Click(Sender: TObject);
begin
 if OpenDialog1.Execute then
   Memo1.Lines.LoadFromFile(OpenDialog1.FileName);
end;

procedure TForm1.Button2Click(Sender: TObject);
var
 find: string;
 text: string;
 st, len: integer;
 res: integer;
begin
 if Memo1.SelStart >= Length(Memo1.Text) then
   Memo1.SelStart := 0;
 st := Memo1.SelStart + 1;
 if (Memo1.SelLength <= 0) or (not CheckBox1.Checked) then
 begin
   inc(st, Memo1.SelLength);
   len := Length(Memo1.Text) - st;
 end
 else
   len := Memo1.SelLength;
 text := copy(Memo1.Text, st, len);
 find := Edit1.Text;
 res := pos(find, text);
 if res = 0 then
 begin
   ShowMessage("Search string "" + find + "" not found");
   Exit;
 end;
 Memo1.SelStart := res + st - 2;
 Memo1.SelLength := length(find);
end;

Вот эшо статья из delphiWorld"Как реализовать поиск по тексту"
Разбераться я в ней не буду - я и так забыл про свой проект блин,и даю тебе возможность разобраться самому


 
Dr. Andrew   (2006-05-10 18:14) [22]

Word_Count - компиллятор пишет ошибка. Функция в целом (NumWorld) не хочет работать. В качестве параметра Pos: Integer в ней я ставлю параметр Х из обработчика события onMouseDown (RichEdit). Что не верно?


 
Jeer ©   (2006-05-10 18:20) [23]


> Необходимо этот алгоритм в поиской системе по историям


Задача все усложняется.
Не вдаваясь в подробности по организации такой поисковой системы, решение может быть таким:
- есть куча текстовый файлов с описаниями
- загружаем один в мемо
- выделяем мышкой нужное слово и копируем в переменную s: string
- далее, составляем список оставшихся файлов, перебираем файлы и читаем
каждый построчно
- каждая строка "бьется" на слова и среди них ищется искомое
- каждый удачный поиск добавляет новую позицию в динамический массив arPos array of integer, предварительно увеличивая его длину на 1 с помощью SetLength(arPos, Length(arPos) + 1)


 
Loginov Dmitry ©   (2006-05-10 18:45) [24]

Может еще какая мысль возникнет - не откажите записать ее на эту страницу.

Мысля 1: оставить Delphi в покое, и пользоваться известными средствами: MS Word, Notepad, и т.д. Там кстати даже поиск реализован.

Мысля 2: см [11]


 
Loginov Dmitry ©   (2006-05-10 18:46) [25]

Dr. Andrew   (10.05.06 18:00) [18]
Может еще какая мысль возникнет - не откажите записать ее на эту страницу.


Мысль 1: Оставить Delphi в покое, и пользоваться известными средствами: MS Word, Notepad (там кстати даже поиск есть).

Мысль 2: см [11]


 
Dr. Andrew   (2006-05-10 18:56) [26]

Спасибо. А в практической реализации алгоритма? Есть еще решения? Может где-то выложен подобный алгоритм?


 
Dstr ©   (2006-05-10 18:57) [27]


> Word_Count - компиллятор пишет ошибка. Функция в целом (NumWorld)
> не хочет работать. В качестве параметра Pos: Integer в ней
> я ставлю параметр Х из обработчика события onMouseDown (RichEdit).
>  Что не верно?


Правильно блин - Он тебе курсор мыши и выдаст блин по отношению к форме, ща подумаю че можно придумать чтобы определить позицию мерцающего курсора,или у вас выделение?


 
Dr. Andrew   (2006-05-10 18:58) [28]

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


 
Loginov Dmitry ©   (2006-05-10 19:10) [29]

Dr. Andrew   (10.05.06 18:56) [26]
Может где-то выложен подобный алгоритм?


Какой алгоритм? Что вам нужно, объясните толком, и что вы будете делать с тем, что получится?


 
Dstr ©   (2006-05-10 19:12) [30]

1.Попробуй использовать мою функцию,вводя целые числа
2.Если получится используй Api функцию GetCurrentPosition


 
Dr. Andrew   (2006-05-10 19:18) [31]

Loginov Dmitry
Алгоритм возвращения порядкового номера слова в тексте. Например в тексте есть 10 слов. В возрастающем порядке каждое из них имеет свой порядковый номер. Вот мне -то его и необходимо возвратить.

Dstr
Пробую, но пока ничего не выходит. Да и мне необходимо кк-то оттолкнуться от обработчика события onMouseDown или onMouseUp.


 
Dstr ©   (2006-05-10 19:27) [32]

Можно тупо отсеч текст после курсора и подсчитать количество пробелов


 
Dr. Andrew   (2006-05-10 19:36) [33]

Dstr

Что-то вроде получается.
А как реализовать правильно Вашу мысль - "Хотя проще вместо Delete использовать SetLength,Я токо про нее вспомнил":

Function NumWorld(S: String; Pos: Integer): Longint;
Begin;
Delete(S, Pos, Length(S) - Pos);
Result := WordCount(S);
End;


Как правильно будет выглядеть тогда код функции NumWorld?
Спасибо всем Мастерам за помощь и Вам в частности!


 
Dstr ©   (2006-05-10 19:46) [34]

Кстати я думаю в вашей теме понадобится статейка:
Текст выглядит лучше, если он выровнен по двух краям. Для этого пробелы в каждой строке нужно удлинять или укорачивать так, чтобы все строки имели одну длину.

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

Эта программа выводит на экран текст из файла C:\text.txt, выравнивая его по двум краям.

type
 ...
 TLine = record
   s: string;
   wrap: boolean;
   length: integer;
end;

var
 Form1: TForm1;

implementation

{$R *.DFM}

const
 FileName = "C:\text.txt";

var
 s: string;
 bm: TBitMap;
 LineH: integer;
 MaxTextWidth: integer;

procedure TForm1.FormCreate(Sender: TObject);
var
 F: TFileStream;
 buf: array [0..127] of char;
 l: integer;
begin
 ScrollBar1.Kind := sbVertical;
 bm := TBitMap.Create;
 with bm.Canvas.Font do
 begin
   name := "Serif";
   Size := 12;
 end;
 LineH := bm.Canvas.TextHeight("123");

 if not FileExists(FileName) then
 begin
   ShowMessage("Can not find file " + FileName);
   Exit;
 end;
 F := TFileStream.Create(FileName, fmOpenRead);
 repeat
   l := F.read(buf, 128);
   if l = 128 then
     s := s + buf
   else
     s := s + copy(buf, 1, l);
 until
   l < 128;
 F.Destroy;
end;

procedure TForm1.FormResize(Sender: TObject);
begin
 PaintBox1.Left := 0;
 PaintBox1.Top := 0;
 PaintBox1.Height := Form1.ClientHeight;
 PaintBox1.Width := Form1.ClientWidth - ScrollBar1.Width;
 ScrollBar1.Left := PaintBox1.Width;
 ScrollBar1.Top := 0;
 ScrollBar1.Height := PaintBox1.Height;
 bm.Width := PaintBox1.Width;
 bm.Height := PaintBox1.Height;
 ScrollBar1.Max := 1000;
 MaxTextWidth := PaintBox1.Width - 20;
end;

function RealTextWidth(s: string): integer;
var
 i: integer;
begin
 result := bm.Canvas.TextWidth(s);
 for i := 1 to Length(s) do
   if s[i] = #9 then
     inc(result, 40 - bm.Canvas.TextWidth(#9));
end;

function GetLine(index: integer): TLine;
var
 i: integer;
 s1: string;
 first: integer;
begin
 if (s[index] = #13) and (s[index + 1] = #10) then
 begin
   result.s := "";
   result.length := 2;
   result.wrap := true;
   Exit;
 end;
 first := index;
 while (first <= Length(s)) and (s[first] in [ #32]) do
   inc(first);
 i := first;
 repeat
   while (i <= Length(s)) and (not (s[i] in [ #9, #32])) and (s[i] <> #13) do
     inc(i);
   s1 := copy(s, first, i - index);
   inc(i);
 until
   (i >= Length(s)) or (s[i-1] = #13) or (RealTextWidth(s1) > MaxTextWidth);
 if RealTextWidth(s1) > MaxTextWidth then
 begin
   result.wrap := false;
   if i < Length(s) then
   begin
     dec(i, 2);
     while (i > 0) and (not (s[i] in [ #9, #32])) do
       dec(i);
     result.Length := i - index;
     while (i > 0) and (s[i] in [ #9, #32]) do
       dec(i);
   end;
   result.s := copy(s, first, i - index + 1);
   if result.s[length(result.s)] = #32 then
     delete(result.s, length(result.s) , 1);
 end
 else
 begin
   result.length := i - index + 1;
   s1 := copy(s, first, i - index + 1);
   if length(s1) > 0 then
   begin
     if s1[Length(s1)] = #9 then
       delete(s1, Length(s1), 1);
     if s1[length(s1) - 1] + s1[length(s1)] = #13#10 then
       delete(s1, length(s1) - 1, 2);
   end;
   result.s := s1;
   result.wrap := true;
 end;
end;

procedure draw;
var
 i, j: integer;
 line: TLine;
 OneWord: string;
 LineN: integer;
 SpaceCount: integer;
 TextLeft: integer;
 shift, allshift: integer;
 d: integer;
 LineCount: integer;
begin
 with bm.Canvas do
 begin
   FillRect(ClipRect);
   i := 1;
   LineCount := 0;
   for j := 1 to Form1.ScrollBar1.Position do
   begin
     line := GetLine(i);
     inc(i, line.length);
     inc(LineCount);
   end;
   LineN := 0;
   repeat
     line := GetLine(i);
     SpaceCount := 0;
     TextLeft := 0;
     for j := 1 to Length(line.s) do
       if line.s[j] = #32 then
         inc(SpaceCount);
     if line.wrap = false then
       allshift := MaxTextWidth - RealTextWidth(line.s)
     else
       allshift := 0;
     if allshift > 40 * SpaceCount then
       allshift := 0;
     shift := 0;
     for j := 1 to Length(line.s) do
     begin
       if (not (line.s[j] in [ #9, #32])) and (j < Length(line.s)) then
       begin
         OneWord := OneWord + line.s[j];
       end
       else
       begin
         OneWord := OneWord + line.s[j];
         if OneWord = #9 then
         begin
           inc(TextLeft, 40);
         end
         else
         begin
           if OneWord = #13#10 then
           begin
             inc(LineN);
           end
           else
           begin
             TextOut(10 + TextLeft, LineN * LineH, OneWord);
             if SpaceCount = 0 then
               d := 0
             else
               d := (allshift - shift) div (SpaceCount);
             inc(shift, d);
             inc(TextLeft, TextWidth(OneWord) + d);
             dec(SpaceCount);
           end;
         end;
         OneWord := "";
       end;
     end;
     inc(i, line.length);
     inc(LineN);
   until
     (LineN * LineH > Form1.PaintBox1.Height) or (i >= Length(s));

   repeat
     line := GetLine(i);
     inc(i, line.length);
     inc(LineCount);
   until
     i >= Length(s);

   inc(LineCount, LineN);
   Form1.ScrollBar1.Max := LineCount -
   Form1.PaintBox1.Height div LineH;
 end;
 Form1.PaintBox1.Canvas.Draw(0, 0, bm);
end;

procedure TForm1.PaintBox1Paint(Sender: TObject);
begin
 draw;
end;

procedure TForm1.ScrollBar1Change(Sender: TObject);
begin
 draw;
end;


 
Dstr ©   (2006-05-10 19:47) [35]


> А как реализовать правильно Вашу мысль - "Хотя проще вместо
> Delete использовать SetLength,Я токо про нее вспомнил":
>
> Function NumWorld(S: String; Pos: Integer): Longint;
> Begin;
> Delete(S, Pos, Length(S) - Pos);
> Result := WordCount(S);
> End;
>
> Как правильно будет выглядеть тогда код функции NumWorld?
>  
> Спасибо всем Мастерам за помощь и Вам в частности!


Вот так
Function NumWorld(S: String; Pos: Integer): Longint;
Begin;
SetLength(S, Pos);
Result := WordCount(S);
End;


 
Loginov Dmitry ©   (2006-05-10 20:25) [36]

Вы что, компонент написать хотите?

Насколько понял [31], для решения сабжа этого достаточно:

procedure TForm1.Button1Click(Sender: TObject);
var
 List: TStringList;
begin
 List := TStringList.Create;
 GetWordList(List);
 ShowMessage(IntToStr(List.IndexOf(Edit1.Text)));
 List.Free;
end;

procedure TForm1.GetWordList(List: TStrings);
var
 S: string;
 I: Integer;
const
 WordChars = ["a".."z", "A".."Z", "0".."9", "а".."я", "А".."Я", "ё", "Ё"];
begin
 S := Memo1.Text;
 for I := 1 to Length(S) do
   if not (S[I] in WordChars) then S[I] := " ";
 List.DelimitedText := S;
end;


 
Dr. Andrew   (2006-05-10 20:33) [37]

Всем спасибо! Вроде бы получается. Остальное попробую доработать самостоятельно.



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

Текущий архив: 2006.06.18;
Скачать: CL | DM;

Наверх




Память: 0.58 MB
Время: 0.011 c
2-1149014525
Blari
2006-05-30 22:42
2006.06.18
Перетаскивание в ListBox


1-1147409869
Kostromich
2006-05-12 08:57
2006.06.18
TWebBrowser Навигация


2-1149012980
JTAG
2006-05-30 22:16
2006.06.18
Господа подскажите пожалуйста как сделать чтобы в StringGrid


3-1145891342
linx
2006-04-24 19:09
2006.06.18
Запретить чтение ячейки данных в таблице


11-1128890490
DillerXX
2005-10-10 00:41
2006.06.18
Центр экрана





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
Английский Французский Немецкий Итальянский Португальский Русский Испанский