2016-02-06 12 views
6

Próbuję zastąpić wszystkie wystąpienia ~ z {~} w skrypcie wsadowym, tak aby VBScript nie zakładał, że mam zamiar nacisnąć klawisz enter, gdy przekazuję go do sendKeys, jednak zmienna ciągle jest zastępowana przez ciąg znaków, który ma na celu Wymień.Jak uciec "~" w poleceniu set/replace?

Na przykład

rem Intended function; 
rem input = Hello~world. 
rem output = Hello{~}world. 

Jednak za każdym razem, to albo nie robi nic, albo ustawia refVar do dosłownego łańcucha refVar:~={~}.

Oto, co próbowałem;

set refVar=!refVar:~={~}! 

rem Also 
set refVar=%refVar:~={~}% 

rem and... 
set refVar=%refVar:^~=^{^~^}% 

rem and of course; 
set refVar=!refVar:^~=^{^~^}! 

rem and even: 

set "pre=~" 
set "post={~}" 
set refVar=%refVar:!pre!=!post!% 

rem and the same set commands with; 
set refVar=!refVar:%pre%=%post%! 

Czy brakuje mi sposobu na ucieczkę? Jestem prawie pewien, że ma to związek z ~ będącym postacią pozycyjną w podobnym poleceniu.

Rozumiem, że istnieje sposób, aby to naprawić w VBScript, ale to przeszkadza mi mieć ograniczenie bez widocznych obejść do mnie.

Odpowiedz

2

I oferują prawdopodobnie łatwiejsze rozwiązanie zastępujące każdą znak tyldy w łańcuchu wartości zmiennej środowiskowej przez tyldę w nawiasach klamrowych niż Dennis van Gils, która również dobrze się nadaje do tego zadania.

@echo off 
set "TestVar=hello~world." 
echo TestVar has value: %TestVar% 
call :SpecialReplace "TestVar" 
echo TestVar has value: %TestVar% 
goto :EOF 


rem The subroutine below expects the name of an environment variable as 
rem parameter. The subroutine does nothing if called without parameter. 
rem Also nothing is done if specified environment variable is not defined. 

rem Each tilde character in value of this environment variable is replaced 
rem by {~} by this subroutine. 

rem Note: This subroutine can be also easily modified to replace other 
rem special characters like the equal sign by a different string which 
rem can be also no string in case of special character should be removed. 
rem Just modify Search and Replace variables for a different replace. But 
rem be aware of more code must be changed if search string is longer than 
rem one character. Length of replace string does not matter on code below. 

:SpecialReplace 
if "%~1" == "" exit /B 
setlocal EnableDelayedExpansion 
set "VarName=%~1" 
set "VarValue=!%VarName%!" 
if "!VarValue!" == "" endlocal & exit /B 
set "NewValue=" 
set "Search=~" 
set "Replace={~}" 

:ReplaceLoop 
if "!VarValue:~0,1!" == "!Search!" (
    set "NewValue=!NewValue!!Replace!" 
) else (
    set "NewValue=!NewValue!!VarValue:~0,1!" 
) 
set "VarValue=!VarValue:~1!" 
if not "!VarValue!" == "" goto ReplaceLoop 
endlocal & set "%VarName%=%NewValue%" & exit /B 

dla zrozumienia używanych poleceń i jak one działają, otwórz okno wiersza polecenia, wykonaj następujące polecenia tam i czytać w całości wszystkie strony pomocy wyświetlane dla każdego polecenia bardzo uważnie.

  • call /?
  • echo /?
  • endlocal /?
  • exit /?
  • goto /?
  • if /?
  • rem /?
  • set /?
  • setlocal /?

I patrz również odpowiedzieć na Single line with multiple commands using Windows batch file zrozumieć, w jaki sposób te dwie linie z wielu poleceń z pojedynczym ampersand między poleceniami pracy.


Wariant powyżej z rozszerzonym podprogramu SpecialReplace wspieranie teraz zmienną wyszukiwania i zamiany ciągów podanych jako drugiego i trzeciego parametru jak sugeruje Dennis van Gils.

@echo off 
set "TestVar=hello~world." 
echo TestVar has value: %TestVar% 
call :SpecialReplace "TestVar" "~" "{~}" 
echo TestVar has value: %TestVar% 
goto :EOF 


rem The subroutine below expects the name of an environment variable as 
rem first parameter. The subroutine does nothing if called without parameter. 
rem Also nothing is done if specified environment variable is not defined. 

rem The second parameter must be the search string. The subroutine does 
rem nothing if there is no second parameter which can be also just one 
rem or two double quotes enclosed in double quotes. 

rem The third parameter is an optional replace string. Without a third 
rem parameter all occurrences of search string are removed from value 
rem of the specified environment variable. 

rem With just one or two double quotes in double quotes as search string 
rem no replace string can be defined because command processor does not 
rem parse the quote strings as most would expect it. But """ or """" as 
rem search string with no replace string works and results in removing 
rem all double quotes or all occurrences of two double quotes. 

:SpecialReplace 
if "%~1" == "" exit /B 
setlocal EnableDelayedExpansion 
set "VarName=%~1" 
set "VarValue=!%VarName%!" 
if "!VarValue!" == "" endlocal & exit /B 
set "NewValue=" 
set "Search=%~2" 
if "!Search!" == "" endlocal & exit /B 
set "Replace=%~3" 
set "Length=0" 

:GetLength 
if "!Search:~%Length%,1!" == "" goto ReplaceLoop 
set /A Length+=1 
goto GetLength 

:ReplaceLoop 
if "!VarValue:~0,%Length%!" == "!Search!" (
    set "NewValue=!NewValue!!Replace!" 
    set "VarValue=!VarValue:~%Length%!" 
) else (
    set "NewValue=!NewValue!!VarValue:~0,1!" 
    set "VarValue=!VarValue:~1!" 
) 
if not "!VarValue!" == "" goto ReplaceLoop 
endlocal & set "%VarName%=%NewValue%" & exit /B 
+0

To jest rzeczywiście zgrabna odpowiedź. Lepszym rozwiązaniem może być użycie argumentów do wyszukiwania i zamiany łańcucha, aby ułatwić obsługę. –

+0

Haha, znam polecenia skryptów w plikach wsadowych w większości przypadków, ale domyślam się, że polecenia pomocy są dla innych szukających tej samej odpowiedzi. Tak czy owak, dziękuję, cieszę się, że wiem, że rozwiązaniem nie była tylko pewna kolejność ograniczników, których nie udało mi się wypróbować. – Bloodied

+0

@Arescet Tak, dodałem listę używanych poleceń i jak uzyskać pomoc dla wszystkich czytających tę odpowiedź, ponieważ prawie zawsze robię odpowiedzi na mój plik wsadowy. Było dla mnie oczywiste, że dobrze znasz wszystkie te polecenia. – Mofi

4

Więc zrobiłem trochę kopania i wygląda na to, że to naprawdę, bardzo trudne. Ja jednak nie znaleźliśmy this, która rozwiązuje ten problem, dając nam to jako kod:

@echo off 

set "input=Hello~world." 
echo input: %input% 

call :wavereplacer "%input%" {tilde} output 
set output=%output:{tilde}={~}% 
echo output: %output% 
pause 

::your own code should be above this. 
goto :eof 

:wavereplacer String Replacer [RtnVar] 
setlocal 
rem the result of the operation will be stored here 
set "result=#%~1#" 
set "replacer=%~2" 
call :strlen0.3 result wl 
call :strlen0.3 replacer rl 

:start 

    set "part1=" 
    set "part2=" 

    rem splitting the string on two parts 
    for /f "tokens=1* delims=~" %%w in ("%result%") do (
    set "part1=%%w" 
    set "part2=%%x" 
) 

    rem calculating the count replace strings we should use 
    call :strlen0.3 part1 p1l 
    call :strlen0.3 part2 p2l 
    set /a iteration_end=wl-p1l-p2l 

    rem creating a sequence with replaced strings 
    setlocal enableDelayedExpansion 
    set "sequence=" 
    for /l %%i in (1,1,%iteration_end%) do (
    set sequence=!sequence!%replacer% 
) 
    endlocal & set "sequence=%sequence%" 

    rem adjust the string length 
    set /a wl=wl+iteration_end*(rl-1) 

    rem replacing for the current iteration 
    set result=%part1%%sequence%%part2% 
    rem if the second part is empty the task is over 
    if "%part2%" equ "" (
    set result=%result:~1,-1% 
    goto :endloop 
) 


    goto :start 

:endloop 
endlocal & if "%~3" neq "" (set %~3=%result%) else echo %result% 
exit /b 

:strlen0.3 StrVar [RtnVar] 
    setlocal EnableDelayedExpansion 
    set "s=#!%~1!" 
    set "len=0" 
    for %%A in (2187 729 243 81 27 9 3 1) do (
    set /A mod=2*%%A 
    for %%Z in (!mod!) do (
     if "!s:~%%Z,1!" neq "" (
     set /a "len+=%%Z" 
     set "s=!s:~%%Z!" 

    ) else (
     if "!s:~%%A,1!" neq "" (
      set /a "len+=%%A" 
      set "s=!s:~%%A!" 
     ) 
    ) 
    ) 
) 
    endlocal & if "%~2" neq "" (set %~2=%len%) else echo %len% 
exit /b 

ten nie może zastąpić ~ do {~} natychmiast, trzeba użyć tymczasowego podstawienie, w tym przypadku {tilde}

1
@ECHO Off 
SETLOCAL ENABLEDELAYEDEXPANSION 
SET "input=hello~world" 

SET "xhead=!input:%input:*~=%=!" 
SET "output=!input:%xhead%=%xhead:~0,-1%{~}!" 

ECHO %input% --^> %output% 

GOTO :EOF 

To wydaje się działać pod warunkiem, że jest dokładnie jeden ~ i że nie jest to początkowy lub terminal charakter.