shell脚本技巧—创建和清空文件

写这篇文章的起因是因为在分析一个脚本时,看到这么一句脚本,如下所示

................................
>$RMAN_BACKUP/script/rman_backup.sql
................................

最开始我以为它就是将文件rman_backup.sql清空,这个类似于 cat /dev/null > $RMAN_BACKUP/script/rman_backup.sql命令。但是实际调试过程发现它的功能不局限于此。其实它的功能为:如果文件不存在时创建这么一个空文件,如果文件存在时清空该文件。猜测编写者就是想通过这么一句简单代码去实现这样的两个功能,他都懒得去写一段逻辑判断:"如果文件不存在就创建空的文件,如果文件存在就清空文件"。说这个脚本精妙吧,倒也谈不上。但是也确实够精简,代价就是阅读起来晦涩难懂。尤其对shell脚本不精通的人。只是shell编程中确实有一些奇淫技巧.这里不谈其优点与缺点,存在既是合理。

那么在shell中,有哪些创建文件的技巧呢?下面简单总结一下,方便以后编写shell脚本时使用,也方便你阅读shell代码。

方法1

touch命令可以用来创建一个空文件,如果文件已经存在,则会更新其时间戳。

案例

if [ ${MYSQL_SERVICE_EXISTS} -eq 1 ] && [ -f "$MYSQL_SERVICE_FILE" ]; then
    touch /usr/lib/systemd/system/mysqld.service
    chmod 644 /usr/lib/systemd/system/mysqld.service
    cat >/usr/lib/systemd/system/mysqld.service <<EOF
    ..................................
    ..................................
fi

方法2:

使用>重定向操作符将输出重定向到一个文件,如果这个文件不存在,就创建这个文件。

案例

................................
echo  "[error]:`date '+%Y%m%d %H:%M:%S'`> $log_msg" > $LOG_FILE
................................

一般会用上面这种写法居多,使用最上面那种写法倒是比较少见。还有一些结合重定向符与其它命令实现的,如下所示

printf "it is test!\n" > filename.txt

方法3:

使用tee命令可以用来读取标准输入并将其写入到文件和标准输出。如果文件不存在,tee命令会创建它

案例

>$RMAN_BACKUP/script/rman_backup.sql
rman_backup |tee -a $RMAN_BACKUP/rman_backup.sql

方法4:

使用cat命令创建一个文件,如下所示,这里使用Here Document(<<EOF),它允许你输入多行文本,直到遇到EOF标记,

案例,创建一个mysqd.service的文件,并向其写入一段启动mysql服务的代码。

cat >/usr/lib/systemd/system/mysqld.service <<EOF
[Unit]
Description=MySQL Server
Documentation=man:mysqld(8)
Documentation=http://dev.mysql.com/doc/refman/en/using-systemd.html
After=network.target
After=syslog.target
 
[Install]
WantedBy=multi-user.target
 
[Service]
User=mysql
Group=mysql
 
# Have mysqld write its state to the systemd notify socket
Type=notify
 
# Disable service start and stop timeout logic of systemd for mysqld service.
TimeoutSec=0
 
# Start main service
ExecStart=/opt/mysql/mysql8.0/bin/mysqld --defaults-file=/data/mysql/conf/my.cnf $MYSQLD_OPTS
 
# Use this to switch malloc implementation
EnvironmentFile=-/etc/sysconfig/mysql
 
# Sets open_files_limit
LimitNOFILE = 100000
 
Restart=on-failure
 
RestartPreventExitStatus=1
 
# Set environment variable MYSQLD_PARENT_PID. This is required for restart.
Environment=MYSQLD_PARENT_PID=1
 
PrivateTmp=false
EOF

在shell脚本中,清空文件有哪些方法呢?一般来说有下面一些方法。

方法1

案例

> mysql_error.log

方法2

案例

echo ''  > mysql_error.log

这个方法比方法1更容易理解,读懂。

方法3

将/dev/null(一个空设备文件)的内容重定向到文件中,可以清空文件. 写法类似这样:cat /dev/null > filename

案例

if [ -f "${SLOW_LOG_PATH}/${NEW_SLOW_LOG_FILE}_${LOG_ROTATION_DATE}.log" ] ; then 
    echo "MySQL慢查询日志已经切换(rotation)"
else
    cat $SLOW_LOG_PATH/$SLOW_LOG_FILE > $SLOW_LOG_PATH/${NEW_SLOW_LOG_FILE}_${LOG_ROTATION_DATE}.log
    #最后运行的命令的结束代码(返回值)
    if [ $? -eq 0 ]; then
      cat /dev/null > $SLOW_LOG_PATH/$SLOW_LOG_FILE
    else
      echo "failed to empty file"
    fi
fi

方法4

truncate命令可以用来减少文件的大小,将其设置为0字节,从而清空文件。

truncate -s 0 mysql_error.log

方法5

: > filename.txt 这种方式来清空文件.

: > mysql_error.log

其中:表示什么也不做。前面命令中它会清空文件内容(截断文件);与rm不同,因为rm实际上会完全删除文件。 此外,如果文件不存在,则 :>filename.txt 实际上会创建该文件。更多关于这个命令的介绍如下所示,参考What is :>filename.txt Doing?[1]文中内容。

As you have discovered, this just empties the file contents (it truncates the file); that is different from rm as rm would actually remove the file altogether. Additionally, :>file.txt will actually create the file if it didn't already exist.

: is a "do nothing command" that will exit with success and produce no output, so it'
s simply a short method to empty a file. In most shells, you could simply do >file.txt to get the same result. It also could be marginally faster than other methods such as echo >file.txt as echo could potentially be an external command.

Additionally, echo >file.txt would put a blank line in file.txt where :>file.txt would make the file have no contents whatsoever.
参考资料
[1]

1: https://unix.stackexchange.com/questions/552436/what-is-filename-txt-doing

扫描上面二维码关注我
如果你真心觉得文章写得不错,而且对你有所帮助,那就不妨帮忙“推荐"一下,您的“推荐”和”打赏“将是我最大的写作动力!
本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接.