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

Вниз

Pos с конца строки   Найти похожие ветки 

 
Vlad0   (2007-11-01 18:49) [0]

Существует ли в Delphi функция Pos, которая ищет вхождения с конца строки? Если нет, то возможно у вас есть хорошо оптимизированный вариант?


 
Palladin ©   (2007-11-01 18:52) [1]

Function RPos(Const s:String;c:Char):Integer;
Var
 i:Integer;
Begin
 For i:=Length(s) Downto 1 Do
  If s[i]=c Then
   Begin
    Result:=i;  
    Exit;
   End;
 Result:=0;
End;


 
Vlad0   (2007-11-01 18:54) [2]

Хмм.. а если у меня для поиска строка целая нужна, а не Char...


 
Плохиш ©   (2007-11-01 18:59) [3]

PosEx + голова


 
palva ©   (2007-11-01 20:01) [4]

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

{$APPTYPE CONSOLE}
uses SysUtils, StrUtils;

const
 vbBinaryCompare = 0;
 vbTextCompare = 1;

{
The InStrRev function returns the position of the first occurrence
of one string within another. The search begins from the end of string,
but the position returned counts from the beginning of the string.

The InStrRev function can return the following values:
}

function InStrRev(const Str, Pat: String; Strt: Integer = -1;
          Scmp: Integer = vbBinaryCompare): Integer;

{
Parameter Description
Str - The string to be searched
Pat - The string expression to search for
Strt Optional. - Specifies the starting position for each search.
The search begins at the last character position by default (-1)
Scmp Optional. - Specifies the string comparison to use. Default is 0
Can have one of the following values:
0 = vbBinaryCompare - Perform a binary comparison
1 = vbTextCompare - Perform a textual comparison

If Str is "" - InStrRev returns 0
If Pat is "" - InStrRev returns start
If Pat is not found - InStrRev returns 0
If Pat is found within Str - InStrRev returns the position at which match
is found
If Strt > Length(Str) - InStrRev returns 0
}

var
 Len, LenP: Integer;
 P: PChar;
 so: TStringSearchOptions;
begin
 Len := Length(Str);
 LenP := Length(Pat);
 if Strt = -1 then Strt := Len;
 if (Len = 0) Or (Strt > Len) Or (LenP > Len) then
   Result := 0
 else if LenP = 0 then
   Result := Strt
 else begin
   if Strt <=0 then raise Exception.Create("Wrong InStrRev parameter");
   so := [];
   if Scmp = vbBinaryCompare then
     so := [soMatchCase];
   P := SearchBuf(PChar(Str), Len, Strt, 0, Pat, so);
   if P = Nil then
     Result := 0
   else
     Result := P - @Str[1] + 1;
 end;
end;

begin
 WriteLn(InStrRev("qWeQwEqWe", "we", 7, vbTextCompare)); // 5
end.


 
Marser ©   (2007-11-01 20:31) [5]

Обратить обе строки и заюзать обычный pos - проще некуда :-)


 
Dib@zol ©   (2007-11-01 21:49) [6]

Собсна вот. Кодил прямо щас, на ходу, поэтому наверняка где-нибудь можно оптимизировать. Читает строчку str с конца, а результат возвращает "с начала" этой строчки. Если нужно отсчитывать "с конца" - раскомментируйте 3 строчки блока @done.

function PosR(sub, str:string):Cardinal;
asm
 PUSH EAX;
 PUSH EBX;
 PUSH EDX;
 PUSH EDI;
 PUSH ESI;

 MOV ESI, str;
 MOV EDI, sub;
 MOV Result, 0;
 XOR EAX, EAX;
 MOV EBX, 1;

 MOV EAX, DWORD PTR [ESI-4];
 MOV EBX, DWORD PTR [EDI-4];

 @rept:
 CMP EAX, 0;
 JE @null;
 MOV DL, BYTE PTR [ESI+EAX-1];
 CMP DL, BYTE PTR [EDI+EBX-1];
 JNE @next;

 MOV Result, EAX;

 CMP EBX, 1;
 JE @done;
 DEC EBX;
 JMP @norm;

 @next:
 CMP Result, 0;
 JNE @null;
 @norm:
 DEC EAX;
 JMP @rept;

 @done:
 //MOV EBX, DWORD PTR [ESI-4];
 //SUB EBX, EAX;
 //ADD EBX, 1;
 MOV Result, EBX;
 JMP @exit;

 @null:
 MOV Result, 0;

 @exit:
 POP ESI;
 POP EDI;
 POP EDX;
 POP EBX;
 POP EAX;
end;


 
Dib@zol ©   (2007-11-01 21:59) [7]

Тфу. Звиняюсь, не работает :(


 
Dib@zol ©   (2007-11-01 22:18) [8]

function PosR(sub, str:string):Cardinal;
asm
 PUSH EAX;
 PUSH EBX;
 PUSH EDX;
 PUSH EDI;
 PUSH ESI;

 MOV ESI, str;
 MOV EDI, sub;

 MOV EAX, DWORD PTR [ESI-4];

 @eval:
 MOV Result, 0;
 MOV EBX, DWORD PTR [EDI-4];

 @rept:
 CMP EAX, 0;
 JE @null;
 MOV DL, BYTE PTR [ESI+EAX-1];
 CMP DL, BYTE PTR [EDI+EBX-1];
 JNE @next;

 MOV Result, EAX;

 CMP EBX, 1;
 JE @done;
 DEC EBX;
 JMP @norm;

 @next:
 CMP Result, 0;
 JNE @eval;
 
 @norm:
 DEC EAX;
 JMP @rept;

 @done:
 //MOV EBX, DWORD PTR [ESI-4];
 //SUB EBX, EAX;
 //MOV EAX, EBX;
 //ADD EAX, 1;
 MOV Result, EAX;
 JMP @exit;

 @null:
 MOV Result, 0;

 @exit:
 POP ESI;
 POP EDI;
 POP EDX;
 POP EBX;
 POP EAX;
end;

Вот. Тесты не выявили существенных отличий от быстродействия стандартного Pos :)



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

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

Наверх




Память: 0.49 MB
Время: 0.018 c
1-1188972200
MZ
2007-09-05 10:03
2007.11.25
Главное меню используя ToolBar2000


2-1194208293
Xak
2007-11-04 23:31
2007.11.25
Как правельно запустить *.bat


3-1183922494
IMHO
2007-07-08 23:21
2007.11.25
Zeos


15-1193284803
vajo
2007-10-25 08:00
2007.11.25
Не срабатывает PHP скрипт


2-1194235125
Bess
2007-11-05 06:58
2007.11.25
Вставить свой пункт в меню IE