当前位置: 首页 > news >正文

【Linux探索学习】第二十七弹——信号(一):Linux 信号基础详解

Linux学习笔记:

https://blog.csdn.net/2301_80220607/category_12805278.html?spm=1001.2014.3001.5482

前言:

前面我们已经将进程通信部分讲完了,现在我们来讲一个进程部分也非常重要的知识点——信号,信号也是进程间通信的一种,本篇主要讲解信号的概念和信号的几种产生方法及对应的场景

目录

一、引言

二、信号的概念

2.1 什么是信号

2.2 信号的作用

2.3 信号的特点

2.4 常见信号列表

​编辑

三、信号的产生

3.1 前台进程和后台进程

3.2 用户产生信号

3.3 系统产生信号

3.4 软件产生信号

四、信号的处理

4.1 默认处理方式

4.2 自定义信号处理函数

五、总结


一、引言

在 Linux 操作系统中,信号(Signal)是一种进程间通信(IPC,Inter - Process Communication)的机制,它用于通知进程发生了某种异步事件。信号可以来自内核,也可以来自其他进程。进程接收到信号后,会根据信号的类型以及自身的处理方式做出相应的反应。理解信号对于编写健壮的 Linux 程序以及深入理解 Linux 操作系统的运行机制至关重要。

二、信号的概念

2.1 什么是信号

信号是一种软中断,它是一种异步通知机制。当某个特定事件发生时,如用户按下特定组合键、系统资源耗尽、进程异常终止等,系统会向相关进程发送一个信号。每个信号都有一个对应的编号和名称,例如信号 1 表示 SIGHUP(挂起信号),信号 9 表示 SIGKILL(强制终止信号)。

2.2 信号的作用

信号的主要作用是让进程能够对异步事件做出响应。例如,当用户在终端中按下 Ctrl + C 组合键时,系统会向当前前台进程发送 SIGINT 信号,通常进程会接收到这个信号后停止当前正在执行的任务并退出。信号还可以用于进程间的通信,一个进程可以向另一个进程发送信号来通知其执行某些操作。

结合2.1和2.2我们来讲解一个概念:信号是一种软中断,是什么意思呢?当我们往键盘中输入内容时是如何告诉给内核的?ctrl+c又是如何被解释为指令的呢?

我们先来看下面这张图:

        键盘实际上是通过中断来让操作系统知道自己要写入内容的,键盘被按下时,就会触发硬件中断,不同的硬件对应着不同的中断号,中断单元就可以通过它们的中断号将它们与CPU中不同的键位相连,从而使CPU中这个方向的寄存器(32位)特定位置产生电信号,操作系统中有一个叫中断向量表的类似于函数指针结构体的结构,里面保存着访问各种外设的方法,操作系统通过CPU产生的电信号就辨别出要获取哪种硬件的信息,从而通过中断向量表中的方法,将硬件中的信息拷贝到操作系统的文件缓冲区中(操作系统下一切皆文件,且每一个文件都有自己的文件缓冲中区),然后再拷贝到用户缓冲区
       同时比如键盘等外键,操作系统在获取键盘上的信息时会先进行识别,会对数据进行判断,如果是控制进程的比如ctrl+c等组合键就不会往缓冲区中拷贝,我们可以发现我们学习的信号与上面的中断过程很像,其实信号,就是用软件方式,模拟的对讲程的硬件中断,所以信号也被叫做软中断

2.3 信号的特点

  1. 异步性:信号的产生是异步的,与进程的执行顺序无关。进程在运行过程中可能随时收到信号。
  1. 简单性:信号机制相对简单,只需要一个信号编号就可以标识不同的信号。
  1. 有限性:Linux 系统中定义的信号数量是有限的,不同的系统可能略有差异,但通常在几十种左右。

2.4 常见信号列表

信号编号

信号名称

含义

默认处理方式

1

SIGHUP

挂起信号,通常在终端关闭时发送给相关进程

终止进程

2

SIGINT

中断信号,由用户按下 Ctrl + C 组合键产生

终止进程

3

SIGQUIT

退出信号,由用户按下 Ctrl + \ 组合键产生

终止进程并生成核心转储文件

9

SIGKILL

强制终止信号,不能被捕获、阻塞或忽略

立即终止进程

15

SIGTERM

终止信号,通常用于正常终止进程

终止进程

18

SIGCONT

继续信号,用于恢复被暂停的进程

继续执行进程

19

SIGSTOP

停止信号,用于暂停进程,不能被捕获、阻塞或忽略

暂停进程

可以通过kill -l指令查看所有信号

kill -l

三、信号的产生

3.1 前台进程和后台进程

先来科普一个小知识点:前台进程和后台进程,来看下面一个程序

#include<iostream>
#include<unistd.h>
using namespace std;
int main()
{while(true){cout<<"I am a crazy process"<<endl;sleep(1);}return 0;
}

我们进行编译后会得到一个可执行程序

./myfile

我们这样执行时我们会发现在程序运行的时候,我们输入其它指令比如Is,pwd等都不会有结果,进程还在继续运行,除非用ctrl+c终止掉进程,这样的进程称为前台进程

./myfile &

这种的后面加上地址符的叫做后台进程,后台进程可以被其它进程命令临时打断并执行这个命令,比如我们输入ls指令,进程就会暂停并且输出Is的结果,但是最后需要自己把进程结束掉

Linux中,一次登陆中, 一个终端,一般会配上一个bash,每一个登陆,只允许一个进程是前台进程,可以允许多个进程是后台进程
当./process运行时,输入指令之所以不能运行就是因为此时的前台进程由bash转变为了process

  • 终端占用情况
    • 前台进程:会独占终端,直到进程执行完成或者被挂起,在这期间终端无法接受其他命令输入,用户只能与该进程进行交互。
    • 后台进程:不会占用终端,终端可以继续接受用户输入的其他命令,用户可以在同一个终端中同时启动多个后台进程,并随时切换到其他任务。
  • 运行特性
    • 前台进程:其执行过程会受到用户操作的直接影响,比如用户可以通过键盘输入来中断或暂停进程。如果终端关闭,前台进程通常会被终止,除非进行了特殊的设置。
    • 后台进程:通常是长时间运行的,不受终端关闭的影响,除非明确地对其进行停止或重启操作。它按照自身的逻辑和任务需求在后台持续运行,不会因为用户的一些常规操作而中断。

3.2 用户产生信号

  1. 键盘输入:用户可以通过在终端中按下特定的组合键来产生信号。例如:
    • Ctrl + C:产生 SIGINT 信号,用于中断当前正在运行的进程。比如,我们在终端中运行一个长时间运行的命令while true; do echo "Hello"; sleep 1; done,按下 Ctrl + C 后,该命令对应的进程会接收到 SIGINT 信号并终止。
    • Ctrl + \:产生 SIGQUIT 信号,不仅会终止进程,还会生成核心转储文件(如果系统配置允许,一般在云服务器上是默认关闭的,虚拟机上可能是开启的)。例如,运行一个简单的 C 程序#include <stdio.h> int main() { while(1); return 0; },编译运行后,按下 Ctrl + \,进程会终止并生成核心转储文件(在当前目录下,文件名为 core,具体名称和位置可能因系统配置而异)。(了解即可,这个生成core文件的内容与进程退出部分也有联系,有想了解的可以单独去搜索一下)
  1. 使用 kill 命令:用户可以使用 kill 命令向指定进程发送信号。kill 命令的基本语法是kill [信号编号] 进程ID。例如,要向进程 ID 为 1234 的进程发送 SIGTERM 信号(信号编号为 15),可以在终端中输入kill -15 1234,也可以使用信号名称kill -SIGTERM 1234。如果省略信号编号或名称,默认发送 SIGTERM 信号。

3.3 系统产生信号

  1. 进程异常:当进程发生异常时,如段错误(访问非法内存地址)、除零错误等,系统会向该进程发送相应的信号。
    • 段错误(Segmentation Fault):当进程访问了不属于它的内存区域时,会产生段错误,一般都是野指针问题,系统会向该进程发送 SIGSEGV 信号。例如,下面的 C 代码会导致段错误:
#include <stdio.h>int main() {int *ptr = NULL;*ptr = 10; // 试图向空指针指向的地址写入数据,会引发段错误return 0;}

编译运行这段代码,程序会崩溃,并提示 “Segmentation fault”,这是因为进程接收到了 SIGSEGV 信号。

  • 除零错误(Division by Zero):当进程执行除法运算时,如果除数为零,会产生除零错误,系统会向该进程发送 SIGFPE 信号。例如:
#include <stdio.h>int main()
{int a = 10;int b = 0;int c = a / b; // 除零操作,会引发除零错误return 0;
}

运行这段代码,程序会崩溃,并提示 “Floating point exception”,这是因为进程接收到了 SIGFPE 信号。

2. 系统资源相关:当系统资源达到一定阈值时,也可能产生信号。例如,当进程使用的内存超过了系统限制时,系统可能会发送 SIGKILL 信号来终止该进程,以防止系统内存耗尽。不过,这种情况通常需要系统进行相关的配置和监控。

3.4 软件产生信号

  1. 使用 kill 函数:在 C 语言编程中,可以使用 kill 函数向指定进程发送信号。kill 函数的原型可以用man手册查看,如下:
 man 2 kill

其中,pid 是目标进程的 ID,sig 是要发送的信号编号。例如,下面的代码演示了如何使用 kill 函数向另一个进程发送 SIGTERM 信号:

#include <stdio.h>
#include <sys/types.h>
#include <signal.h>
#include <unistd.h>
int main()
{pid_t target_pid = 1234; // 假设目标进程ID为1234int result = kill(target_pid, SIGTERM);if (result == -1){perror("kill failed");}else{printf("SIGTERM sent to process %d\n", target_pid);}return 0;
}

在实际使用中,需要将target_pid替换为真实的目标进程 ID。

2. 使用 raise 函数:进程可以使用 raise 函数向自身发送信号。raise 函数的原型也可以通过man手册来查看,如下:

man raise

其中,sig 是要发送的信号编号。例如,下面的代码演示了如何使用 raise 函数向自身发送 SIGINT 信号:

#include <stdio.h>
#include <signal.h>
int main()
{int result = raise(SIGINT);if (result != 0){perror("raise failed");}else{printf("SIGINT sent to self\n");}return 0;
}

运行这段代码,进程会接收到自己发送的 SIGINT 信号并终止。

四、信号的处理

4.1 默认处理方式

每个信号都有一个默认的处理方式,常见的默认处理方式包括:

  1. 终止进程:如 SIGINT、SIGTERM 等信号的默认处理方式是终止进程。
  1. 生成核心转储文件并终止进程:例如 SIGQUIT 信号,在终止进程的同时会生成核心转储文件,该文件包含了进程在收到信号时的内存状态等信息,可用于调试程序。
  1. 忽略信号:有些信号(如 SIGCHLD,子进程状态改变时发送给父进程的信号)的默认处理方式是忽略。

4.2 自定义信号处理函数

进程可以通过调用 signal 函数或 sigaction 函数来设置自定义的信号处理函数。

  1. signal 函数:signal 函数的原型如下:
man signal

其中,signum 是信号编号,handler 是指向信号处理函数的指针。例如,下面的代码演示了如何使用 signal 函数设置 SIGINT 信号的自定义处理函数:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
void signal_handler(int signum)
{printf("Received SIGINT. Cleaning up...\n");// 在这里进行一些清理工作,如关闭文件、释放资源等_exit(0); // 退出进程
}
int main()
{signal(SIGINT, signal_handler);while (1){printf("Running...\n");sleep(1);}return 0;
}

在这个例子中,当进程接收到 SIGINT 信号时,会调用signal_handler函数,而不是默认的终止进程操作。

2. sigaction 函数:sigaction 函数比 signal 函数提供了更丰富的功能,它可以设置信号处理函数、处理信号时的掩码、信号的标志等。sigaction 函数的原型如下:

#include <signal.h>int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);struct sigaction {void (*sa_handler)(int);void (*sa_sigaction)(int, siginfo_t *, void *);sigset_t sa_mask;int sa_flags;void (*sa_restorer)(void);};

其中,signum 是信号编号,act 是指向新的信号处理动作的结构体指针,oldact 是指向旧的信号处理动作的结构体指针(如果不需要获取旧的处理动作,可以设为 NULL)。例如,下面的代码演示了如何使用 sigaction 函数设置 SIGINT 信号的自定义处理函数:

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
void signal_handler(int signum)
{printf("Received SIGINT. Cleaning up...\n");// 在这里进行一些清理工作,如关闭文件、释放资源等_exit(0); // 退出进程
}
int main()
{struct sigaction new_action, old_action;new_action.sa_handler = signal_handler;sigemptyset(&new_action.sa_mask);new_action.sa_flags = 0;sigaction(SIGINT, &new_action, &old_action);while (1){printf("Running...\n");sleep(1);}return 0;
}

这段代码与使用 signal 函数的例子功能类似,但使用 sigaction 函数可以更灵活地配置信号处理方式。

五、总结

信号是 Linux 系统中一种重要的进程间通信和异步事件通知机制。通过本文,我们详细了解了信号的概念,信号的产生和部分信号的处理工作,后面我们还会讲解信号的捕捉等处理工作,学习信号可以帮助我们更好的实现进程通信和异步处理等诸多操作

本篇笔记:


感谢各位大佬观看,创作不易,还请各位大佬点赞支持!!!

相关文章:

【Linux探索学习】第二十七弹——信号(一):Linux 信号基础详解

Linux学习笔记&#xff1a; https://blog.csdn.net/2301_80220607/category_12805278.html?spm1001.2014.3001.5482 前言&#xff1a; 前面我们已经将进程通信部分讲完了&#xff0c;现在我们来讲一个进程部分也非常重要的知识点——信号&#xff0c;信号也是进程间通信的一…...

【数学】矩阵、向量(内含矩阵乘法C++)

目录 一、前置知识&#xff1a;向量&#xff08;一列或一行的矩阵&#xff09;、矩阵1. 行向量2. 列向量3. 向量其余基本概念4. 矩阵基本概念5. 关于它们的细节 二、运算1. 转置&#xff08;1&#xff09;定义&#xff08;2&#xff09;性质 2. 矩阵&#xff08;向量&#xff0…...

设置git区分大小写

设置git区分大小写 1.全局设置 (影响全部仓库): git config --global core.ignorecase false2.仓库级别设置 (影响当前仓库): git config core.ignorecase false3.已经提交了大小写不一致的文件处理: git mv -f OldName newName # 强制重命名 git commit -m "Fix cas…...

排序算法与查找算法

1.十大经典排序算法 我们希望数据以一种有序的形式组织起来&#xff0c;无序的数据我们要尽量将其变得有序 一般说来有10种比较经典的排序算法 简单记忆为Miss D----D小姐 时间复杂度 &#xff1a;红色<绿色<蓝色 空间复杂度&#xff1a;圆越大越占空间 稳定性&…...

Github 2025-01-31Java开源项目日报 Top10

根据Github Trendings的统计,今日(2025-01-31统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Java项目10C项目1Kotlin项目1Bazel:快速、可扩展的多语言构建系统 创建周期:3564 天开发语言:Java协议类型:Apache License 2.0Star数量:2…...

Java进阶笔记(中级)

-----接Java进阶笔记&#xff08;初级&#xff09;----- 目录 集合多线程 集合 ArrayList 可以通过List来接收ArrayList对象&#xff08;因为ArrayList实现了List接口&#xff09; 方法&#xff1a;接口名 柄名 new 实现了接口的类(); PS: List list new ArrayList();遍历…...

2025游戏行业的趋势预测

一、市场现状 从总产值的角度来看&#xff0c;游戏总产值的增长率已经放缓&#xff0c;由增量市场转化为存量市场&#xff0c;整体的竞争强度将会加大&#xff0c;技术水平不强&#xff08;开发技术弱、产品品质低、开发效率低&#xff09;的公司将会面临更大的生存的困难。 从…...

4-ET框架demo的运行

开始操作 开始运行报错 编译Unity项目 状态同步demo运行 1-设置构建参数 CodeMode&#xff1a;ClientServer EPlayMode:EditorSimulateMode 点击ReGeneratoePerojectFiles调整代码结构 2-编译 在Unity.sln下编译项目 3-运行 以帧同步模式运行 1-合端运行 1.1 修改帧同步标记 …...

kamailio源文件modules.lst的内容解释

在执行make cfg 后&#xff0c;在kamailio/src目录下有一个文件modules.lst&#xff0c;内容如下&#xff1a; # this file is autogenerated by make modules-cfg# the list of sub-directories with modules modules_dirs:modules# the list of module groups to compile cf…...

亚远景-从SPICE到ASPICE:汽车软件开发的标准化演进

一、SPICE标准的起源与背景 SPICE&#xff0c;全称“Software Process Improvement and Capability dEtermination”&#xff0c;即“软件流程改进和能力测定”&#xff0c;是由国际标准化组织ISO、国际电工委员会IEC、信息技术委员会JTC1联合发起制定的ISO 15504标准。该标准旨…...

vue3 + ElementPlus 封装列表表格组件包含分页

在前端开发中&#xff0c;封装组件是必不可少的。今天就来封装一个通用的列表表格组件&#xff0c;包含分页功能&#xff0c;可以提高代码的复用性和可维护性。 1. 组件设计 Props&#xff1a; tableData&#xff1a;表格数据。columns&#xff1a;表格列配置。total&#xff…...

挑战项目 --- 微服务编程测评系统(在线OJ系统)

一、前言 1.为什么要做项目 面试官要问项目&#xff0c;考察你到底是理论派还是实战派&#xff1f; 1.希望从你的项目中看到你的真实能力和对知识的灵活运用。 2.展示你在面对问题和需求时的思考方式及解决问题的能力。 3.面试官会就你项目提出一些问题&#xff0c;或扩展需求…...

Med-R2:基于循证医学的检索推理框架:提升大语言模型医疗问答能力的新方法

Med-R2 : Crafting Trustworthy LLM Physicians through Retrieval and Reasoning of Evidence-Based Medicine Med-R2框架Why - 这个研究要解决什么现实问题What - 核心发现或论点是什么How - 1. 前人研究的局限性How - 2. 你的创新方法/视角How - 3. 关键数据支持How - 4. 可…...

Oh3.2项目升级到Oh5.0(鸿蒙Next)具体踩坑记录(一)

目录 1.自动修复部分 Cause: The project structure and configuration require an upgrade. Solution: 1. Use Migrate Assistant to auto-upgrade the project structure and configuration. 2. Manually upgrade the project structure and configuration by following th…...

【自动化办公】批量图片PDF自定义指定多个区域识别重命名,批量识别铁路货物运单区域内容改名,基于WPF和飞桨ocr深度学习模型的解决方案

项目背景介绍 铁路货运企业需要对物流单进行长期存档&#xff0c;以便后续查询和审计。不同的物流单可能包含不同的关键信息&#xff0c;通过自定义指定多个区域进行识别重命名&#xff0c;可以使存档的图片文件名具有统一的规范和明确的含义。比如&#xff0c;将包含货物运单…...

Spring Boot篇

为什么要用Spring Boot Spring Boot 优点非常多&#xff0c;如&#xff1a; 独立运行 Spring Boot 而且内嵌了各种 servlet 容器&#xff0c;Tomcat、Jetty 等&#xff0c;现在不再需要打成 war 包部署到 容器 中&#xff0c;Spring Boot 只要打成一个可执行的 jar 包就能独…...

Unity3D学习笔记(二)

一、Unity编辑器相关 1、 Unity特殊的专属文件夹 1&#xff09; Editor&#xff1a;编辑器相关资源可以放到此文件中&#xff0c;包括图片、脚本等文件。 2&#xff09;Editor Default Resources:配合Editor使用不会打包到包中 3&#xff09;Plugins&#xff1a;存放第三方SD…...

个人毕业设计--基于HarmonyOS的旅行助手APP的设计与实现(挖坑)

在行业混了短短几年&#xff0c;却总感觉越混越迷茫&#xff0c;趁着还有心情学习&#xff0c;把当初API9 的毕业设计项目改成API13的项目。先占个坑&#xff0c;把当初毕业设计的文案搬过来 摘要&#xff1a;HarmonyOS&#xff08;鸿蒙系统&#xff09;是华为公司推出的面向全…...

游戏引擎 Unity - Unity 打开项目、Unity Editor 添加简体中文语言包模块、Unity 项目设置为简体中文

Unity Unity 首次发布于 2005 年&#xff0c;属于 Unity Technologies Unity 使用的开发技术有&#xff1a;C# Unity 的适用平台&#xff1a;PC、主机、移动设备、VR / AR、Web 等 Unity 的适用领域&#xff1a;开发中等画质中小型项目 Unity 适合初学者或需要快速上手的开…...

python开发:爬虫示例——GET和POST请求处理

一、Get请求 import json import requests#输入示例&#xff1a;urlhttps://www.baidu.com #RequestHeader:F12标头-请求标头-原始-复制到这&#xff08;忽略第一句&#xff09; def GetRequest(url,RequestHeader""):try:dic{}RequestHeaderList RequestHeader.s…...

MPNet:旋转机械轻量化故障诊断模型详解python代码复现

目录 一、问题背景与挑战 二、MPNet核心架构 2.1 多分支特征融合模块(MBFM) 2.2 残差注意力金字塔模块(RAPM) 2.2.1 空间金字塔注意力(SPA) 2.2.2 金字塔残差块(PRBlock) 2.3 分类器设计 三、关键技术突破 3.1 多尺度特征融合 3.2 轻量化设计策略 3.3 抗噪声…...

rknn优化教程(二)

文章目录 1. 前述2. 三方库的封装2.1 xrepo中的库2.2 xrepo之外的库2.2.1 opencv2.2.2 rknnrt2.2.3 spdlog 3. rknn_engine库 1. 前述 OK&#xff0c;开始写第二篇的内容了。这篇博客主要能写一下&#xff1a; 如何给一些三方库按照xmake方式进行封装&#xff0c;供调用如何按…...

大数据零基础学习day1之环境准备和大数据初步理解

学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 &#xff08;1&#xff09;设置网关 打开VMware虚拟机&#xff0c;点击编辑…...

python爬虫:Newspaper3k 的详细使用(好用的新闻网站文章抓取和解析的Python库)

更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 一、Newspaper3k 概述1.1 Newspaper3k 介绍1.2 主要功能1.3 典型应用场景1.4 安装二、基本用法2.2 提取单篇文章的内容2.2 处理多篇文档三、高级选项3.1 自定义配置3.2 分析文章情感四、实战案例4.1 构建新闻摘要聚合器…...

NFT模式:数字资产确权与链游经济系统构建

NFT模式&#xff1a;数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新&#xff1a;构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议&#xff1a;基于LayerZero协议实现以太坊、Solana等公链资产互通&#xff0c;通过零知…...

C/C++ 中附加包含目录、附加库目录与附加依赖项详解

在 C/C 编程的编译和链接过程中&#xff0c;附加包含目录、附加库目录和附加依赖项是三个至关重要的设置&#xff0c;它们相互配合&#xff0c;确保程序能够正确引用外部资源并顺利构建。虽然在学习过程中&#xff0c;这些概念容易让人混淆&#xff0c;但深入理解它们的作用和联…...

【p2p、分布式,区块链笔记 MESH】Bluetooth蓝牙通信 BLE Mesh协议的拓扑结构 定向转发机制

目录 节点的功能承载层&#xff08;GATT/Adv&#xff09;局限性&#xff1a; 拓扑关系定向转发机制定向转发意义 CG 节点的功能 节点的功能由节点支持的特性和功能决定。所有节点都能够发送和接收网格消息。节点还可以选择支持一个或多个附加功能&#xff0c;如 Configuration …...

DBLP数据库是什么?

DBLP&#xff08;Digital Bibliography & Library Project&#xff09;Computer Science Bibliography是全球著名的计算机科学出版物的开放书目数据库。DBLP所收录的期刊和会议论文质量较高&#xff0c;数据库文献更新速度很快&#xff0c;很好地反映了国际计算机科学学术研…...

0x-3-Oracle 23 ai-sqlcl 25.1 集成安装-配置和优化

是不是受够了安装了oracle database之后sqlplus的简陋&#xff0c;无法删除无法上下翻页的苦恼。 可以安装readline和rlwrap插件的话&#xff0c;配置.bahs_profile后也能解决上下翻页这些&#xff0c;但是很多生产环境无法安装rpm包。 oracle提供了sqlcl免费许可&#xff0c…...

规则与人性的天平——由高考迟到事件引发的思考

当那位身着校服的考生在考场关闭1分钟后狂奔而至&#xff0c;他涨红的脸上写满绝望。铁门内秒针划过的弧度&#xff0c;成为改变人生的残酷抛物线。家长声嘶力竭的哀求与考务人员机械的"这是规定"&#xff0c;构成当代中国教育最尖锐的隐喻。 一、刚性规则的必要性 …...