综合实践样例
根据Linux笔记中已经学习的命令,综合运用于实际实践中。
04 执行任务并定时关闭
首行是标记使用的脚本的类型,此处为bash
脚本。
两个命令间用&
连接,前面的命令执行后挂在后台并立即开始执行后面的命令。用&&
连接,只有前面的命令执行完毕并退出后才会执行后面的命令。
第一行开启roscore
后,计时两秒后结束roscore
。kill -2
相当于Ctrl+C
。-2
就是Ctrl+C
发出的siginit
。
#!/bin/bash
roscore &
sleep 3 && kill -2
05 解压文件名有规律的系列文件
这一系列的文件名为“17.7z”、“18.7z”…“21.7z”。解压密码为“blue”。
#!/bin/bash
set -u
for i in {17..21}
do
filename=$i".7z"
7z x $filename -pblue
done
解压所在文件夹内的所有以“.7z”结尾的文件,解压密码为“blue”。
ls | while read line
do
file=$line
if echo $file | grep -q -E '\.7z$'
then
echo $file
7z x $file -pblue
fi
done
07 修改某文件夹下的特定文件
遍历所在指定文件夹及其所有子文件夹中的markdown文档,并把第四行修改为<日期+时间+时区>
#!/bin/bash
files=$(find "." -type f)
# 遍历文件
for file in $files; do
# 检查文件名是否符合特定首尾条件
if [[ "$file" == *_posts* && "$file" == *.md ]]; then
# 获取文件的最近修改时间
last_modified=$(stat -c "%Y" "$file")
# 将时间戳转换为日月年时分秒的格式
last_modified_formatted=$(date -d @"$last_modified" +"%Y-%m-%d %H:%M:%S %z")
# 替换文件的第四行为最近修改时间
sed -i "4s/.*/date: $last_modified_formatted/" "$file"
echo "Updated date in $file to $last_modified_formatted"
fi
done
09 批量删除某个名称的文件夹
find . -type d -name .git -prune -exec rm -r {} \;
.
: 指定查找的范围,即在呢个文件夹内查找-name
: find命令的参数,后面接要查找的文件夹名称.git
: 我们想要删除的文件夹的名称,此处我想删除的文件夹为“.git”-exec
:{}
: can be read as “for each matching file/ folder”\;
is a terminator for the-exec
clause
下面的这个命令也可以使用。
rm -rf `find . -type d -name a`
11 统计文件名
ls | while read line
do
file=$line
if echo $file | grep -q -E '\.bag$'
then
if [ 18 = `echo ${#file}` ]
then
# time=`rosbag info $file | grep start`
# str=`date -d @${time: 0-14: 13} "+%Y-%m-%d %H:%M:%S"`
time=${file: 0: 14}
echo $time >> date_t.txt
fi
fi
done
13 query ros topics
topic_name[0]="/image"
topic_name[1]="/lidar"
topic_name[2]="/rtk"
for topic in ${topic_name[@]};
do
echo $topic
timeout 3 rostopic hz $topic
echo "----------------------"
done
14 bind several processes via service
We can create several peocesses and kill them in one time via service. Here we create a user service, which is different from system service.
First, # create a folder, which will contain service file and shell script. Assume username is “bs”.
mkdir -p /home/bs/.config/systemd/user/
cd /home/bs/.config/systemd/user/
Second, create a service file named myservice.service
like below.
[Unit]
Description=Discribe your service here
# This service starts your custom script after the network is available. If this service id fully independent and does not rely on the state of any other services, you can delete this line
After=network.target
[Service]
Type=simple
# Use `ExecStart` to call a wrapper script to ensure that the script handles the termination of all its child processes upon receiving a signal to stop.
# shell script, it must be absolute path
ExecStart=/home/bs/.config/systemd/user/shell.sh
# ensures that all processes started by the service (including any child processes started by the script) will be terminated when the service itself is stopped.
KillMode=control-group
# on-failure: restart this service if this service dies; no: Don't restart the service automatically
Restart=on-failure
[Install]
WantedBy=default.target
# This service will be started at the default runlevel
Third, reload the user-level systemd configuration.
systemctl --user daemon-reload
Fourth, enable the service.
systemctl --user enable myservice.service
Fifth, don’t forget to create shell script. Now let’s create a script named shell.sh
like below. Here I take ros as example.
#!/bin/bash
# Function to terminate all roslaunch processes
cleanup() {
echo "Stopping all ROS nodes..."
kill -SIGINT "$roslaunch1_pid" "$roslaunch2_pid" "$roslaunch3_pid"
}
# The trap command in the script sets up a handler for SIGTERM, which is the signal sent by systemd when the service is stopped. This handler function, cleanup, will explicitly kill all the background processes.
trap cleanup SIGTERM
# Start roslaunch commands in background
sleep 1 && roslaunch your_package launch_file1.launch & roslaunch1_pid=$!
sleep 2 && roslaunch your_package launch_file2.launch & roslaunch2_pid=$!
sleep 3 && roslaunch your_package launch_file3.launch & roslaunch3_pid=$!
# The wait command makes the main script wait for all the background processes, which means the main service process remains active as long as the background processes are running.
wait $roslaunch1_pid $roslaunch2_pid $roslaunch3_pid
We have completed service and shell file. We can use this service now.
systemctl --user start myservice.service # start service
systemctl --user status myservice.service # check service
systemctl --user restart myservice.service # restart service
systemctl --user stop myservice.service # stop service
Note that if you make some changes to service file, don’t forget to run this cmd systemctl --user daemon-reload
to make it effect.
15 update date from local ntp server
create ntp server
Assume that ip of ntp server is 192.168.1.100 and host name is “bs”. Run cmds below:
sudo apt install ntp # install ntp
sudo systemctl start ntp # start ntp
set client
Run cmd below on client that need update date and time.
sudo nano /etc/systemd/timesyncd.conf # you can also use vim or gedit
Edit like below
[Time]
NTP=bs 192.168.1.100
FallbackNTP=bs 192.168.1.100
Then save and close the file. Restart the timesyncd service with cmd below:
sudo systemctl restart systemd-timesyncd
Check that the NTP synchronization is working correctly with:
timedatectl status
If everything is going well, you will see System lock synchronized
is yes
. You can also run date
and check whether the result is right.
16 Creating a Script to Run at Boot with Cron
- Prepare Your Script: Ensure your script in
/usr/local/func/
is executable. If it’s not, make it executable:
sudo chmod +x /usr/local/func/myscript
- Edit the Crontab:
crontab -e
Add the following line to schedule your script to run at every system boot:
@reboot /usr/local/func/myscript
This line tells cron to run /usr/local/func/myscript every time the system boots up.
-
Save and Exit: After adding the line, save and exit the editor. cron will automatically install the new crontab.
-
Verify: To ensure your crontab is set up correctly, you can list your current user’s crontab entries:
crontab -l
This command will show all scheduled cron jobs, including your new @reboot job.
- Additional Considerations
Environment: Cron jobs run with a limited environment, meaning many of the environmental variables available in a full shell session (like those started by logging in) may not be available. If your script relies on certain environment variables, you might need to define them in the script or in the crontab entry. Or you can run
source /home/${user}/.bahsrc
to make script run like in a terminal.
Logging: Since cron jobs don’t have a terminal, you might want to redirect your script’s output to a file for debugging:
@reboot /usr/local/func/myscript >> /var/log/myscript.log 2>&1
This setup ensures that your script runs at system boot without involving system-wide services or requiring systemd, fitting your preference for a simpler, less formal method.
17 split and convert string
#!/bin/bash
# colorful output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[0;33m'
BLUE='\033[0;34m'
NC='\033[0m'
if [ "$#" -eq 0 ]; then
echo -e "${RED}Should contain project name.${NC}"
exit
fi
# (absolute path (relative path))
SHELL_DIR=$(realpath $(dirname "${BASH_SOURCE[0]}"))
name=$1
# whether file exists
# if [ -e "$name" ]; then
# echo -e "${BLUE}File ${name} exists, skip generatation.${NC}"
# exit
# fi
# if [ -d "$name" ]; then
# echo -e "${YELLOW}Directory ${name} exists, skip creation.${NC}"
# exit
# fi
# Internal Field Separator
IFS='_'
# Convert string into an array
read -ra PARTS <<< "$name"
# Reset IFS if needed
unset IFS
# STRING="part1_part2_part3"
# # Get specific parts
# FIRST_PART=$(echo "$STRING" | cut -d '_' -f 1)
# SECOND_PART=$(echo "$STRING" | cut -d '_' -f 2)
# THIRD_PART=$(echo "$STRING" | cut -d '_' -f 3)
# traverse an array
for PART in "${PARTS[@]}"; do
# echo "$PART"
# Convert first character to uppercase
Name=${Name}$(echo "$PART" | sed 's/./\U&/')
# convert all charactera to uppercase
NAME=${NAME}${PART^^}
NA_ME=${NA_ME}${PART^^}"_"
done
mkdir -p $name
cd $name
echo $Name
echo $NAME
echo $NA_ME
# "eval" allows you to construct and execute commands dynamically.
eval "${NAME}VERSION=1.0"
echo "${COVFERVERSION}"
date "+%Y-%m-%d %H:%M" # use date as separator
var_name="${NAME}VERSION"
# Create the variable dynamically
declare "$var_name=2.0"
# Access the newly created variable using indirect reference
echo "${!var_name}" # Correct way to reference it