Имя пользователя:
Пароль:  
Помощь | Регистрация | Забыли пароль?  | Правила  

Компьютерный форум OSzone.net » Программирование, базы данных и автоматизация действий » Скриптовые языки администрирования Windows » CMD/BAT - поиск и добавление текста в файл с условием

Ответить
Настройки темы
CMD/BAT - поиск и добавление текста в файл с условием

Пользователь


Сообщения: 60
Благодарности: 1

Профиль | Отправить PM | Цитировать


В продолжение этой темы
Есть папка с различными файлами, из которой запускается батник, преобразующий кодировку файлов *.txt и *.csv из этой папки и после выполняющий программу конвертируя их в файлы *.kml
Нужно:
1. добавить проверку: если в начале файла есть строка Name,Latitude,Longitude,Description или Name;Latitude;Longitude;Description не добавлять ее в файл. Если нет определить какой разделитель в файле "," или ";" и добавить в начало файла строку Name,Latitude,Longitude,Description с соответстующим разделителем.
2. Проверить код на универсальность, если есть грубые ляпы убрать их.

Заранее спасибо.

Файлы могут быть следующего вида
Код: Выделить весь код
Name,Latitude,Longitude,Description
Первая точка,33.32456,65.345678,ВЛ-10/4
Вторая_точка,34.234565,54.5456675,ВЛ-11/1
или
Код: Выделить весь код
Name;Latitude;Longitude;Description
Первый_знак;33.32456;65.345678;ВЛ-10/4
Второй_знак;34.234565;54.5456675;ВЛ-11/1

Сам батник:
Код: Выделить весь код
@Echo on

:: =========================================
::      смена кодировки cp1251 на utf-8
:: =========================================

:: создаем папку "utf8"
md utf8
:: перекодируем все файлы с расширением *.txt и *.csv,
:: и кладем их в папку "utf8"
:: в папке с программой должна лежать программа iconv
for %%i in (*.txt *.csv) do ".\iconv\iconv" -t UTF-8 "%%i" > "utf8\%%i"

:: =========================================
::              перевод в *kml
:: =========================================

:: определяем разрядность системы (определяем наличие GPSBabel в установленных программах)
If Defined ProgramFiles(x86) (
	Set gpsbabel="%ProgramFiles(x86)%\GPSBabel\gpsbabel.exe" -w -i unicsv -f
) Else (
	Set gpsbabel="%ProgramFiles%\GPSBabel\gpsbabel.exe" -w -i unicsv -f
)

:: добавляем во все файлы *.txt в папке utf8 строку "Name,Latitude,Longitude,Description"
:: конвертируем и кладем их в папку с исходными

FOR %%i IN (".\utf8\*.txt") DO (
	>"%0.tmp" Echo Name,Latitude,Longitude,Description
	>>"%0.tmp" Type "%%i"
	%gpsbabel% "%0.tmp" -o kml -F "%%~ni.kml" 
)
del "%0.tmp" 2>nul

:: добавляем во все файлы *.csv в папке utf8 строку "Name;Latitude;Longitude;Description"
:: конвертируем и кладем их в папку с исходными

FOR %%i IN (".\utf8\*.csv") DO (
	>"%0.tmp" Echo Name;Latitude;Longitude;Description
	>>"%0.tmp" Type "%%i"
	%gpsbabel% "%0.tmp" -o kml -F "%%~ni.kml" 
)
del "%0.tmp" 2>nul

:: удаляем каталог utf8
rd /s /q "%~dp0\utf8"

exit/b

Пробовал команду:
Код: Выделить весь код
@type "%~dp0\utf8\test2.txt" | find /i "Name,Latitude,Longitude,Description" >nul && exit
>> "%~dp0\utf8\test2.txt" (echo Name,Latitude,Longitude,Description)
проверку выполняет, и добавляет строку, но только в конец файла.

Отправлено: 14:50, 30-06-2016

 
mwz mwz вне форума

Аватара для mwz

Ушел из жизни


Сообщения: 8595
Благодарности: 2127

Профиль | Сайт | Отправить PM | Цитировать


Цитата Sta1917:
добавляет строку, но только в конец файла »
Оператор >> именно это и делает: дописывает файл. Если надо вписать в начало — то поменять порядок с учётом того, что в этом случае придётся использовать промежуточный файл, который затем переименовывать: иначе исходный файл затрётся в процессе операции.

-------
Mikhail Zhilin


Отправлено: 21:45, 30-06-2016 | #2



Для отключения данного рекламного блока вам необходимо зарегистрироваться или войти с учетной записью социальной сети.

Если же вы забыли свой пароль на форуме, то воспользуйтесь данной ссылкой для восстановления пароля.


Ветеран


Сообщения: 1274
Благодарности: 564

Профиль | Отправить PM | Цитировать


Sta1917,
в вашей предыдущей теме в п4 я давал пример как сделать запись в файл первой строкой.

Отправлено: 22:37, 30-06-2016 | #3


Ветеран


Contributor


Сообщения: 2709
Благодарности: 1685

Профиль | Отправить PM | Цитировать


Я слегка запутался в Вашей постановке, что в каком порядке надо делать.
У меня:
1. Файл анализируется на заголовок, если надо, заголовок добавляется. Результат - промежуточный файл.
2. Промежуточный файл перекодируется в UTF-8. Результат - исходный дополненный перекодированный файл.
Перекодировка мной не отлаживалась - нечем и не на чем.
3. Модифицированный исходный конвертируется. Конвертация мной не отлаживалась - нечем и не на чем.
4. Промежуточный файл удаляем.
Код: Выделить весь код
@Echo Off

Chcp 1251 >nul

:Возможные заголовки
Set "Str1=Name,Latitude,Longitude,Description"
Set "Str2=Name;Latitude;Longitude;Description"

:Возможные разделители
Set "Sep1=,"
Set "Sep2=;"

:Путь к программе gpsbabel
If Exist "%ProgramFiles%\GPSBabel\gpsbabel.exe" Set gpsbabel="%ProgramFiles%\GPSBabel\gpsbabel.exe" -w -i unicsv -f
If Exist "%ProgramFiles(x86)%\GPSBabel\gpsbabel.exe" Set gpsbabel="%ProgramFiles(x86)%\GPSBabel\gpsbabel.exe" -w -i unicsv -f

:Путь к программе Iconv
Set "Iconv=.\iconv\iconv.exe"

:Маски обрабатываемых файлов
Set "Files=*.txt *.csv"

SetLocal EnableExtensions EnableDelayedExpansion
For %%f in (%Files%) Do (
	Call :Separator "%%f"
)
GoTo :Eof

:Separator

FOR /F "usebackq delims=" %%i IN (%1) DO (
	Set /A Str=0

	Set "String=%%i"
        Set "String=!String: =!"

	If /I "!String!"=="%Str1%" Set /A Str=1
	If /I "!String!"=="%Str2%" Set /A Str=2
	If !Str!==0 (
		Set "String=%%i"
		If Not "!String:%Sep1%=!"=="%%i" (>"%~1.tmp" Echo %Str1%)
		If Not "!String:%Sep2%=!"=="%%i" (>"%~1.tmp" Echo %Str2%)
		>>"%~1.tmp" type %1
	) Else (
		>"%~1.tmp" type %1
	)
	
	"%Iconv%" -t UTF-8 "%~1.tmp" >%1
	%gpsbabel% %1 -o kml -F "%~n1.kml" 

	Del "%~1.tmp" >nul
	GoTo :Eof
)

GoTo :Eof
Цитата:
Проверить код на универсальность, если есть грубые ляпы убрать их.
Теорема: В любой сколь угодно малой программе существует бесконечное кол-во ошибок. Доказывается методом математической индукции из утверждения, что в любой программе имеется хотя бы одна ошибка. Вследствие очевидности доказательство не привожу и предлагаемый процесс предлагаю произвести заказчику

-------
Даже самая сложная проблема обязательно имеет простое, лёгкое для понимания, неправильное решение. Каждое решение плодит новые проблемы.


Последний раз редактировалось megaloman, 01-07-2016 в 00:19. Причина: Убирал грубые ляпы

Это сообщение посчитали полезным следующие участники:

Отправлено: 00:06, 01-07-2016 | #4


Ветеран


Сообщения: 27449
Благодарности: 8086

Профиль | Отправить PM | Цитировать


Цитата megaloman:
Теорема: В любой сколь угодно малой программе существует бесконечное кол-во ошибок. »
Конечное, коллега, конечное .

Отправлено: 00:20, 01-07-2016 | #5


Пользователь


Сообщения: 60
Благодарности: 1

Профиль | Отправить PM | Цитировать


megaloman, Спасибо, работает. Если сконвертировать его второй раз, получаются крокозябры. Можно чтобы конвертировались промежуточные файлы, либо была проверка на кодировку?

Объясните пожалуйста как добавить другие варианты заголовка и разделителя (табуляции). чтобы код работал?

Цитата megaloman:
:Возможные заголовки
Set "Str1=Name,Latitude,Longitude,Description"
Set "Str2=Name;Latitude;Longitude;Description"

:Возможные разделители
Set "Sep1=,"
Set "Sep2=;" »

Отправлено: 16:38, 01-07-2016 | #6


Ветеран


Contributor


Сообщения: 2709
Благодарности: 1685

Профиль | Отправить PM | Цитировать


Sta1917, У меня есть твёрдое убеждение: исходные файлы, какие бы они ни были, изменять НИЗЗЗЬЯ!! Но Вы моих убеждений, похоже, не разделяете.
Я бы организовал всё по другому.
1. Новые исходные файлы помещал бы в отдельную папку, например Box_In.
2. Обработанные по полной программе файлы помещал бы тоже в другую папку, например, Box_Out
3. Обработанные исходные перемещал бы в папку, например, Box_Arc

Еще одно моё убеждение, если не продумать организацию системы, то обеспечен постоянный головняк. Когда-то меня натыркивали носом: не бросайся писать код, проработай алгоритм задачи.
Если Вас привлечёт мой подход к реализации, могу изваять вариант.

-------
Даже самая сложная проблема обязательно имеет простое, лёгкое для понимания, неправильное решение. Каждое решение плодит новые проблемы.

Это сообщение посчитали полезным следующие участники:

Отправлено: 16:52, 01-07-2016 | #7


Ветеран


Сообщения: 27449
Благодарности: 8086

Профиль | Отправить PM | Цитировать


megaloman, всецело поддерживаю!

Отправлено: 17:01, 01-07-2016 | #8


Ветеран


Contributor


Сообщения: 2709
Благодарности: 1685

Профиль | Отправить PM | Цитировать


Sta1917,
Здесь вариант, где я постарался по максимуму учесть Ваши пожелания
Код: Выделить весь код
@Echo Off
cls

SetLocal EnableExtensions EnableDelayedExpansion

:Путь к папке, где расположены Box_In, Box_Out, Box_Arc
:::Set "Where=Z:\tralala"
Set "Where="

:Маски файлов
Set "Files=*.txt *.csv"

:Список возможных разделителей
Set "Delim=;,."

:Шаблон заголовка
Set "Str00=Name Latitude Longitude Description"

:Путь к программе gpsbabel
If Exist "%ProgramFiles%\GPSBabel\gpsbabel.exe" Set gpsbabel="%ProgramFiles%\GPSBabel\gpsbabel.exe" -w -i unicsv -f
If Exist "%ProgramFiles(x86)%\GPSBabel\gpsbabel.exe" Set gpsbabel="%ProgramFiles(x86)%\GPSBabel\gpsbabel.exe" -w -i unicsv -f

:Путь к программе Iconv
Set "Iconv=.\iconv\iconv.exe"

:Если батник запускаем с параметрами
If Not "%~1"=="" (
	Echo ----------------------------------------------
	Set "Files=%1" & Echo 1 %1
	If Not "%~2"=="" Set "Files=!Files! %2" & Echo 2 %2
	If Not "%~3"=="" Set "Files=!Files! %3" & Echo 3 %3
	If Not "%~4"=="" Set "Files=!Files! %4" & Echo 4 %4
	If Not "%~5"=="" Set "Files=!Files! %5" & Echo 5 %5
	If Not "%~6"=="" Set "Files=!Files! %6" & Echo 6 %6
	If Not "%~7"=="" Set "Files=!Files! %7" & Echo 7 %7
	If Not "%~8"=="" Set "Files=!Files! %8" & Echo 8 %8
	If Not "%~9"=="" Set "Files=!Files! %9" & Echo 9 %9
	Echo ----------------------------------------------
	Echo Press Enter to continue or Ctrl/C to abort
	Pause >nul
)

:Определение числа возможных разделителей
FOR /L %%d IN (0,1,255) DO (
	Set Dlm=!Delim:~%%d,1!
	If "!Dlm!"=="" GoTo :Cont1
	Set /A Nd=%%d
)
:Cont1

:Делаем рабочие папки если их нет. Если расположение не указано, работаем в папке с батником
If "%Where%"=="" Set "Where=%~dp0" & Set "Where=!Where:~0,-1!"
Md "%Where%\Box_In" 2>nul
Md "%Where%\Box_Out" 2>nul
Md "%Where%\Box_Arc" 2>nul

Pushd "%Where%\Box_In"
Del *.tmp 2>nul

If "%~1"=="" (
	For %%f in (%Files%) Do Call :Text "%Where%\Box_In\%%f"
) Else (
	For %%f in (%Files%) Do Call :Text "%%~f"
)

Pause

Popd
GoTo :Eof

:Separ
: Поиск символа разделителя в строке

Set "Str=%~1"
FOR /L %%s IN (0,1,255) DO (
	Set "Str=%~1"
	Set "Sep=!Str:~%%s,1!"
	If "!Sep!"=="" (
		Set "%~2= "
		Set "%~3=%Str00%"
		GoTo :Eof
	) Else (
		FOR /L %%d IN (0,1,%Nd%) DO (
			Set "%~2=!Delim:~%%d,1!"
			If !%~2!==!Sep! (
				Set "%~3=%Str00: =!Sep!%"
				GoTo :Eof
			)
		)
	)
)
GoTo :Eof

:Text

FOR /F "usebackq delims=" %%i IN (%1) DO (
	Set /A Str=0
	Set "String=%%i"
        Set "String=!String: =!"

	Call :Separ "!String!" "Dlm" "Head"

	If /I Not "!String!"=="!Head!" ( 
		>"%~1.tmp" Echo !Head!
		>>"%~1.tmp" type %1
	) Else (
		>"%~1.tmp" type %1
	)
	"%Iconv%" -t UTF-8 "%~1.tmp" >"%Where%\Box_Out\%~nx1"
	%gpsbabel% "%Where%\Box_Out\%~nx1" -o kml -F "%Where%\Box_Out\%~n1.kml"

rem Имитация программы Iconv  
rem 	Echo "%Iconv%" -t UTF-8 "%~1.tmp" ^>"%Where%\Box_Out\%~nx1"
rem 	Copy "%~1.tmp" "%Where%\Box_Out\%~nx1"
rem Имитация программы gpsbabel
rem 	Echo %gpsbabel% "%Where%\Box_Out\%~nx1" -o kml -F "%Where%\Box_Out\%~n1.kml"
rem 	Copy "%Where%\Box_Out\%~nx1" "%Where%\Box_Out\%~n1.kml"

	Move /Y %1 "%Where%\Box_Arc\" >nul
	Del "%~1.tmp"
	Del "%Where%\Box_Out\%~nx1"
	GoTo :Eof
)

GoTo :Eof
Здесь пояснения к этому варианту

Мысли по организации процесса: В параметре Where должна быть прописана папка, где расположены данные, распределённые по папкам: Box_In, Box_Out, Box_Arc. Если строка пустая, как сейчас в примере, то подразумевается, что эти папки расположены в той же папке, что и батник. Если батник запускается без параметров, то исходные данные берутся из Box_In. Шаблоны обрабатываемых файлов задаются в параметре Files. Если в параметрах имена файлов, то обрабатываются они. В частности, если на рабочем столе создать значок со ссылкой на батник, а затем пометить в папке с данными несколько файлов (не более 9!) и затянуть их на этот значок, то обработаются помеченные файлы. Результаты - в подпапке Box_Out указанной папки в параметре Where. Данные из исходного места перемещаются в Box_Arc. Исходные данные не изменяются.
Разделители: Список разрешенных задаётся строкой параметра Delim. Разделитель из нескольких символов, в частности, запятая с пробелом, точка с запятой с пробелом мной не рассматривались. Ограничение: в данных первого поля не должно быть символов из списка разделителей, иначе возможно неверное определение разделителя полей. Шапка задаётся шаблоном в параметре Str00. В шаблоне наименования полей перечислены через пробел, при отсутствии в файле шапки, для обработки шапка добавляется в промежуточный файл с разделителем, взятым из первой строки, если он содержится в списке возможных разделителей.
У меня нет программ Iconv и gpsbabel, поэтому я использовал для отладки их имитации посредством команды copy. Реально конвертация мной не отлаживалась.

-------
Даже самая сложная проблема обязательно имеет простое, лёгкое для понимания, неправильное решение. Каждое решение плодит новые проблемы.


Последний раз редактировалось megaloman, 06-07-2016 в 17:13.

Это сообщение посчитали полезным следующие участники:

Отправлено: 17:07, 06-07-2016 | #9



Компьютерный форум OSzone.net » Программирование, базы данных и автоматизация действий » Скриптовые языки администрирования Windows » CMD/BAT - поиск и добавление текста в файл с условием

Участник сейчас на форуме Участник сейчас на форуме Участник вне форума Участник вне форума Автор темы Автор темы Шапка темы Сообщение прикреплено

Похожие темы
Название темы Автор Информация о форуме Ответов Последнее сообщение
CMD/BAT - Поиск текста по файлам с записью в отдельный текстовый файл Freddy1984 Скриптовые языки администрирования Windows 0 08-02-2016 17:12
PowerShell - [решено] Поиск текста в xml и запись в файл. voleja Скриптовые языки администрирования Windows 5 16-04-2015 11:29
CMD/BAT - BAT поиск текста в файле с условием rosalin Скриптовые языки администрирования Windows 10 05-05-2014 14:56
CMD/BAT - [решено] Добавление числа после точки с условием foma24 Скриптовые языки администрирования Windows 27 08-06-2013 12:08
CMD/BAT - [решено] Поиск с условием и сравнение 63RW51 Скриптовые языки администрирования Windows 3 19-10-2010 07:25




 
Переход