所有的MIS系统都存在一个同样的需求,就是对于特定的数据,在一次批量操作过程中,如果数据已经存在,则对存在的数据按照现有情况进行更新,如果不存在,则需要加入数据库。这时,我们就可以考虑采用 Oracle 的 MERGE 函数,其具体用法如下:
sql 代码
- MERGE INTO [your table-name] [rename your table here]
- USING
- (
- [write your query here]
- )[rename your query-sql and using just like a table]
- ON
- ([conditional expression here] AND [...]...)
- WHEN
- MATHED
- THEN
- [here you can execute some update sql or something else ]
- WHEN
- NOT MATHED
- THEN
- [execute something else here ! ]
下面我再进行详细的说明:
上述代码格式中的加粗字体表示为 Oracle 关键字,[]以及其中的文字均是说明,在实际使用中不应有 [ words ] 出现。要注意()[圆括号]也是程序的组成部分。
为了能够使问题与实际问题更加贴切,不妨假设我们现在要给计算机系某个班的学生批量录入学生成绩。但是,录入时,如果学生的成绩已经存在时,老师只想对成绩进行修改,而如果成绩不存在则直接添加到库中。我们就老师的这些需求来构造一个执行语句。
sql 代码
- DEFINE TABLE :
- SCORE : using for save the students' score informations
- STUDENTS : the base information of students
- DEFINE COLUMNS :
- STUNO : the students' ID in the University
- STUNAME : students' name
- COURSENAME : course name
- COURSESCORE : the study-results of the reference course
- CLASSNAME : where the students study in
- STUGRADE : the students grade
- TERMNAME : the term which the reference course studied
- NOW BEAGIN TO WRITE DOWN THE STATEMENT HERE BLOW THIS LINE !
-
- MERGE INTO SCORE S
- USING
- (
- SELECT A.*,B.*,? MYSCORE FROM SCORE A,STUDENT B
- WHERE
- A.CLASSNO=? AND A.GRADE=?
- AND A.TERMNAME=? AND A.COURSENAME=?
- A.STUNO=B.STUNO(+)
- )X
- ON
- (S.STUNO=X.STUNO)
- WHEN
- MATHED
- THEN
- UPDATE SET COURSESCORE=X.MYSCORE
- WHEN
- NOT MATHED
- THEN
- INSERT
- (
- STUNO,STUNAME,COURSENAME,COURSESCORE,
- CLASSNAME,STUGRADE,TERMNAME
- )
- VALUES
- (
- X.STUNO,X.STUNAME,X.COURSENAME,X.MYSCORE,
- X.CLASSNAME,X.STUGRADE,X.TERMNAME
- );
注意到 MERGE 语句在最后的“;”(分号),这仅仅代表 MERGE 为一条完整的 SQL 语句。同时,要说明一下 USING 语句下方的 SQL 语句。这个语句仅仅是为了给后面语句的执行做准备性的工作,因此,如果你需要的数据仅仅是通过参数传入的那些值的话你就不需要再利用传入进来的参数在重新从库中查询。在 Oracle 的系统表中,有张 Dual 表,这样,你便可以使用 “select [your arguments] from dual ”的方式来构建这里的 SQL 语句,其中 [your arguments] 是你得到的一系列的参数,由于Dual表是系统表,因此可以大幅提升SQL的执行效率。
这时,如果你需要在你的 Java 程序中使用上述方法执行相应操作,则仅需要将其放入一个 for 循环中即可。由于是批量更新数据,因此,如果你不想对中间出现异常的数据进行提交,导致数据的不完整,则可以考虑使用 Java 的事务回滚机制。具体示例代码如下:
java 代码
- public yourMethod(statement,...){
- try{
- Connection conn=...;
- PreparedStatement ps=...;
- Resultset rs=...;
- conn.setAutoCommit(false);
- for(int i=0;i<...;i++){
-
- ......
- ps.addBatch();
- }
- ps.executeBatch();
- conn.commit();
- }catch(Exception e){
- try{
- conn.rollback();
- }catch(Exception el){}
- }
- }
这时,你会发现,在代码中直接使用 Merge 时,代码会变的非常复杂,首先是 SQL 的拼接变得非常复杂,接下来便是程序写完后的查错。因此,自然而然就会想到使用存储过程。接下来,我们来看看如何使用存储过程实现 Merge 调用过程。
Oracle 存储过程定义格式如下:
sql 代码
-
- CREATE OR REPLACE PROCEDURE PRO_YOUR_PROCEDURE (
- ELEMENT_01 IN ELEMENT_TYPE,
- ....... .... ..... ....
- ELEMENT_0S OUT ELEMENT_TYPE,
- .... ... ... ....
- )
- AS
- ARGUMENT_01 ARGUMENT_TYPE(ARGUMENT_RANGE);
- ...................
- BEGIN
- MERGE INTO YOUR_TABLE_NAEM [RENAEM_YOUR_TABLE_HERE]
-
- END;
- EXCEPTION
- WHEN
- OTHERS
- THEN
- RAISE_APPLICATION_ERROR(-20003,[YOUR EXCEPITON MESSAGE HERE !]);
-
- END;
- COMMIT;
- END PRO_YOUR_PROCEDURE;
其中,[RAISE_APPLICATION_ERROR(-20003,[YOUR EXCEPITON MESSAGE HERE !]);]中的“-20003”是 Oracle 提供的用于用户进行错误自定义的扩充代码。其值可以随便定义,但是也有范围: -20000 到-20999的负整数。
接下来就是如何来在 Java 程序中调用你的存储过程。Oracle为了方便开发人员调用其存储过程,开发了一个 [ OracleCallableStatement ] 位于 oracle.jdbc 包内。
核心代码如下:
java 代码
- OracleCallableStatement cal = null;
- cal=(OracleCallableStatement)conn.getMetaData().getConnection().prepareCall("call PRO_......");
- ........
- .............
- .......
- for(………………){
- …………
- cal.setDouble(i,ARGUMENTS);
- …………
- cal.executeUpdate();
- }
从中可以看出,OracleCallableStatement对象事实上承担了PreparedStatement对象的使命,当然如果需要的话,能够支持事务回滚是个不错的主意,关于这个问题,可以参考上面直接使用 Merge 函数的 Java 代码。
分享到:
相关推荐
本教程介绍了 Rust 的基础语法、所有权模型、函数与模块设计,以及高级特性(如错误处理、闭包和并发编程)。通过提供实用的练习和答案,帮助您快速掌握 Rust 的核心概念,为系统级编程打下扎实基础。
基于springboot+Web的毕业设计选题系统源码数据库文档.zip
# 基于Spring和MyBatis的疫情防控管理系统 ## 项目简介 本项目是一个基于Spring和MyBatis框架的疫情防控管理系统,旨在实现数据在管理员、医务人员、患者和数据上报者之间的流转。系统涵盖了用户登录、注册、信息管理、预约管理、检测结果管理等功能,支持多角色的权限管理和数据操作。 ## 项目的主要特性和功能 1. 用户管理 用户登录、注册和登出功能。 用户个人信息管理,包括核酸检测记录和预约记录。 2. 管理员管理 管理员列表管理,支持增删改查操作。 医院信息管理,支持增删改查操作。 3. 医务人员管理 核酸预约和疫苗接种预约管理。 核酸检测结果和疫苗接种结果管理。 患者信息管理。 4. 数据上报者管理 风险区域数据上报。 患者信息上报。 5. 系统安全 通过拦截器实现用户登录状态检查,未登录用户将被重定向到登录页面。
对数据集进行二分类,有数据集和源码以及模型,二分类是识别猫和不是猫的情况,可做毕业设计。
最新完美版积分商城系统,网购商城系统源码,是更新的奇偶商城系统源码, 它拥有独立代理后台,而且内附搭建教程。 企业猫搭建了下,感觉这个源码很新颖的,购买商品后可以选择直接发货还是拆红包升级购买的商品升级成别的商品。
编译好的Linux版网络调试助手,方便直接使用,免得每次都需要重新编译
office使用软件
# 基于Python和Flask的博客管理系统 ## 项目简介 本项目是一个基于Python和Flask框架的博客管理系统,旨在为用户提供一个简单易用的博客平台。用户可以注册、登录、发布博客、管理博客内容以及评论等功能。 ## 项目的主要特性和功能 1. 用户管理 用户注册和登录功能。 用户个人信息管理。 2. 博客管理 发布、编辑和删除博客文章。 博客分类和标签管理。 3. 评论系统 用户可以对博客文章进行评论。 评论的查看和管理。 4. 权限管理 管理员可以管理用户和博客内容。 普通用户只能管理自己的博客和评论。 ## 安装使用步骤 1. 环境准备 确保已安装Python 3.x。 安装Flask框架pip install Flask。 2. 数据库配置 在config.py文件中配置数据库连接信息。
基于springboot+vue的实践性教学系统源码数据库文档.zip
基于springboot二手物品交易系统源码数据库文档.zip
基于springboot餐品美食论坛源码数据库文档.zip
人工智能开发项目深度学习项目源码带指导视频词云提取方式是百度网盘分享地址
基于springboot企业员工薪酬管理系统源码数据库文档.zip
基于springboot+JavaWeb图书管理系统源码数据库文档.zip
数据库设计管理课程设计系统设计报告(powerdesign+sql+DreamweaverCS)销售管理系统设计与开发提取方式是百度网盘分享地址
1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
基于springboot的小说阅读平台源码数据库文档.zip
# 基于PaddleDetection框架的人流量统计系统 ## 项目简介 本项目是一个基于PaddleDetection框架的人流量统计系统,专注于静态和动态场景下的人员计数和行人检测。项目涵盖了从数据准备、模型选择、训练、评估、优化到预测和部署的完整流程,旨在提供高效、准确的人流量统计解决方案。 ## 主要特性和功能 多模型支持支持多种模型选择,如DeepSORT、JDE和FairMOT,适用于多目标追踪场景。 模型优化提供多种优化策略,包括数据增强、可变形卷积、syncbn+ema、attention和GIoU Loss,以提升模型精度。 性能加速支持TensorRT推理加速,显著提升模型性能。 数据增强提供多种数据增强方式,如cutmix、syncbn和ema,进一步优化模型性能。 模型导出支持模型导出,便于模型部署和上线。 ## 安装使用步骤 1. 安装PaddleDetection框架 bash
基于springboot的银行信用卡额度管理系统源码数据库文档.zip
# 基于Arduino的蒸发冷却系统 ## 项目简介 本项目旨在创建一个蒸发冷却系统(即沼泽冷却器),这是一种在干燥炎热气候下提供能源效率替代空调的系统。该系统使用Arduino 2560和各种传感器来监控和控制冷却过程。 ## 项目的主要特性和功能 水位监控通过水位传感器监控水箱中的水位,并在水位过低时发出警报。 温度和湿度显示在LCD屏幕上显示空气温度和湿度。 风扇控制根据温度范围控制风扇电机。 系统开关通过用户按钮控制系统的开关。 日志记录记录电机激活和停用的时间和日期。 ## 安装使用步骤 2. 硬件连接按照项目文档中的电路图连接所有硬件组件,包括水位传感器、LCD显示屏、实时时钟模块、DHT11传感器和风扇电机。 3. 上传代码将下载的代码上传到Arduino 2560开发板。 4. 启动系统通过用户按钮启动系统,观察LCD屏幕上的温度和湿度显示,并监控水位传感器的状态。