【pl/sql番外篇】 存储过程 游标

 SELECT语句用于从数据库中查询数据,当在PL/SQL中使用SELECT语句时,要与INTO子句一起使用,查询的返回值被赋予INTO子句中的变量,变量的声明是在DELCARE中。SELECT             INTO语法如下: 
     SELECT [DISTICT|ALL]{*|column[,column,...]} 
     INTO (variable[,variable,...] |record) 
     FROM {table|(sub-query)}[alias] 
     WHERE............ 
    PL/SQL中SELECT语句只返回一行数据。如果超过一行数据,那么就要使用显式游标(对游标的讨论我们将在后面进行),INTO子句中要有与SELECT子句中相同列数量的变量。INTO子句中也可以是记录变量。

%TYPE属性 
     在PL/SQL中可以将变量和常量声明为内建或用户定义的数据类型,以引用一个列名,同时继承他的数据类型和大小。这种动态赋值方法是非常有用的,比如变量引用的列的数据类型和大小改变了,如果使用了%TYPE,那么用户就不必修改代码,否则就必须修改代码。

例: 
v_empno SCOTT.EMP.EMPNO%TYPE; 
v_salary EMP.SALARY%TYPE; 
 不但列名可以使用%TYPE,而且变量、游标、记录,或声明的常量都可以使用%TYPE。这对于定义相同数据类型的变量非常有用。 
    DELCARE 
    V_A NUMBER(5):=10; 
    V_B V_A%TYPE:=15; 
    V_C V_A%TYPE; 
    BEGIN 
      DBMS_OUTPUT.PUT_LINE 
      ('V_A='||V_A||'V_B='||V_B||'V_C='||V_C); 
    END 
     
    SQL>/ 
    V_A=10 V_B=15 V_C= 
     PL/SQL procedure successfully completed. 
     SQL> 
      
其他DML语句 
    其它操作数据的DML语句是:INSERT、UPDATE、DELETE和LOCK TABLE,这些语句在PL/SQL中的语法与在SQL中的语法相同。我们在前面已经讨论过DML语句的使用这里就不再重复了。在DML语句中可以使用任何在DECLARE部分声明的变量,如果是嵌套块,那么要注意变量的作用范围。

例: 
CREATE OR REPLACE PROCEDURE FIRE_EMPLOYEE (pempno in number) 
 AS 
    v_ename EMP.ENAME%TYPE; 
    BEGIN 
     SELECT ename INTO v_ename 
      FROM emp 
      WHERE empno=p_empno; 
      INSERT INTO FORMER_EMP(EMPNO,ENAME) 
      VALUES (p_empno,v_ename); 
      DELETE FROM emp 
      WHERE empno=p_empno; 
      UPDATE former_emp 
      SET date_deleted=SYSDATE 
      WHERE empno=p_empno; 
       
    EXCEPTION 
       WHEN NO_DATA_FOUND THEN 
       DBMS_OUTPUT.PUT_LINE('Employee Number Not Found!'); 
    END

DML语句的结果 
    当执行一条DML语句后,DML语句的结果保存在四个游标属性中,这些属性用于控制程序流程或者了解程序的状态。当运行DML语句时,PL/SQL打开一个内建游标并处理结果,游标是维护查询结果的内存中的一个区域,游标在运行DML语句时打开,完成后关闭。隐式游标只使用SQL%FOUND,SQL%NOTFOUND,SQL%ROWCOUNT三个属性.SQL%FOUND,SQL%NOTFOUND是布尔值,SQL%ROWCOUNT是整数值。

SQL%FOUND和SQL%NOTFOUND 
    在执行任何DML语句前SQL%FOUND和SQL%NOTFOUND的值都是NULL,在执行DML语句后,SQL%FOUND的属性值将是:

. TRUE :INSERT 
. TRUE ELETE和UPDATE,至少有一行被DELETE或UPDATE. 
. TRUE :SELECT INTO至少返回一行 
当SQL%FOUND为TRUE时,SQL%NOTFOUND为FALSE。

SQL%ROWCOUNT 
   在执行任何DML语句之前,SQL%ROWCOUNT的值都是NULL,对于SELECT             INTO语句,如果执行成功,SQL%ROWCOUNT的值为1,如果没有成功,SQL%ROWCOUNT的值为0,同时产生一个异常NO_DATA_FOUND.

SQL%ISOPEN 
SQL%ISOPEN是一个布尔值,如果游标打开,则为TRUE, 如果游标关闭,则为FALSE.对于隐式游标而言SQL%ISOPEN总是FALSE,这是因为隐式游标在DML语句执行时打开,结束时就立即关闭。

事务控制语句 
    事务是一个工作的逻辑单元可以包括一个或多个DML语句,事物控制帮助用户保证数据的一致性。如果事务控制逻辑单元中的任何一个DML语句失败,那么整个事务都将回滚,在PL/SQL中用户可以明确地使用COMMIT、ROLLBACK、SAVEPOINT以及SET TRANSACTION语句。 
     COMMIT语句终止事务,永久保存数据库的变化,同时释放所有LOCK,ROLLBACK终止现行事务释放所有LOCK,但不保存数据库的任何变化,SAVEPOINT用于设置中间点,当事务调用过多的数据库操作时,中间点是非常有用的,SET TRANSACTION用于设置事务属性,比如read-write和隔离级等。

显式游标 
    当查询返回结果超过一行时,就需要一个显式游标,此时用户不能使用select into语句。PL/SQL管理隐式游标,当查询开始时隐式游标打开,查询结束时隐式游标自动关闭。显式游标在PL/SQL块的声明部分声明,在执行部分或异常处理部分打开,取数据,关闭。

使用游标 
    这里要做一个声明,我们所说的游标通常是指显式游标,因此从现在起没有特别指明的情况,我们所说的游标都是指显式游标。要在程序中使用游标,必须首先声明游标。

声明游标 
语法: 
    CURSOR cursor_name IS select_statement;

在PL/SQL中游标名是一个未声明变量,不能给游标名赋值或用于表达式中。

例: 
    DELCARE 
    CURSOR C_EMP IS SELECT empno,ename,salary 
    FROM emp 
    WHERE salary>2000 
    ORDER BY ename; 
    ........ 
    BEGIN 
    在游标定义中SELECT语句中不一定非要表可以是视图,也可以从多个表或视图中选择的列,甚至可以使用*来选择所有的列 。 
     
打开游标 
使用游标中的值之前应该首先打开游标,打开游标初始化查询处理。打开游标的语法是: 
    OPEN cursor_name 
       cursor_name是在声明部分定义的游标名。 
     
例: 
     OPEN C_EMP; 
         
关闭游标 
语法: 
     CLOSE cursor_name 
     
例: 
     CLOSE C_EMP;

从游标提取数据 
    从游标得到一行数据使用FETCH命令。每一次提取数据后,游标都指向结果集的下一行。语法如下: 
     FETCH cursor_name INTO variable[,variable,...] 
     对于SELECT定义的游标的每一列,FETCH变量列表都应该有一个变量与之相对应,变量的类型也要相同。

例: 
   SET SERVERIUTPUT ON 
   DECLARE 
     v_ename EMP.ENAME%TYPE; 
     v_salary EMP.SALARY%TYPE; 
     CURSOR c_emp IS SELECT ename,salary FROM emp; 
     BEGIN 
       OPEN c_emp; 
          FETCH c_emp INTO v_ename,v_salary; 
            DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary); 
          FETCH c_emp INTO v_ename,v_salary; 
            DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary); 
          FETCH c_emp INTO v_ename,v_salary; 
            DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary); 
       CLOSE c_emp; 
     END 
      
    这段代码无疑是非常麻烦的,如果有多行返回结果,可以使用循环并用游标属性为结束循环的条件,以这种方式提取数据,程序的可读性和简洁性都大为提高,下面我们使用循环重新写上面的程序: 
SET SERVERIUTPUT ON 
DECLARE 
v_ename EMP.ENAME%TYPE; 
v_salary EMP.SALARY%TYPE; 
CURSOR c_emp IS SELECT ename,salary FROM emp; 
BEGIN 
OPEN c_emp; 
    LOOP 
      FETCH c_emp INTO v_ename,v_salary; 
      EXIT WHEN c_emp%NOTFOUND; 
      DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename ||'is'|| v_salary); 
END

记录变量 
    定义一个记录变量使用TYPE命令和%ROWTYPE,关于%ROWsTYPE的更多信息请参阅相关资料。 
    记录变量用于从游标中提取数据行,当游标选择很多列的时候,那么使用记录比为每列声明一个变量要方便得多。 
    当在表上使用%ROWTYPE并将从游标中取出的值放入记录中时,如果要选择表中所有列,那么在SELECT子句中使用*比将所有列名列出来要安全得多。

例: 
SET SERVERIUTPUT ON 
DECLARE 
R_emp EMP%ROWTYPE; 
CURSOR c_emp IS SELECT * FROM emp; 
BEGIN 
OPEN c_emp; 
   LOOP 
     FETCH c_emp INTO r_emp; 
     EXIT WHEN c_emp%NOTFOUND; 
     DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary); 
   END LOOP; 
CLOSE c_emp; 
END;

%ROWTYPE也可以用游标名来定义,这样的话就必须要首先声明游标:

SET SERVERIUTPUT ON 
DECLARE 
CURSOR c_emp IS SELECT ename,salary FROM emp; 
R_emp c_emp%ROWTYPE; 
BEGIN 
OPEN c_emp; 
LOOP 
    FETCH c_emp INTO r_emp; 
    EXIT WHEN c_emp%NOTFOUND; 
    DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary); 
END LOOP; 
CLOSE c_emp; 
END;

带参数的游标 
    与存储过程和函数相似,可以将参数传递给游标并在查询中使用。这对于处理在某种条件下打开游标的情况非常有用。它的语法如下:

CURSOR cursor_name[(parameter[,parameter],...)] IS select_statement;

定义参数的语法如下: 
   Parameter_name [IN] data_type[{:=|DEFAULT} value]

  与存储过程不同的是,游标只能接受传递的值,而不能返回值。参数只定义数据类型,没有大小。 
   另外可以给参数设定一个缺省值,当没有参数值传递给游标时,就使用缺省值。游标中定义的参数只是一个占位符,在别处引用该参数不一定可靠。

在打开游标时给参数赋值,语法如下:

OPEN cursor_name[value[,value]....]; 
参数值可以是文字或变量。

例: 
DECALRE 
CURSOR c_dept IS SELECT * FROM dept ORDER BY deptno; 
CURSOR c_emp (p_dept VARACHAR2) IS 
SELECT ename,salary 
FROM emp 
WHERE deptno=p_dept 
ORDER BY ename 
r_dept DEPT%ROWTYPE; 
v_ename EMP.ENAME%TYPE; 
v_salary EMP.SALARY%TYPE; 
v_tot_salary EMP.SALARY%TYPE; 
BEGIN 
OPEN c_dept; 
     LOOP 
        FETCH c_dept INTO r_dept; 
        EXIT WHEN c_dept%NOTFOUND; 
        DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname); 
        v_tot_salary:=0; 
        OPEN c_emp(r_dept.deptno); 
            LOOP 
               FETCH c_emp INTO v_ename,v_salary; 
               EXIT WHEN c_emp%NOTFOUND; 
               DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary); 
               v_tot_salary:=v_tot_salary+v_salary; 
            END LOOP; 
        CLOSE c_emp; 
        DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary); 
     END LOOP; 
CLOSE c_dept; 
END;

游标FOR循环 
在大多数时候我们在设计程序的时候都遵循下面的步骤: 
1、打开游标 
2、开始循环 
3、从游标中取值 
4、检查那一行被返回 
5、处理 
6、关闭循环 
7、关闭游标 
    可以简单的把这一类代码称为游标用于循环。但还有一种循环与这种类型不相同,这就是FOR循环,用于FOR循环的游标按照正常的声明方式声明,它的优点在于不需要显式的打开、关闭、取数据,测试数据的存在、定义存放数据的变量等等。游标FOR循环的语法如下:

FOR record_name IN 
(corsor_name[(parameter[,parameter]...)] 
| (query_difinition) 
LOOP 
statements 
END LOOP;

下面我们用for循环重写上面的例子: 
DECALRE 
CURSOR c_dept IS SELECT deptno,dname FROM dept ORDER BY deptno; 
CURSOR c_emp (p_dept VARACHAR2) IS 
SELECT ename,salary 
FROM emp 
WHERE deptno=p_dept 
ORDER BY ename 
v_tot_salary EMP.SALARY%TYPE; 
BEGIN 
   FOR r_dept IN c_dept LOOP 
     DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname); 
     v_tot_salary:=0; 
     FOR r_emp IN c_emp(r_dept.deptno) LOOP 
    DBMS_OUTPUT.PUT_LINE('Name:' || v_ename || 'salary:' || v_salary);   
    v_tot_salary:=v_tot_salary+v_salary; 
     END LOOP; 
     DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary); 
END LOOP; 
END;

在游标FOR循环中使用查询 
    在游标FOR循环中可以定义查询,由于没有显式声明所以游标没有名字,记录名通过游标查询来定义。 
DECALRE 
v_tot_salary EMP.SALARY%TYPE; 
BEGIN 
FOR r_dept IN (SELECT deptno,dname FROM dept ORDER BY deptno) LOOP 
     DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname); 
     v_tot_salary:=0; 
     FOR r_emp IN (SELECT ename,salary 
               FROM emp 
               WHERE deptno=p_dept 
               ORDER BY ename) LOOP 
       DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary); 
       v_tot_salary:=v_tot_salary+v_salary; 
     END LOOP; 
DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary); 
END LOOP; 
END;

游标中的子查询 
    语法如下: 
     
CURSOR C1 IS SELECT * FROM emp 
WHERE deptno NOT IN (SELECT deptno 
   FROM dept 
   WHERE dname!='ACCOUNTING'); 
可以看出与SQL中的子查询没有什么区别。

游标中的更新和删除 
    在PL/SQL中依然可以使用UPDATE和DELETE语句更新或删除数据行。显式游标只有在需要获得多行数据的情况下使用。PL/SQL提供了仅仅使用游标就可以执行删除或更新记录的方法。 
UPDATE或DELETE语句中的WHERE CURRENT OF子串专门处理要执行UPDATE或DELETE操作的表中取出的最近的数据。要使用这个方法,在声明游标时必须使用FOR UPDATE子串,当对话使用FOR UPDATE子串打开一个游标时,所有返回集中的数据行都将处于行级(ROW-LEVEL)独占式锁定,其他对象只能查询这些数据行,不能进行UPDATE、DELETE或SELECT...FOR            UPDATE操作。

语法: 
    FOR UPDATE [OF [schema.]table.column[,[schema.]table.column].. 
    [nowait] 
     
    在多表查询中,使用OF子句来锁定特定的表,如果忽略了OF子句,那么所有表中选择的数据行都将被锁定。如果这些数据行已经被其他会话锁定,那么正常情况下ORACLE将等待,直到数据行解锁。

在UPDATE和DELETE中使用WHERE CURRENT OF子串的语法如下:

WHERE{CURRENT OF cursor_name|search_condition}

例: 
DELCARE 
CURSOR c1 IS SELECT empno,salary 
FROM emp 
WHERE comm IS NULL 
FOR UPDATE OF comm; 
v_comm NUMBER(10,2); 
BEGIN 
   FOR r1 IN c1 LOOP 
     IF r1.salary<500 THEN 
       v_comm:=r1.salary*0.25; 
     ELSEIF r1.salary<1000 THEN 
       v_comm:=r1.salary*0.20; 
     ELSEIF r1.salary<3000 THEN 
       v_comm:=r1.salary*0.15; 
     ELSE 
         v_comm:=r1.salary*0.12; 
     END IF; 
   UPDATE emp; 
   SET comm=v_comm 
   WHERE CURRENT OF c1l; 
   END LOOP; 
END

转载于:https://www.cnblogs.com/JSD1207ZX/p/9386356.html

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

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

相关文章

路由器开发(二)—— 路由器工作原理

当信息需要在两个网络之间传输时&#xff0c;常用路由器这种互连设备来负责数据的传输。路由器的主要工作是&#xff1a;路径的决定和数据包的转发&#xff08;从路由器一个接口输入&#xff0c;然后选择合适接口输出&#xff09;&#xff1b;维护路由表。 路由器工作的方式非常…

Android颜色渐变的分隔线(ListView)

2019独角兽企业重金招聘Python工程师标准>>> shape.xml xx <?xml version"1.0" encoding"utf-8"?><shape xmlns:android"http://schemas.android.com/apk/res/android" > <gradient android:startColor&qu…

iOS 层层推进实现代理模式

1.代理模式核心思想&#xff1a;A类委托B类做某件事&#xff0c;然后A类获取B类的执行的返回结果&#xff01; 举例&#xff1a;女孩想去买电影票&#xff0c;但是自己不亲自去而是委托男孩了解电影电影票信息&#xff0c;同时女孩获得男孩买票的结果&#xff0c;代码模拟实现&…

项目实践中Linux集群的总结和思考

2019独角兽企业重金招聘Python工程师标准>>> 前言&#xff1a;作为一名Linux/unix系统工程师、项目实施工程师&#xff0c;这几年一直在涉及到对外项目&#xff0c;经手过许多小中型网站的架构&#xff0c;F5、LVS及Nginx接触的都比较多&#xff0c;我想一种比较通俗…

路由器基础知识详解

第一章 网络互联 网络的根本目的非常简单&#xff1a;方便人们交换所获得的信息。但是网络的应用需求非常复杂&#xff1a;有的用户希望高带宽&#xff0c;但并不要求很长的传输距离&#xff1b;有的用户要求很长的距离&#xff0c;但对带宽要求很低&#xff1b;有的对网络的…

事务与锁机制

2019独角兽企业重金招聘Python工程师标准>>> 事务定义&#xff1a; 访问并可能更新数据库&#xff1a;一句或一组SQL&#xff0c;或者是一段程序&#xff0c;反正update了就是事务 ACID的4原则&#xff1a; 原子性&#xff1a; 一致性&#xff1a; 隔离性&#xff1…

路由器 VS OSI七层模型

OSI Open Source Initiative&#xff08;简称OSI&#xff0c;有译作开放源代码促进会、开放原始码组织&#xff09;是一个旨在推动开源软件发展的非盈利组织。OSI参考模型&#xff08;OSI/RM&#xff09;的全称是开放系统互连参考模型&#xff08;Open System Interconnection …

Linux运维系统工程师系列---04

和用户登录相关的文件用户登录是需要读取的文件1、/etc/profile2、/etc/bashrc3、/etc/profile.d/*4、~user/.bashrc5、~user/.bash_profile登录shell和非登录shell&#xff1a;[rootlocalhost ~]# su - usr4登录shell[rootlocalhost ~]# su usr4非登录shell非登录shell使用的是…

Linux Wireless架构总结

1、无线网络驱动(ath9k_htc) ath9k_htc是一个基于USB接口的SoftMAC无线网络适配器。为了其驱动能正常工作&#xff0c;首先必须调用usb_register来注册驱动定义的usb_driver&#xff0c;以借助USB Core的力量来处理与USB协议相关的事件。其代码如下&#xff1a;[cpp] view plai…

MySQL 日志文件 说明

MySQL 5.5 官方文档上有关日志的分类&#xff1a;By default, nologs are enabled. The following log-specific sections provide information about the server options that enable logging.--默认情况下&#xff0c;没有启动任何log&#xff0c;可以通过如下log 选项来启动…

Linux 下wifi 驱动开发(四)—— USB接口WiFi驱动浅析

前面学习了SDIO接口的WiFi驱动&#xff0c;现在我们来学习一下USB接口的WiFi驱动&#xff0c;二者的区别在于接口不同。而USB接口的设备驱动&#xff0c;我们前面也有学习&#xff0c;比如USB摄像头驱动、USB鼠标驱动&#xff0c;同样都符合LinuxUSB驱动结构&#xff1a; USB设…

我的第一程序语言python

我接触的第一门语言是python,因此对python 可能有着更深厚的感情。感觉学的时间长了&#xff0c;就像是把把它当成是一个朋友&#xff0c;一个伴侣&#xff0c;想去读懂他&#xff0c;理解他&#xff0c;就像是自己的身体可以收放自如。从接触python 到现在也快有五个月了&…

Linux 下wifi 驱动开发(三)—— SDIO接口WiFi驱动浅析

SDIO-Wifi模块是基于SDIO接口的符合wifi无线网络标准的嵌入式模块&#xff0c;内置无线网络协议IEEE802.11协议栈以及TCP/IP协议栈&#xff0c;能够实现用户主平台数据通过SDIO口到无线网络之间的转换。SDIO具有传输数据快&#xff0c;兼容SD、MMC接口等特点。 对于SDIO接口的w…

2312llvm,09clang工具下

ClangQuery 在LLVM3.5中引入ClangQuery工具,它可读入一个源文件,交互查询它所关联的ClangAST节点.是帮助查看并学习前端如何表达每行代码的很好工具. 然而,它的主要目标是,你不但可查看程序的AST,而且可测试AST匹配器. 编写重构工具时,你会对使用,包含匹配感兴趣的ClangAST片…

Erlang并发机制 –进程调度

2019独角兽企业重金招聘Python工程师标准>>> Erlang调度器主要完成对Erlang进程的调度&#xff0c;它是Erlang实现软件实时和进程之间公平使用CPU的关键。Erlang运行时&#xff0c;有4种任务需要被调度&#xff1a;进程&#xff0c;Port&#xff0c;Linked-in drive…

Linux 下wifi 驱动开发(二)—— WiFi模块浅析

一、什么是wifi 模块 百度百科上这样定义&#xff1a; Wi-Fi模块又名串口Wi-Fi模块&#xff0c;属于物联网传输层&#xff0c;功能是将串口或TTL电平转为符合Wi-Fi无线网络通信标准的嵌入式模块&#xff0c;内置无线网络协议IEEE802.11b.g.n协议栈以及TCP/IP协议栈。传统的硬件…

VSS2005 上传pdf 空白

加补丁 VS80-KB943847-X86-INTL.exe

Linux 下wifi 驱动开发(一)—— WiFi基础知识解析

一、WiFi相关基础概念 1、什么是wifi 我们看一下百度百科是如何定义的&#xff1a; Wi-Fi是一种可以将个人电脑、手持设备&#xff08;如pad、手机&#xff09;等终端以无线方式互相连接的技术&#xff0c;事实上它是一个高频无线电信号。[1] 无线保真是一个无线网络通信技术…

poj 3348 Cows 求凸包以及凸包的面积

题目来源&#xff1a; http://poj.org/problem?id3348 1:任意多边形p[0,n-1]的面积为 for(int i0 ; i<n-1 ; i){ sum (sk[i]^sk[(i1)%(n) ] )*0.5; } 2: 求凸包&#xff0c; 用graham 模板 代码如下&#xff1a; #include <iostream> #include <algorithm> #in…

Linux 网络设备驱动开发(一) —— linux内核网络分层结构

Linux内核对网络驱动程序使用统一的接口&#xff0c;并且对于网络设备采用面向对象的思想设计。 Linux内核采用分层结构处理网络数据包。分层结构与网络协议的结构匹配&#xff0c;既能简化数据包处理流程&#xff0c;又便于扩展和维护。 一、内核网络结构 在Linux内核中&#…