Java程序员的日常——存储过程知识普及

简介:

存储过程是保存可以接受或返回用户提供参数的SQL语句集合。在日常的使用中,经常会遇到复杂的业务逻辑和对数据库的操作,使用存储过程可以进行封装。可以在数据库中定义子程序,然后把子程序存储在数据库服务器,之后通过名称调用。

特点

1 提高性能

存储过程是预先编译过,进行优化后,存储在SQL的内存中,使用的时候不需要重新编译,提高工作效率。

2 减少网络流量

存储过程的代码直接存储在数据库中,用户通过名称进行调用,减小网络流量,加快执行速度。如:百万以上的数据查询,存储过程分页要比其他方式的分页快得多

3 提高安全性

存储过程可以减少SQL注入攻击,提高系统的安全性。执行的过程也受到用户的身份权限控制,因此没有数据操作权限的用户只能在权限控制下间接的存储数据。

4 事务处理机制

在同时进行主从表以及夺标的数据维护和有效性验证时,存储过程比较方便,可以有效地利用SQL的事务处理机制。

5 分离设计编码与使用

使用存储过程,可以实现存储过程设计和编码工作分开进行,只要存储过程名、参数、及返回信息告诉编码人员即可。

6 (缺点)不易移植和修改

使用存储过程封装业务逻辑将限制应用程序的可移植性;另外,如果更改参数或者返回的数据以及类型,需要修改相关的代码,比较繁琐。

语法结构

完整的过程结构如下:

create to replace procedure 过程名 as 
声明语句段;
begin
执行语句段;
exception
异常处理语句段;
end;

举例子:

-- 学生表student
create table student(
    sno number(6),
    sname varchar2(25),
    pno number(6) primary key
);

-- 存储过程
create or replace procedure stu_proc as 
p_name varchar2(25);
begin
    select sname into p_name from student where sno=1;
    dbms_output.put_line(p_name);
end;

-- 调用存储过程
call stu_proc();

关于参数的类型,存储过程大致提供下面几种

无参数的存储过程

定义

create or replace procedure stu_proc as 
pname varchar2(25);
begin
    select sname into pname from student where sno=1;
    dbms_output.put_line(pname);
end;

使用方法为:call stu_proc();

仅有输入参数的过程

create or replace procedure stu_proc1(pno in student.sno%type) as 
pname varchar2(25);
begin
    select sname into pname from student where sno=pno;
    dbms_output.put_line(pname);
end;

使用方法为:call stu_proc1('001')

仅有输出参数的过程

create or replace procedure stu_proc2(pname out student.sname%type) as 
begin
    select sname into pname from student where sno=1;
    dbms_output.put_line(pname);
end;

这种存储过程不能直接用call调用,需要在oracle函数调用。使用方法为:call stu_proc2(name)

有输入、输出的存储过程

create or replace procedure stu_proc3(pname out student.sname%type,pname out student.sname%type) as 
begin
    select sname into pname from student where sno=pno;
    dbms_output.put_line(pname);
end;

使用方法为:call stu_proc3(name,'001')

存储过程的异常处理

为了提高存储过程的健壮性,避免运行错误,建立存储过程时,应该包含异常处理的部分。异常包括预定义异常、非预定义异常和自定义异常。

  • 预定义异常:PL\SQL提供的系统异常
  • 费预定义异常:用于处理与预定义异常无关的Oracle错误
  • 自定义异常:处理Oracle错误之外的一些异常

使用方法:

create or replace procedure stu_proc6(pno in student.sno%type,pname out student.sname%type)
    is 
    begin
        select sname into pname from student where sno=pno;
    EXCEPTION
        when NO_DATA_FOUND then
        RAISE_APPLICATION_ERROR
            (-20011,'ERROR:不存在!');
end;

常用的异常处理:

命名的系统异常 产生原因
ACCESS_INTO_NULL 定义对象
CASE_NOT_FOUND CASE中未包含相应的WHEN,并且没有设置集合元素的初始化
COLLECTION_IS_NULL 集合元素未初始化
CURSER_ALREADY_OPEN 游标已经打开
DUP_VAL_ON_INDEX 唯一索引对应的列上有重复的值
INVALID_CURSOR 在不合法的游标上进行操作
INVALID_NUMBER 内嵌的SQL语句不能讲字符穿换成数字
NO_DATA_FOUND 使用select into 未返回行,或者应用索引表未初始化的
TOO_MANY_ROWS 执行select into,结果集超过一行
ZERO_DIVIDE 除数为0
SUBSCRIPT_BEYOND_COUNT 元素下表超过嵌套表或VARRAY的最大值
SUBSCRIPT_OUTSIDE_LIMIT 使用嵌套类或VARRAY时,将下表指定为负数
VALUE_ERROR 赋值时,变量长度不足以容纳实际数据
LOGIN_DENIED PL\SQL应用连接到oracle时,提供了 不正确的用户名密码
NOT_LOGGED_ON PL\SQL应用程序在没有连接oracle数据的情况下访问数据
PROGRAM_ERROR PL\SQL内部问题,可能需要重装数据字典
ROWTYPE_MISMATCH 主游标变量与PLSQL游标变量的返回类型不兼容
SELF_IS_NULL 使用对象类型时,在null对象上调用对象方法
STORAGE_ERROR 运行PL\SQL时,超出内存空间
SYS_INVALIDE_ID 无效的ROWID字符串
TIMEOUT_ON_RESOURCE Oracle在等待资源连接超时

存储过程与函数的区别

在定义上

定义的名称这个就不说了,一个是FUNCTION,一个是PROCEDURE;

  • 存储过程的参数列表有输入参数、输出参数、输入输出参数
  • 函数的参数只有输入参数,最后会加上一个return返回值。

在返回值上

  • 存储过程的返回值,可以有多个
  • 函数的返回值只有一个

调用方式上

  • 存储过程的调用方式有:exec、execute、语句块调用
  • 函数的调用方式有:可以在函数块中、也可以直接在sql中使用,比如:
create or replace function add_three_numbers
(
    a NUMBER:=0,b NUMBER:=0,c NUMBER:=0
)
return number is
begin
return a+b+c;
end;

select add_three_numbers(1,2,3) from dual;

事务处理

  • 1 事务用于确保数据的一致性,要么全部确认,要不全部取消。
  • 2 档执行事务操作的时候,Oracle会作用在表上加锁,防止其他的用户改变表。同时也会在被作用的行上加行锁,以防止其他事务在相应行上执行DML操作
  • 3 执行事务提交或者事务回滚时,Oracle会确认事务变化或回滚事务、结束事务、删除保存点、释放锁

参考

Oracle存储过程

本文转自博客园xingoo的博客,原文链接:Java程序员的日常——存储过程知识普及,如需转载请自行联系原博主。
相关文章
|
10月前
|
人工智能 Kubernetes Java
回归开源,两位 Java 和 Go 程序员分享的开源贡献指引
Higress是一个基于Istio和Envoy的云原生API网关,支持AI功能扩展。它通过Go/Rust/JS编写的Wasm插件提供可扩展架构,并包含Node和Java的console模块。Higress起源于阿里巴巴,解决了Tengine配置重载及gRPC/Dubbo负载均衡问题,现已成为阿里云API网关的基础。本文介绍Higress的基本架构、功能(如AI网关、API管理、Ingress流量网关等)、部署方式以及如何参与开源贡献。此外,还提供了有效的开源贡献指南和社区交流信息。
1348 33
|
10月前
|
Java 程序员 应用服务中间件
【高薪程序员必看】万字长文拆解Java并发编程!(2 2-2)
📌 核心痛点暴击:1️⃣ 面了8家都被问synchronized锁升级?一张图看懂偏向锁→重量级锁全过程!2️⃣ 线程池参数不会配?高并发场景下这些参数调优救了项目组命!3️⃣ volatile双重检测单例模式到底安不安全?99%人踩过的内存可见性大坑!💡 独家亮点抢先看:✅ 图解JVM内存模型(JMM)三大特性,看完再也不怕指令重排序✅ 手撕ReentrantLock源码,AQS队列同步器实现原理大揭秘✅ 全网最细线程状态转换图(附6种状态转换触发条件表)
190 0
|
10月前
|
存储 缓存 Java
【高薪程序员必看】万字长文拆解Java并发编程!(5):深入理解JMM:Java内存模型的三大特性与volatile底层原理
JMM,Java Memory Model,Java内存模型,定义了主内存,工作内存,确保Java在不同平台上的正确运行主内存Main Memory:所有线程共享的内存区域,所有的变量都存储在主存中工作内存Working Memory:每个线程拥有自己的工作内存,用于保存变量的副本.线程执行过程中先将主内存中的变量读到工作内存中,对变量进行操作之后再将变量写入主内存,jvm概念说明主内存所有线程共享的内存区域,存储原始变量(堆内存中的对象实例和静态变量)工作内存。
323 0
|
10月前
|
设计模式 缓存 安全
【高薪程序员必看】万字长文拆解Java并发编程!(8):设计模式-享元模式设计指南
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。
221 0
|
10月前
|
存储 安全 Java
【高薪程序员必看】万字长文拆解Java并发编程!(7):不可变类设计指南
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中Java不可变类设计指南,废话不多说让我们直接开始。
202 0
|
10月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
396 0
|
存储 Java 关系型数据库
java调用mysql存储过程
在 Java 中调用 MySQL 存储过程主要借助 JDBC(Java Database Connectivity)。其核心原理是通过 JDBC 与 MySQL 建立连接,调用存储过程并处理结果。具体步骤包括:加载 JDBC 驱动、建立数据库连接、创建 CallableStatement 对象、设置存储过程参数并执行调用。此过程实现了 Java 程序与 MySQL 数据库的高效交互。
|
11月前
|
人工智能 Java 程序员
Java程序员在AI时代必会的技术:Spring AI
在AI时代,Java程序员需掌握Spring AI技术以提升竞争力。Spring AI是Spring框架在AI领域的延伸,支持自然语言处理、机器学习集成与自动化决策等场景。它简化开发流程,无缝集成Spring生态,并提供对多种AI服务(如OpenAI、阿里云通义千问)的支持。本文介绍Spring AI核心概念、应用场景及开发步骤,含代码示例,助你快速入门并构建智能化应用,把握AI时代的机遇。
2337 61
|
10月前
|
存储 监控 算法
Java程序员必学:JVM架构完全解读
Java 虚拟机(JVM)是 Java 编程的核心,深入理解其架构对开发者意义重大。本文详细解读 JVM 架构,涵盖类加载器子系统、运行时数据区等核心组件,剖析类加载机制,包括加载阶段、双亲委派模型等内容。阐述内存管理原理,介绍垃圾回收算法与常见回收器,并结合案例讲解调优策略。还分享 JVM 性能瓶颈识别与调优方法,分析 Java 语言特性对性能的影响,给出数据结构选择、I/O 操作及并发同步处理的优化技巧,同时探讨 JVM 安全模型与错误处理机制,助力开发者提升编程能力与程序性能。
Java程序员必学:JVM架构完全解读
|
人工智能 Java 程序员
【AI程序员】通义灵码 AI 程序员全面上线JAVA使用体验
通过 AI 程序编写一个JAVA后台项目登陆页面
907 42