【Linux命令行与Shell脚本编程】第十六章 Shell函数

news2024/12/27 13:10:20

Linux命令行与Shell脚本编程

第一章


文章目录

  • Linux命令行与Shell脚本编程
  • 六.函数
    • 6.1.脚本函数基础
      • 6.1.1.创建函数
      • 6.1.2.使用函数
    • 6.2.函数返回值
      • 6.2.1.默认的退出状态码
      • 6.2.2.使用return命令
      • 6.2.3.使用函数输出
    • 6.3.函数中使用变量
      • 6.3.1.向函数传递参数
      • 6.3.2.在函数中处理变量
        • 全局变量
        • 局部变量
    • 6.4.数组变量和函数
      • 6.4.1.向函数中传递数组
      • 6.4.2.从函数返回数组
    • 6.5.函数递归
    • 6.6.创建库
    • 6.7.在命令行中使用函数
      • 6.7.1.命令行中创建函数
    • 6.7.2.在.bashrc文件中定义函数
    • 6.8.共享库函数
      • 6.8.1.下载安装


六.函数

  • 脚本函数基础
  • 函数返回值
  • 在函数中使用变量
  • 数组变量和函数
  • 函数递归
  • 创建库
  • 在命令行中使用函数

可以将shell脚本代码放入函数中封装起来,这样就能在脚本的任意位置多次使用.

6.1.脚本函数基础

函数是一个脚本代码块,可以并在脚本中的任何位置重用它。当需要在脚本中使用该代码块时,直接通过函数名调用.

6.1.1.创建函数

bash shell脚本中创建函数的语法有两种。

  • 使用关键字function
function name {//函数的唯一名称。脚本中的函数名不能重复
    commands //组成函数的一个或多个bash shell命令。  
}
  • 近其他编程语言中定义函数的方式
name() {//函数名后的空括号表明正在定义的是一个函数
commands
}

6.1.2.使用函数

使用函数名调用函数

$ cat test1
#!/bin/bash
# using a function in a script
function func {
   echo "This is an example of a function"
}
count=1
while [ $count -le 5 ]
do
   func
   count=$[ $count + 1 ]
done
$ ./test1
This is an example of a function
This is an example of a function
This is an example of a function
This is an example of a function
This is an example of a function
  • 函数定义不一定非要放在shell脚本的最开始部分,但必须定义在调用之后,在函数被定义之前调用,则会收到一条错误消息.
func: command not found
  • 如果定义了同名函数,新定义就会覆盖函数原先的定义,而不会有任何错误消息.

6.2.函数返回值

bash shell把函数视为一个小型脚本,运行结束时会返回一个退出状态码.

有3种方法能为函数生成退出状态码。

6.2.1.默认的退出状态码

函数的退出状态码是函数中最后一个命令返回的退出状态码。函数执行结束后,可以使用标准变量 $? 来确定函数的退出状态码.

  • 函数执行一结束就立刻读取返回值。用$?变量提取函数返回值之前执行了其他命令,函数的返回值会丢失。
#!/bin/bash
func1() {
    ## 命令执行失败
    ls -l badfile
}
echo "testing the function: "
func1
echo "The exit status is: $?"
testing the function:
trying to display a non-existent file
ls: badfile: No such file or directory
The exit status is: 1

退出状态码是1,因为函数中的最后一个命令执行失败.使用函数的默认退出状态码是一种危险的做法.
ps: 函数最执行成功退出状态码为0.

6.2.2.使用return命令

bash shell会使用return命令以特定的退出状态码退出函数。return命令允许指定一个整数值作为函数的退出状态码.

$ cat test5
#!/bin/bash
# using the return command in a function

function dbl {
   read -p "Enter a value: " value
   echo "doubling the value"
   return $[ $value * 2 ]
}

dbl
echo "The new value is $?"
$
$ ./test5
Enter a value: 200
doubling the value
The new value is 1
$

大于255的任何数值都会产生错误的值.

6.2.3.使用函数输出

需要返回较大的整数值或字符串,可以将函数的’输出’保存到shell变量中.

result=$(function_name)
$ cat test5b
#!/bin/bash
# using the echo to return a value
function dbl {
   read -p "Enter a value: " value
   echo $[ $value * 2 ]
}
result=$(dbl)
echo "The new value is $result"
$
$ ./test5b
Enter a value: 200
The new value is 400
$
$ ./test5b
Enter a value: 1000
The new value is 2000
$

函数会用echo语句来显示计算结果。脚本会获取dbl函数的输出,而不是查看退出状态码。
dbl函数实际上输出了两条消息。read命令输出了一条简短的消息来向用户询问输入值。
bash shell并不将其作为STDOUT输出的一部分,而是直接忽略。
如果用echo语句生成这条消息("Enter a value: ")来询问用户,消息就会与输出值一起被读入shell变量。
借助返回值方法,还可以返回浮点值和字符串.

6.3.函数中使用变量

在函数中使用变量时,需要注意变量的定义方式和处理方式。

6.3.1.向函数传递参数

bash shell会将函数当作脚本来对待。可以像普通脚本那样向函数传递参数.
函数名保存在$0变量中,函数参数依次保存在$1、 2 等变量中 . 可以用特殊变量 2等变量中.可以用特殊变量 2等变量中.可以用特殊变量#来确定传给函数的参数数量.
(处理用户输入章节)

脚本中调用函数时,必须将参数和函数名放在同一行.然后用位置变量来获取参数值。

func_name $value1 10

示例:

$ cat test6
#!/bin/bash
function addem {
   if [ $# -eq 0 ] || [ $# -gt 2 ]
   then
      echo -1
   elif [ $# -eq 1 ]
   then
      echo $[ $1 + $1 ]
   else
      echo $[ $1 + $2 ]
   fi
}

echo -n "Adding 10 and 15: "
value=$(addem 10 15)
echo $value
echo -n "Let's try adding just one number: "
value=$(addem 10)
echo $value
echo -n "Now try adding no numbers: "
value=$(addem)
echo $value
echo -n "Finally, try adding three numbers: "
value=$(addem 10 15 20)
echo $value
$
$ ./test6
Adding 10 and 15: 25
Let's try adding just one number: 20
Now try adding no numbers: -1
Finally, try adding three numbers: -1

函数使用位置变量访问函数参数,无法直接获取脚本的命令行参数。

$ cat badtest1
#!/bin/bash
# trying to access script parameters inside a function

function badfunc1 {
   echo $[ $1 * $2 ]
}

if [ $# -eq 2 ]
then
   value=$(badfunc1)
   echo "The result is $value"
else
   echo "Usage: badtest1 a b"
fi
$
$ ./badtest1
Usage: badtest1 a b
$ ./badtest1 10 15
./badtest1: *  : syntax error: operand expected (error token is "*")
The result is
$

尽管函数使用了$1变量和$2变量,但和脚本主体中的$1变量和$2变量不同。
要在函数中使用脚本的命令行参数,必须在调用函数时手动将其传入.

value=$(badfunc1 $1 $2)

在将$1和$2传给函数后就能跟其他变量一样供函数使用。

6.3.2.在函数中处理变量

变量的作用域是比较麻烦的.作用域是变量的有效区域.
函数有两种类型的变量:

  • 全局变量
  • 局部变量

全局变量

在shell脚本内任何地方都有效的变量。
在默认情况下,在脚本中定义的任何变量都是全局变量。在函数外定义的变量可在函数内正常访问.

$ cat test8
#!/bin/bash
function dbl {
    value=$[ $value * 2 ]
}
read -p "Enter a value: " value
dbl
echo "The new value is: $value"
$
$ ./test8
Enter a value: 450
The new value is: 900

$value变量在函数外定义并被赋值。如果变量在函数内被赋予了新值,那么在脚本中引用该变量时,新值仍可用。

局部变量

在任何在函数内部使用的变量都可以被声明为局部变量.local关键字保证了变量仅在该函数中有效。

  • 在变量声明之前加上local关键字.
  • 在变量赋值语句中使用local关键字.
local temp
local temp=$[ $value + 5 ]

如果函数之外有同名变量,两个变量的值互不干扰。

function func1 {
   local temp=$[ $value + 5 ]
   result=$[ $temp * 2 ]
}

6.4.数组变量和函数

在函数中使用数组变量.

6.4.1.向函数中传递数组

将数组变量当作单个参数传递不起作用.
数组变量作为函数参数进行传递,则函数只会提取数组变量的第一个元素。

$ cat badtest3
#!/bin/bash
function testit {
    echo "The parameters are: $@"
    thisarray=$1
    echo "The received array is ${thisarray[*]}"
}
myarray=(1 2 3 4 5)
echo "The original array is: ${myarray[*]}"
testit $myarray
$
$ ./badtest3
The original array is: 1 2 3 4 5
The parameters are: 1
The received array is 1

必须先将数组变量拆解成多个数组元素,然后将这些数组元素作为函数参数传递。最后在函数内部,将所有的参数重新组合成一个新的数组变量。

$ cat test11
#!/bin/bash
function addarray {
    local sum=0
    local newarray
    newarray=(`echo "$@"`)
    for value in ${newarray[*]}
        do
        sum=$[ $sum + $value ]
    done
    echo $sum
}
myarray=(1 2 3 4 5)
echo "The original array is: ${myarray[*]}"
arg1=$(echo ${myarray[*]})
result=$(addarray $arg1)
echo "The result is $result"
$
$ ./test11
The original array is: 1 2 3 4 5
The result is 15

6.4.2.从函数返回数组

函数向shell脚本返回数组变量先用echo语句按正确顺序输出数组的各个元素,脚本再将数组元素重组成一个新的数组变量.

$ cat test12
#!/bin/bash
function arraydblr {
   local origarray
   local newarray
   local elements
   local i
   origarray=($(echo "$@"))
   newarray=($(echo "$@"))
   elements=$[ $# - 1 ]
   for (( i = 0; i <= $elements; i++ ))
   {
      newarray[$i]=$[ ${origarray[$i]} * 2 ]
   }
   echo ${newarray[*]}
}
myarray=(1 2 3 4 5)
echo "The original array is: ${myarray[*]}"
arg1=$(echo ${myarray[*]})
result=($(arraydblr $arg1))
echo "The new array is: ${result[*]}"
$
$ ./test12
The original array is: 1 2 3 4 5
The new array is: 2 4 6 8 10

通过$arg1变量将数组元素作为参数传给arraydblr函数。
arraydblr函数将传入的参数重组成新的数组变量,生成数组变量的副本。对数据元素进行操作,并将结果存入函数中的数组变量副本。

6.5.函数递归

函数可以递归地调用.
递归函数通常有一个最终可以迭代到的基准值。
递归算法的经典例子是计算阶乘:

$ cat test13
#!/bin/bash
function factorial {
   if [ $1 -eq 1 ]
   then
      echo 1
   else
      local temp=$[ $1 - 1 ]
      local result=$(factorial $temp)
      echo $[ $result * $1 ]
   fi
}
read -p "Enter value: " value
result=$(factorial $value)
echo "The factorial of $value is: $result"
$
$ ./test13
Enter value: 5
The factorial of 5 is: 120

6.6.创建库

使用函数可以为脚本省去一些重复性的输入工作.如果要在多个脚本中使用同一段代码,通过库解决.
bash shell允许创建函数库文件,然后在多个脚本中引用此库文件。

  1. 创建一个包含脚本中所需函数的公用库文件。
    $ cat myfuncs
    function addem {
        echo $[ $1 + $2 ]
    }
    function multem {
        echo $[ $1 * $2 ]
    }
    
  2. 在需要用到这些函数的脚本文件中包含myfuncs库文件.
    shell函数的作用域和环境变量一样,shell函数仅在定义它的shell会话内有效。
    如果在shell命令行界面运行myfuncs脚本,那么shell会创建一个新的shell并在其中运行这个脚本。
    这种情况下,函数会定义在新shell中,当运行另一个要用到这些函数的脚本时,它们是无法使用的。
    如果尝试像普通脚本文件那样运行库文件,函数也不会出现在脚本中.
    $ cat badtest4
    #!/bin/bash
    # using a library file the wrong way
    ./myfuncs
    result=$(addem 10 15)
    echo "The result is $result"
    $
    $ ./badtest4
    ./badtest4: addem: command not found
    The result is
    
    使用函数库的关键在于source命令。
    source命令不会创建新的shell,而是在当前shell的上下文中执行命令.
    source命令称作点号操作符。在shell脚本中运行myfuncs库文件,只需添加:
    . ./myfuncs
    
    假定myfuncs库文件和shell脚本位于同一目录。否则需要使用正确路径访问该文件。
    $ cat test14
    #!/bin/bash
    . ./myfuncs
    value1=10
    value2=5
    result1=$(addem $value1 $value2)
    echo "The result of adding them is: $result1"
    $
    $ ./test14
    The result of adding them is: 15
    

6.7.在命令行中使用函数

有时候,在命令行界面用函数执行一些十分复杂的操作.

6.7.1.命令行中创建函数

shell会解释用户输入的命令,可以在命令行中直接定义一个函数.
在命令行创建函数时要特别小心。如果给函数起了一个跟内建命令或另一个命令相同的名字,那么函数就会覆盖原来的命令。

  • 单行方式
    在命令行中定义函数时,必须在每个命令后面加个分号,用于界定命令的起止.
    $ function divem { echo $[ $1 / $2 ];  }
    $ divem 100 5
    20
    
  • 多行方式
    定义时,bash shell会使用次提示符’>'来提示输入更多命令。使用这种方法,无须在每条命令的末尾放置分号,只需按下回车键.
    输入函数尾部的花括号后完成函数的定义.
    $ function multem {
    > echo $[ $1 * $2 ]
    > }
    $ multem 2 5
    10
    

在命令行中直接定义shell函数的一个明显缺点是,在退出shell时,函数也会消失。
在命令行创建函数时要特别小心。如果给函数起了一个跟内建命令或另一个命令相同的名字,函数就会覆盖原来的命令。

6.7.2.在.bashrc文件中定义函数

解决退出shell函数消失的问题.
将函数定义在每次新shell启动时都会重新读取该函数的地方。
.bashrc文件,不管是交互式shell还是从现有shell启动的新shell,bash shell在每次启动时都会在用户主目录中查找这个文件。

  1. 直接定义函数
    直接在用户主目录的.bashrc文件中定义函数。将函数放在文件末尾.
    $ cat .bashrc
    # .bashrc
    # Source global definitions
    if [ -r /etc/bashrc ]; then
            . /etc/bashrc
    fi
    function addem {
       echo $[ $1 + $2 ]
    }
    $
    
    函数会在下次启动新的bash shell时生效。随后就能在系统中的任意地方使用这个函数。
  2. 源引函数文件
    只要是在shell脚本中,就可以用source命令(或别名即点号操作符)将库文件中的函数添加到.bashrc脚本中.确保库文件的路径名正确.
    $ cat .bashrc
    # .bashrc
    # Source global definitions
    if [ -r /etc/bashrc ]; then
            . /etc/bashrc
    fi
    . /home/rich/libraries/myfuncs
    $
    

shell会将定义好的函数传给子shell进程,这些函数就能够自动用于该shell会话中的任何子shell脚本.
用源引库文件,这些函数就可以在shell脚本中顺畅运行。

6.8.共享库函数

在开源世界中,共享代码是必不可少的,同样适用于shell脚本函数。可以下载各种shell脚本函数并将其用于自己的应用程序中。
下载、安装以及使用GNU shtool shell脚本函数库。shtool库提供了一些简单的shell脚本函数,可用于实现日常的shell功能,比如处理临时文件和目录、格式化输出显示等。

6.8.1.下载安装

  1. 将GNU shtool库下载并安装到你的系统中.
    wget http://ftp.gnu.org/gnu/shtool/shtool-2.0.8.tar.gz
    
  2. 将文件复制到主目录中.
  3. 使用tar命令提取文件.
    tar -zxvf shtool-2.0.8.tar.gz
    
  4. 进入到解压新创建的目录
  5. 构建库文件
    使用标准的configure命令和make命令
    $ ./configure
    $ make
    

configure命令会检查构建shtool库文件所必需的软件。发现了所需的工具,会使用工具路径修改配置文件。
make命令负责构建shtool库文件。最终的shtool文件是一个完整的库软件包文件。
可以使用make命令测试库文件

$ make test
Running test suite:
echo...........ok
... ...OK: passed: 19/19

测试模式会测试shtool库中所有的函数。如果全部通过了测试,就可以将库安装到Linux系统中的公用位置,这样你的所有脚本就都能使用这个库了。
以root用户使用make命令的install选项完成安装.

# make install
Password:
./shtool mkdir -f -p -m 755 /usr/local
... ...
./shtool install -c -m 644 sh.version /usr/local/share/shtool/sh.version
./shtool install -c -m 644 sh.path /usr/local/share/shtool/sh.path
# shtool -v
GNU shtool 2.0.8 (18-Jul-2008)

安装成功,可以使用了.

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/845710.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【jvm】jvm整体结构(hotspot)

目录 一、说明二、java代码的执行流程三、jvm的架构模型3.1 基于栈式架构的特点3.2 基于寄存器架构的特点 一、说明 1.hotspot vm是目前市场上高性能虚拟机的代表作之一 2.hotspot采用解释器与即时编译器并存的架构 3.java虚拟机是用来解释运行字节码文件的&#xff0c;入口是字…

微服务技术栈(1.0)

微服务技术栈 认识微服务 单体架构 单体架构&#xff1a;将业务的所有功能集中在一个项目中开发&#xff0c;打成一个包部署 优点&#xff1a; 架构简单部署成本低 缺点&#xff1a; 耦合度高 分布式架构 分布式架构&#xff1a;根据业务功能对系统进行拆分&#xff0c…

深度学习和OpenCV的对象检测(MobileNet SSD图像识别)

基于深度学习的对象检测时,我们主要分享以下三种主要的对象检测方法: Faster R-CNN(后期会来学习分享)你只看一次(YOLO,最新版本YOLO3,后期我们会分享)单发探测器(SSD,本节介绍,若你的电脑配置比较低,此方法比较适合R-CNN是使用深度学习进行物体检测的训练模型; 然而,…

新西兰跨境电商购物网站多语言翻译系统快速开发

搭建一个新西兰跨境电商购物网站的多语言翻译系统&#xff0c;可以按照以下步骤进行快速开发&#xff1a; 步骤1&#xff1a;确定需求和目标 首先&#xff0c;你需要明确你的网站需要支持哪些语言&#xff0c;并确定你想要实现的多语言翻译系统的具体功能和目标。 步骤2&…

unity如何手动更改脚本执行顺序

在Unity中&#xff0c;脚本的执行顺序是由脚本的执行顺序属性决定的。默认情况下&#xff0c;Unity根据脚本在项目中的加载顺序来确定它们的执行顺序。然而&#xff0c;你可以手动更改脚本的执行顺序&#xff0c;以下是一种方法&#xff1a; 在Unity编辑器中&#xff0c;选择你…

【云原生】kubectl命令的详解

目录 一、陈述式资源管理方式1.1基本查看命令查看版本信息查看资源对象简写查看集群信息配置kubectl自动补全node节点查看日志 1.3基本信息查看查看 master 节点状态查看命名空间查看default命名空间的所有资源创建命名空间app删除命名空间app在命名空间kube-public 创建副本控…

Unity背包系统与存档(附下载链接)

下载地址: https://download.csdn.net/download/qq_58804985/88184776 视频演示: 功能: 拖动物品在背包中自由移动,当物品拖动到其他物品上时,和其交换位置.基于EPPlus的背包数据与位置保存 原理: 给定一个道具池表格与一个背包表格 道具池表格负责存储所有道具的信息 背…

岩土工程监测仪器多通道振弦传感器信号转换器应用于铁路监测

岩土工程监测仪器多通道振弦传感器信号转换器应用于铁路监测 岩土工程监测是工程建设和运营过程中必不可少的环节&#xff0c;它主要是通过对地下水位、土体应力、变形、固结沉降等参数进行实时监测&#xff0c;以保证工程施工和运营的安全性和稳定性。而多通道振弦传感器信号…

JavaScript数据结构【进阶】

注&#xff1a;最后有面试挑战&#xff0c;看看自己掌握了吗 文章目录 使用 splice() 添加元素使用 slice() 复制数组元素使用展开运算符复制数组使用展开运算符合并数组使用 indexOf() 检查元素是否存在使用 for 循环遍历数组中的全部元素创建复杂的多维数组将键值对添加到对象…

阶段总结(linux基础)

目录 一、初始linux系统 二、基本操作命令 三、目录结构 四、文件及目录管理命令 查看文件内容 创建文件 五、用户与组管理 六、文件权限与压缩管理 七、磁盘管理 八、系统程序与进程管理 管理机制 文件系统损坏 grub引导故障 磁盘资源耗尽 程序与进程的区别 查…

Grafana V10 告警推送

最近项目建设完成&#xff0c;一个城域网项目&#xff0c;相关zabbix和grafana展示已经完&#xff0c;想了想&#xff0c;不想天天看平台去盯网络监控平台&#xff0c;索性对告警进行分类调整&#xff0c;增加告警的推送&#xff0c;和相关部门的提醒&#xff0c;其他部门看不懂…

绿盾用户使用看图软件每次都需要把图片解密之后才能打开查看,怎么才能不用这么麻烦打开就能看

环境: Win10专业版 绿盾控制台7.0 看图软件FastStone Image Viewer 问题描述: 绿盾用户使用看图软件FastStone Image Viewer每次都需要把图片解密之后才能打开查看,怎么才能不用这么麻烦打开就能看,用户说每次都需要把图片解密之后才能使用,实在是影响效率 解决方案…

4、长度最小的子数组

找到一个数组中&#xff0c;有多少个连续元素的和小于某个值&#xff0c;求出连续元素的长度的最小值。 滑动窗口法&#xff1a; 其本质也是快慢指针&#xff0c;一个指针指向窗口的起始位置&#xff0c;另一个指针指向窗口的终止位置。 1.定义快慢指针&#xff1a; 2.更新慢指…

css-3:什么是响应式设计?响应式的原理是什么?如何做?

1、响应式设计是什么&#xff1f; 响应式网站设计&#xff08;Responsive WEB desgin&#xff09;是一个网络页面设计布局&#xff0c;页面的设计与开发应当根据用户行为以及设备环境&#xff08;系统平台、屏幕尺寸、屏幕定向等&#xff09;进行相应的相应和调整。 描述响应式…

Unity 实现字幕打字效果

Text文本打字效果&#xff0c;TextMeshPro可以对应参考&#xff0c;差距不大&#xff0c;改改参数名就能用。改脚本原本被我集成到其他的程序集中&#xff0c;现在已经分离。 效果 实现功能 1.能够设置每行能够容纳的字数和允许的冗余 2.打字效果 3.每行打完上移 4.开头进入&…

项目实战 — 消息队列(5){统一硬盘操作}

前面已经使用数据库管理了交换机、绑定、队列&#xff0c;然后又使用了数据文件管理了消息。 那么&#xff0c;这里就创建一个类&#xff0c;讲之前的两个部分整合起来&#xff0c;对上层提供统一的一套接口&#xff0c;表示硬盘上存储的所有的类的信息。 /* * 用这个类来管理…

C++笔记之enum class和emun的区别

C笔记之enum class和emun的区别 code review! 代码,使用 enum class 的示例&#xff1a; #include <iostream>enum class Month { January, February, March, April, May, June, July, August, September, October, November, December };int main() {Month currentM…

龙迅LT8711H是Type-C/DP1.2转HDMI1.4芯片 -现货来了

1.概述 LT8711H是一款高性能Type-C / DP1.2至HDMI1.4转换器&#xff0c;旨在将USB Type-C源或DP1.2源连接至HDMI1.4接收器。 LT8711H集成了符合DP1.2的接收器和符合HDMI1.4的发送器。此外&#xff0c;还包括两个CC控制器&#xff0c;用于CC通信以实现DP Alt Mode和功率传输功…

Frida 编译(去特征)

Frida 编译&#xff08;去特征&#xff09; 编译最新版server编译往期版server更改特征使用定制库 hluwa本文引用&#xff1a; 本文环境&#xff1a; kali-linux-xfce 编译最新版server 第一步&#xff1a; 下载frida git clone --recurse-submodules https://github.com/fri…

Redis键值设计

1.1、优雅的key结构 Redis的Key虽然可以自定义&#xff0c;但最好遵循下面的几个最佳实践约定&#xff1a; 遵循基本格式&#xff1a;[业务名称]:[数据名]:[id]长度不超过44字节不包含特殊字符 例如&#xff1a;我们的登录业务&#xff0c;保存用户信息&#xff0c;其key可以…