Linux 调试之strace

news2025/1/18 16:56:22

文章目录

  • 前言
  • 一、strace 例子
    • 1.1 strace 跟踪 free
    • 1.2 strace 跟踪 dd
    • 1.3 strace 其他一些使用选项
  • 二、strace 原理
    • 2.1 ptrace简介
    • 2.2 strace 原理
  • 总结
  • 参考资料

前言

strace命令是Linux系统调用跟踪器,可以跟踪系统调用,为每个系统调用打印一行信息,还可以计算系统调用并打印报告。

可以用来分析应用和内核的边界:系统调用。

strace - trace system calls and signals
strace is a useful diagnostic, instructional, and debugging tool.

在最简单的情况下,strace运行指定的命令,直到退出为止。它拦截并记录进程调用的系统调用和进程接收到的信号。每个系统调用的名称、它的参数及其返回值将打印在标准错误中或用-o选项指定的文件中。
跟踪中的每一行都包含系统调用名称,后面括号中的是参数以及返回值。

strace 可以跟踪进程的系统调用、特定的系统调用以及系统调用的执行时间。很多时候,我们通过系统调用的执行时间,就能判断出业务延迟发生在哪里。

一、strace 例子

1.1 strace 跟踪 free

每3S展示一次free的结果。

[root@localhost ~]# free -s 3
              total        used        free      shared  buff/cache   available
Mem:        7890812      873948      874728      133896     6142136     6401416
Swap:       8126460           0     8126460

              total        used        free      shared  buff/cache   available
Mem:        7890812      874264      874408      133896     6142140     6401096
Swap:       8126460           0     8126460

              total        used        free      shared  buff/cache   available
Mem:        7890812      874152      874488      133896     6142172     6401176

......
 -s, --seconds seconds
 	Continuously  display  the  result  delay  seconds  apart.  
[root@localhost strace]# strace -ttt -T -p `pidof free`
strace: Process 3636 attached
1669012085.343662 restart_syscall(<... resuming interrupted nanosleep ...>) = 0 <0.285379>
1669012085.629251 lseek(3, 0, SEEK_SET) = 0 <0.000134>
1669012085.629664 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000524>
1669012085.630510 write(1, "              total        used "..., 80) = 80 <0.000098>
1669012085.630829 write(1, "Mem:        7890812      874880 "..., 80) = 80 <0.000092>
1669012085.631079 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000095>
1669012085.631308 write(1, "\n", 1)     = 1 <0.000091>
1669012085.631486 nanosleep({3, 0}, NULL) = 0 <3.000331>

1669012088.632074 lseek(3, 0, SEEK_SET) = 0 <0.000148>
1669012088.632422 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000539>
1669012088.633246 write(1, "              total        used "..., 80) = 80 <0.000150>
1669012088.633597 write(1, "Mem:        7890812      875304 "..., 80) = 80 <0.000235>
1669012088.634127 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000209>
1669012088.634640 write(1, "\n", 1)     = 1 <0.000175>
1669012088.635075 nanosleep({3, 0}, NULL) = 0 <3.000270>

1669012091.635563 lseek(3, 0, SEEK_SET) = 0 <0.000130>
1669012091.635878 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000592>
1669012091.636697 write(1, "              total        used "..., 80) = 80 <0.000141>
1669012091.637126 write(1, "Mem:        7890812      875428 "..., 80) = 80 <0.000146>
1669012091.637435 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000122>
1669012091.637655 write(1, "\n", 1)     = 1 <0.000086>
......

free的数据来源于 /proc/meminfo 文件。

[root@localhost strace]# strace -e open free
......
open("/proc/meminfo", O_RDONLY)         = 3
......
-t          在跟踪的每一行前加上当前的时间。

-tt         两个t,则打印的时间将包括微秒。

-ttt        三个t, 打印time-since-epoch的第一列,以秒为单位,分辨率为微秒。
[root@localhost strace]# strace -t -T -p `pidof free`
strace: Process 15738 attached
14:55:30 restart_syscall(<... resuming interrupted nanosleep ...>) = 0 <1.943591>
14:55:32 lseek(3, 0, SEEK_SET)          = 0 <0.000134>
14:55:32 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000558>
14:55:32 write(1, "              total        used "..., 80) = 80 <0.000173>
14:55:32 write(1, "Mem:        7890812      874128 "..., 80) = 80 <0.000102>
14:55:32 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000147>
14:55:32 write(1, "\n", 1)              = 1 <0.000269>
14:55:32 nanosleep({3, 0}, NULL)        = 0 <3.000239>

[root@localhost strace]# strace -tt -T -p `pidof free`
strace: Process 15738 attached
14:55:19.844415 restart_syscall(<... resuming interrupted nanosleep ...>) = 0 <0.806829>
14:55:20.651494 lseek(3, 0, SEEK_SET)   = 0 <0.000147>
14:55:20.651856 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000490>
14:55:20.652657 write(1, "              total        used "..., 80) = 80 <0.000206>
14:55:20.653144 write(1, "Mem:        7890812      874612 "..., 80) = 80 <0.000168>
14:55:20.653579 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000211>
14:55:20.654084 write(1, "\n", 1)       = 1 <0.000100>
14:55:20.654325 nanosleep({3, 0}, NULL) = 0 <3.000433>

[root@localhost strace]# strace -ttt -T -p `pidof free`
strace: Process 15738 attached
1669013708.811926 restart_syscall(<... resuming interrupted nanosleep ...>) = 0 <2.833180>
1669013711.645309 lseek(3, 0, SEEK_SET) = 0 <0.000138>
1669013711.645745 read(3, "MemTotal:        7890812 kB\nMemF"..., 8191) = 1282 <0.000464>
1669013711.646423 write(1, "              total        used "..., 80) = 80 <0.000082>
1669013711.646641 write(1, "Mem:        7890812      874800 "..., 80) = 80 <0.000119>
1669013711.646886 write(1, "Swap:       8126460           0 "..., 44) = 44 <0.000091>
1669013711.647057 write(1, "\n", 1)     = 1 <0.000092>
1669013711.647263 nanosleep({3, 0}, NULL) = 0 <3.000412>

-T : 显示系统调用花费的时间。这将记录每个系统调用的开始和结束之间的时间差。
-p pid:附加到进程 ID pid 的进程并开始跟踪。 跟踪可以随时通过键盘中断信号 (CTRL-C) 终止。 strace 将通过将自己与跟踪的进程分离来做出响应,让它(它们)继续运行。支持-p `pidof PROG` 语法。

1.2 strace 跟踪 dd

-c选项可用于总结系统调用活动,计算每次系统调用的时间、调用次数和错误,并在程序退出时报告跟踪信息。
下面的示例还调用并跟踪一个命令(dd),而不是附加到PID上。
输出结果中,从第三行开始是strace的跟踪信息

[root@localhost strace]#  strace -c dd if=/dev/zero of=/dev/null bs=1k count=5000k
5120000+0 records in
5120000+0 records out
5242880000 bytes (5.2 GB) copied, 143.784 s, 36.5 MB/s
% time     seconds  usecs/call     calls    errors syscall
------ ----------- ----------- --------- --------- ----------------
 50.33   10.667418           2   5120003           read
 49.67   10.526616           2   5120003           write
  0.00    0.000280          23        12         6 open
  0.00    0.000117          13         9           mmap
  0.00    0.000052           6         9           close
  0.00    0.000036           9         4           brk
  0.00    0.000031           8         4           fstat
  0.00    0.000023           6         4           mprotect
  0.00    0.000019          10         2           dup2
  0.00    0.000017          17         1         1 access
  0.00    0.000013          13         1           execve
  0.00    0.000009           9         1           lseek
  0.00    0.000009           9         1           arch_prctl
  0.00    0.000005           3         2           munmap
  0.00    0.000000           0         4           rt_sigaction
------ ----------- ----------- --------- --------- ----------------
100.00   21.194645              10240060         7 total

上述执行了5000k次数的read操作和5000k次数的write操作,每次读写操作分别是2微妙。上述dd命令执行了多次系统调用,使用strace跟踪dd将会产生较大的开销。

time: 显示 system CPU time 花费的百分比
seconds: 总的 system CPU time, 单位是秒
usecs/call: 每次调用的平均 system CPU time,以微秒为单位
calls: 系统调用次数
syscall: 系统调用的名字

其中 system CPU time就是命令在内核中运行的时间,独立于wall clock time。

下面是没有使用strace跟踪 dd命令,执行速度非常块,可见strace开销非常大。

[root@localhost strace]# dd if=/dev/zero of=/dev/null bs=1k count=5000k
5120000+0 records in
5120000+0 records out
5242880000 bytes (5.2 GB) copied, 4.80228 s, 1.1 GB/s

当前版本的strace通过Linux ptrace接口使用基于断点的跟踪。这将为所有系统调用的入口和返回设置断点(即使使用-e选项只选择跟踪一些指定的系统调用)。具有高系统调用率的应用程序可能会发现它们的性能降低了一个数量级。

使用strace所花费的时间:

143.784 s, 36.5 MB/s

没有使用strace所花费的时间:

4.80228 s, 1.1 GB/s

可见使用strace所花费的时间提高了几十倍,因为dd执行系统调用的频率非常高。

因此不推荐在生产环境中使用strace应用程序,特别是该应用程序调用系统调用非常频繁,因为strace会在每个系统调用处设置一个断点,开销非常大,目标进程每执行一次系统调用都会被打断,等 strace 处理完后,目标进程才能继续执行,这就会给目标进程带来比较明显的延迟。

根据应用程序需求,可以在短时间内使用这种样式的跟踪来确定被调用的系统调用类型。如果开销不是问题,strace会更有用。其他类型的跟踪程序,包括perf、Ftrace、BCC和bpftrace,通过使用缓冲跟踪大大减少了跟踪开销,其中事件被写入共享内核环缓冲区,用户级跟踪程序定期读取缓冲区。这减少了用户和内核上下文之间的上下文切换,降低了开销。

未来版本的 strace 可能会通过成为 perf trace 子命令的别名来解决其开销问题。
比如:

strace  ls
perf trace ls

上述两者结果一样。

perf 提供了一个 trace 子命令,是取代 strace 的首选工具。相对于 ptrace 机制(strace 基于系统调用 ptrace 实现)来说,perf trace 基于内核事件,事件被写入共享内核环缓冲区,自然要比进程跟踪的性能好很多。

1.3 strace 其他一些使用选项

跟踪进程的线程:

-f          Trace  child  processes  as  they  are  created by currently traced processes as a result of the fork(2), vfork(2) and clone(2) system
            calls.  Note that -p PID -f will attach all threads of process PID if it is multi-threaded, not only thread with thread_id = PID.

-ff         If the -o filename option is in effect, each processes trace is written to filename.pid where pid is the numeric process  id  of  each
            process.  This is incompatible with -c, since no per-process counts are kept.

将跟踪的结果写入到指定的文件中filename:

-o filename Write  the trace output to the file filename rather than to stderr.  Use filename.pid if -ff is used.  If the argument begins with '|'
                  or with '!' then the rest of the argument is treated as a command and all output is piped to it.  This is convenient  for  piping  the
                  debugging output to a program without affecting the redirections of executed programs.

跟踪指定的系统调用:

-e trace=set
            Trace  only  the  specified set of system calls.  The -c option is useful for determining which system calls might be useful to trace.
            For example, trace=open,close,read,write means to only trace those four system calls.  Be careful when  making  inferences  about  the
            user/kernel boundary if only a subset of system calls are being monitored.  The default is trace=all.

-e trace=file
            Trace   all   system   calls   which   take   a   file  name  as  an  argument.   You  can  think  of  this  as  an  abbreviation  for
            -e trace=open,stat,chmod,unlink,...  which is useful to seeing what files the process is referencing.  Furthermore, using the abbrevi‐
            ation will ensure that you don't accidentally forget to include a call like lstat in the list.  Betchya woulda forgot that one.

-e trace=process
            Trace all system calls which involve process management.  This is useful for watching the fork, wait, and exec steps of a process.

-e trace=network
            Trace all the network related system calls.

-e trace=signal
            Trace all signal related system calls.

-e trace=ipc
            Trace all IPC related system calls.

-e trace=desc
            Trace all file descriptor related system calls.

-e trace=memory
            Trace all memory mapping related system calls.

二、strace 原理

2.1 ptrace简介

strace 基于系统调用 ptrace 实现:

NAME
       ptrace - process trace

SYNOPSIS
       #include <sys/ptrace.h>

       long ptrace(enum __ptrace_request request, pid_t pid,
                   void *addr, void *data);

ptrace()系统调用提供了一种方法,通过这种方法,一个进程(“跟踪者”)可以观察和控制另一个进程(“被跟踪者”)的执行,并检查和更改被跟踪者的内存和寄存器。它主要用于实现断点调试和系统调用跟踪(strace 和 gdb的底层实现都是通过 ptrace 实现)。
跟踪者:the tracer
被跟踪者:the tracee

tracee 首先需要附加到 tracer。 附加和后续命令是针对每个线程的:在多线程进程中,每个线程都可以单独附加到一个(可能不同的)tracer,或者不附加,因此不进行调试。 因此,tracee总是意味着(一个)线程,而不是一个(可能是多线程的)进程。

Ptrace 命令总是使用以下形式的调用发送到特定的跟踪对象(tracee):

ptrace(PTRACE_foo, pid, ...)

其中pid是对应Linux线程的线程ID。

进程可以通过调用 fork 并让生成的子进程执行 PTRACE_TRACEME 来启动跟踪,然后(通常)执行 execve。 或者,一个进程可以使用 PTRACE_ATTACH 或 PTRACE_SEIZE 开始跟踪另一个进程。

在被跟踪时,每次发送信号时 the tracee 都会停止,即使信号被忽略也是如此(一个例外是 SIGKILL)。the tracer将在下次调用 waitpid(或相关的“等待”系统调用之一)时得到通知; 该调用将返回一个状态值,其中包含指示 the tracee 停止原因的信息。 当 the tracee 停止时,the tracer 可以使用各种 ptrace 请求来检查和修改 the tracee。 然后 the tracer 使 the tracee 继续,可选择地忽略传递的信号(或者甚至传递不同的信号)。

2.2 strace 原理

strace 原理图如下所示:
在这里插入图片描述
对于正在运行的进程而言,strace 可以 attach 到目标进程上,这是通过 ptrace 这个系统调用实现的。ptrace 的 PTRACE_SYSCALL 会去追踪目标进程的系统调用;目标进程被追踪后,每次进入 syscall,都会产生 SIGTRAP 信号并暂停执行;追踪者通过目标进程触发的 SIGTRAP 信号,就可以知道目标进程进入了系统调用,然后追踪者会去处理该系统调用,我们用 strace 命令观察到的信息输出就是该处理的结果;追踪者处理完该系统调用后,就会恢复目标进程的执行。被恢复的目标进程会一直执行下去,直到下一个系统调用。

// linux-3.10/include/linux/ptrace.h

/*
 * Ptrace flags
 *
 * The owner ship rules for task->ptrace which holds the ptrace
 * flags is simple.  When a task is running it owns it's task->ptrace
 * flags.  When the a task is stopped the ptracer owns task->ptrace.
 */

#define PT_PTRACED	0x00000001
// linux-3.10/include/uapi/linux/ptrace.h

#define PTRACE_TRACEME		   0
#define PTRACE_PEEKTEXT		   1
#define PTRACE_PEEKDATA		   2
#define PTRACE_PEEKUSR		   3
#define PTRACE_POKETEXT		   4
#define PTRACE_POKEDATA		   5
#define PTRACE_POKEUSR		   6
#define PTRACE_CONT		   7
#define PTRACE_KILL		   8
#define PTRACE_SINGLESTEP	   9

#define PTRACE_ATTACH		  16
#define PTRACE_DETACH		  17

#define PTRACE_SYSCALL		  24
// linux-3.10/include/linux/sched.h

struct task_struct {
	......
	unsigned int ptrace;
	......
	/*
	 * pointers to (original) parent process, youngest child, younger sibling,
	 * older sibling, respectively.  (p->father can be replaced with
	 * p->real_parent->pid)
	 */
	struct task_struct __rcu *real_parent; /* real parent process */
	struct task_struct __rcu *parent; /* recipient of SIGCHLD, wait4() reports */

	/*
	 * ptraced is the list of tasks this task is using ptrace on.
	 * This includes both natural children and PTRACE_ATTACH targets.
	 * p->ptrace_entry is p's link on the p->parent->ptraced list.
	 */
	struct list_head ptraced;
	struct list_head ptrace_entry;
	......
}

当一个进程被跟踪后,其struct task_struct的成员ptrace设置为PT_PTRACED,同时其parent设置为跟踪者,real_parent保留被跟踪者的真正父进程。

总结

对于应用程序的系统调用耗时可以用strace分析,对于内核中的函数调用耗时可以用ftrace分析。

参考资料

Linux 3.10

极客时间:Linux内核技术实战
Systems.Performance.Enterprise.and.the.Cloud.2nd.Edition

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

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

相关文章

浅谈SQL Server索引视图(物化视图)以及索引视图与查询重写

目录 &#xff08;一&#xff09;前言 &#xff08;二&#xff09;正文 1. 物化视图&#xff08;索引视图&#xff09;与查询重写的基本概念 2. 创建测试环境 &#xff08;1&#xff09;建表 &#xff08;2&#xff09;写数据 3. 索引视图创建 &#xff08;1&#xff0…

一篇文章详解Linux的用户和权限

教程推荐&#xff1a;Linux零基础快速入门到精通 认知root用户 root用户&#xff08;超级管理员&#xff09; 无论是Windows、MacOS、Linux均采用多用户的管理模式进行权限管理。 •在Linux系统中&#xff0c;拥有最大权限的账户名为&#xff1a;root&#xff08;超级管理员&a…

第十七届全国人机语音通讯学术会议(NCMMSC 2022) | 早鸟票开放注册了

全国人机语音通讯学术会议是国内语音领域广大专家、学者和科研工作者交流最新研究成果&#xff0c;促进该领域研究和开发工作不断进步的重要舞台。该系列会议自1990年开创以来已成功召开了十六届。2022年第十七届全国人机语音通讯学术会议&#xff08;National Conference on M…

移动WEB开发之流式布局--移动端常见布局--流式布局

移动端技术选型 移动端布局和以前我们学习的PC端有所区别&#xff1a; 1. 单独制作移动端页面&#xff08;主流&#xff09; 流式布局&#xff08;百分比布局&#xff09; flex 弹性布局&#xff08;强烈推荐&#xff09; lessrem媒体查询布局 混合布局 2. 响应式页面兼…

ConcurrentHashMap的实现原理是分段锁?你Out了

前言 Java后端开发面试的时候&#xff0c;一场好的面试&#xff0c;是无论如何也绕不开并发编程的。并发编程里面往往有个很重要的类可能会被拿出来探讨&#xff1a;ConcurrentHashMap。 ConcurrentHashMap是HashMap的线程安全版。大家都知道HashMap的高性能&#xff0c;但是H…

动静态库的制作

目录 一.动静态库的原理 二.静态库 2.1制作静态库 2.2使用静态库 三.动态库 3.1制作动态库 3.2动态库的使用 一.动静态库的原理 首先要知道可执行程序的生成过程&#xff1a;1&#xff0c;预处理 2&#xff0c;编译 3&#xff0c;汇编 4 &#xff0c;链接 1.预处理 头…

03 LaTex之标题页摘要

1.标题页 \title{{ABC}\footnote{explain}}%生成标题和标题的脚注\author{\small $^a$lay \qquad $^b$winter \footnote{super star}\\%换行符 %作者信息 \small $^a$ lays brief\\ \small lays address, 710021\\%换行 \small $^b$ winters introduction \\ \small winters …

0101 蓝桥杯真题04

/* * 马虎的算式 * 小明是个急性子&#xff0c;上小学的时候经常把老师写在黑板上的题目抄错了。 * 有一次&#xff0c;老师出的题目是&#xff1a;36 x 495 ? * 他却给抄成了&#xff1a;396 x 45 ? * 但结果却很戏剧性&#xff0c;他的答案竟然是对的&#xff01;&a…

同花顺_代码解析_技术指标_Z_3

本文通过对同花顺中现成代码进行解析&#xff0c;用以了解同花顺相关策略设计的思想 目录 ZNZ_DPCYC1 ZNZ_DPCYR ZNZ_HLD ZNZ_HUO ZNZ_MYL1 ZNZ_MYP1 ZNZ_PAS ZNZ_PAS1 ZNZ_RPY1 ZNZ_RPY2 ZNZ_SDR ZNZ_TAO ZNZ_YHBOL1 ZNZ_YHCBB ZX ZNZ_DPCYC1 大盘成本均线 行…

python 给图片添加噪声

import numpy as np import cv2 import matplotlib.pyplot as plt import skimage from skimage import io import randomdef addGaussNoise(origin,var0.0005):#添加高斯噪声函数var random.uniform(0.0001, 0.04)noisy skimage.util.random_noise(origin, modegaussian, va…

idea iu 2021 Mac版本的使用,如何创建java web项目,包括tomcat和web包

Java web系列文章目录 第一章 前端学习入门之idea iu 2021版本的使用 目录Java web系列文章目录前言一、Java web是什么&#xff1f;二、配置步骤1.下载Tomcat服务器2.idea iu 2021版本界面总结前言 随着前端的学习路径&#xff0c;java web项目不可避免要学习使用&#xff0…

YUV与RGB 以及之间的转换

目录 一、RGB 二、YUV 三、YUV类型和存储方式 1、类型 2、存储方式 四、分析YUV 4:2:0 1、YU12(I420&#xff0c;YUV420P) 2、YV12 3、NV12(YUV420SP) 4、NV21(YUV420SP) 5、占用空间大小比较 五、RGB与YUV之间的转换 1、转换标准 2、Color Range 3、计算公式 在…

【ArcGIS】属性表导出及乱码问题

这玩意其实说难不难&#xff0c;但是乱码有时候还是烦人 直接复制到EXCEL 部分表细节被我删掉了 直接点击全选&#xff0c;然后复制&#xff0c;再到EXCEL里粘贴。我有时候就是这么干的。而且量大概是二十万行左右。 Table to Table 如果你的属性文件大于65533行&#xff…

十一、组合API(1)

本章概要 为什么要引入组合APIsetup() 函数 组合&#xff08;Composition&#xff09;API 是在 Vue 3.0 中引入的&#xff0c;它是一组附加的、基于函数的 API &#xff0c;允许灵活地组合组件逻辑。 组合 API 并没有引入新的概念&#xff0c;更多地是将 Vue 的核心功能&…

项目相互依赖调用解决方法两种方法

Bmodel依赖于Amodel&#xff0c;但是Amodel又需要BModel的信息。原来是在Amodel创建一块内存&#xff0c;在Bmodel中将内存地址赋给这块内存&#xff0c;然后在Amodel去做其他操作。 方法一&#xff1a;采用静态变量static链接&#xff1a;C开发中一个解决方案里&#xff0c;两…

LeetCode 0808. 分汤:好题【感叹号】

【LetMeFly】808.分汤&#xff1a;好题&#xff01; 力扣题目链接&#xff1a;https://leetcode.cn/problems/soup-servings/ 有 A 和 B 两种类型 的汤。一开始每种类型的汤有 n 毫升。有四种分配操作&#xff1a; 提供 100ml 的 汤A 和 0ml 的 汤B 。提供 75ml 的 汤A 和 2…

Google Earth Engine(GEE)—— 各矿区时序NDVI变化图(包含具体的运行函数)

函数: ee.Filter.eq(name, value) Filter to metadata equal to the given value. Returns the constructed filter. Arguments: name (String): The property name to filter on. value (Object): The value to compare against. Returns: Filter ui.Chart.image.s…

7、Jedis测试

文章目录7、Jedis测试7.1. Jedis所需要的jar包7.2. 连接Redis注意事项7.3. Jedis常用操作7.3.1. 创建动态的工程7.3.2. 创建测试程序7.4. 测试相关数据类型7.4.1. Jedis-API&#xff1a;Key7.4.2. Jedis-API&#xff1a;String7.4.3. Jedis-API&#xff1a;List7.4.4. Jedis-AP…

葡萄糖-聚乙二醇-6-羧甲基荧光素 Glucose-PEG-6-FAM

葡萄糖-聚乙二醇-6-羧甲基荧光素 Glucose-PEG-6-FAM 中文名称&#xff1a;葡萄糖-6-羧甲基荧光素 英文名称&#xff1a;Glucose-6-FAM 别称&#xff1a;6-羧甲基荧光素标记葡萄糖&#xff0c;6-羧甲基荧光素-葡萄糖 PEG接枝修饰葡萄糖 Glucose-PEG-6-FAM 葡萄糖-聚乙二醇…

需求收集方法工具,以及进行需求分析的6大要素

通过本文你将了解&#xff1a;1、需求管理流程包括哪四个步骤&#xff1b;2、如何进行需求收集&#xff1b;3、如何进行需求分析&#xff1f;4、如何进行需求分发&#xff1b;5、如何进行需求验证&#xff1b;6、有哪些辅助软件需求管理的工具系统&#xff1f;一、需求管理包括…