--集合实现游标查询

CREATE OR REPLACE PACKAGE emppkg IS

  TYPE t_record IS RECORD(

       rn INT,

       empno emp.empno%TYPE,

       ename emp.ename%TYPE,

       sal emp.sal%TYPE,

       deptno emp.deptno%TYPE);

  TYPE emplist IS TABLE OF t_record;--定义返回结果的记录集合类型

  --定义实现分页显示的过程

  PROCEDURE sp_page(

            p_page_size IN OUT INT,--每页输出显示的记录数

            p_cur_page_no IN OUT INT,--当前页码

            p_outcollection OUT emplist--接收查询返回结果

            );

END emppkg;

/

CREATE OR REPLACE PACKAGE BODY emppkg IS

  PROCEDURE sp_page(

            p_page_size IN OUT INT,--每页输出记录数

            p_cur_page_no IN OUT INT,--当前页码

            p_outcollection OUT emplist--接收查询返回值

            )

    IS

      v_start_rownum  NUMBER;

      v_end_rownum    NUMBER;

      v_total_records NUMBER;

      v_total_pages   NUMBER;

      indexvalue      BINARY_INTEGER :=1;

    BEGIN

      --查询记录总数

      SELECT COUNT(*) INTO v_total_records FROM emp;

      --验证输入页面显示记录数量,确保大于0

      IF p_page_size < 0 THEN

        p_page_size :=10;

      END IF;

      --计算页面总数

      IF MOD(v_total_records,v_total_pages) = 0 THEN

        v_total_pages := v_total_records / p_page_size;

       ELSE

         v_total_pages := (v_total_records / p_page_size)+1;

      END IF;

      --验证页号

      IF p_cur_page_no < 1 THEN

        p_cur_page_no := 1;

      END IF;

      IF p_cur_page_no > v_total_pages THEN

        p_cur_page_no := v_total_pages;

      END IF;

      --执行分页查询

      --计算当前页开始显示的记录在整个记录中从哪开始

      v_start_rownum := (p_cur_page_no-1)*p_page_size+1;

      --计算当前页的最后一条记录位于哪

      v_end_rownum := p_cur_page_no*p_page_size;

      --构造初始化记录嵌套表

      p_outcollection := emplist();

      --查询记录

      FOR v_emp IN (SELECT * FROM (SELECT ROWNUM rn,empno,ename,sal,deptno FROM

                                       (SELECT empno,ename,sal,deptno FROM emp

                                        WHERE sal IS NOT NULL

                                        ORDER BY sal) A--选出有sal的作为表A

                                   WHERE ROWNUM<=v_end_rownum) B

                    WHERE rn>=v_start_rownum)--在B中筛选去掉当前页之前的所有记录

      LOOP--循环把取出的数赋给接收变量

        p_outcollection.EXTEND;

        p_outcollection(indexvalue):=v_emp;

        indexvalue:=indexvalue+1;

      END LOOP;

   END sp_page;

END emppkg;  

/

 --游标实现分页查询

CREATE OR REPLACE PACKAGE curspkg IS

  TYPE query_cursor IS REF CURSOR;

  PROCEDURE sp_page(

            p_page_size IN OUT  INT,--每页输出记录数

            p_cur_page_no IN OUT INT,--当前页码

            p_sql_select VARCHAR2,--查询语句,包含排序部分

            p_total_pages OUT INT,--返回总页数

            p_total_records OUT INT,--返回总记录数

            p_out_cursor OUT query_cursor--接收返回当前页的数据集

            );

END curspkg;

/

CREATE OR REPLACE PACKAGE BODY curspkg IS

  PROCEDURE sp_page(

            p_page_size IN OUT  INT,--每页输出记录数

            p_cur_page_no IN OUT INT,--当前页码

            p_sql_select VARCHAR2,--查询语句,包含排序部分

            p_total_pages OUT INT,--返回总页数

            p_total_records OUT INT,--返回总记录数

            p_out_cursor OUT query_cursor--接收返回当前页的数据集

            )

   IS

     v_count_sql VARCHAR2(1000);

     v_start_rownum INT;

     v_end_rownum INT;

     v_sql VARCHAR2(4000);

   BEGIN

     --动态sql语句,调用输入的查询语句,计算输入的语句一共查询出多少数据

     v_count_sql := 'SELECT TO_CHAR(COUNT(*)) FROM ('||p_sql_select||')';

     EXECUTE IMMEDIATE v_count_sql INTO p_total_records;

     --验证输入页面显示记录数量,确保大于0

      IF p_page_size < 0 THEN

        p_page_size :=10;

      END IF;

      --计算页面总数

      IF MOD(p_total_records,P_total_pages) = 0 THEN

        P_total_pages := p_total_records / p_page_size;

       ELSE

         P_total_pages := (p_total_records / p_page_size)+1;

      END IF;

      --验证页号

      IF p_cur_page_no < 1 THEN

        p_cur_page_no := 1;

      END IF;

      IF p_cur_page_no > P_total_pages THEN

        p_cur_page_no := P_total_pages;

      END IF;

      --执行分页查询

      --计算当前页开始显示的记录在整个记录中从哪开始

      v_start_rownum := (p_cur_page_no-1)*p_page_size+1;

      --计算当前页的最后一条记录位于哪

      v_end_rownum := p_cur_page_no*p_page_size;

      v_sql:='SELECT * FROM (SELECT ROWNUM rn,A.* FROM

                            ('||p_sql_select||') A

                            WHERE ROWNUM<='||TO_CHAR(v_start_rownum)||') B

              WHERE rn>='||TO_CHAR(v_end_rownum);

      OPEN p_out_cursor FOR v_sql;

   END sp_page;

END curspkg;

/

PLSQL实现分页查询的更多相关文章

  1. JdbcTemplate+PageImpl实现多表分页查询

    一.基础实体 @MappedSuperclass public abstract class AbsIdEntity implements Serializable { private static ...

  2. 用Hibernate和Struts2+jsp实现分页查询、修改删除

    1.首先用get的方法传递一个页数过去 2.通过Struts2跳转到Action 3.通过request接受主页面index传过的页数,此时页数是1, 然后调用service层的方法获取DAO层分页查 ...

  3. MySQL、Oracle和SQL Server的分页查询语句

    假设当前是第PageNo页,每页有PageSize条记录,现在分别用Mysql.Oracle和SQL Server分页查询student表. 1.Mysql的分页查询: SELECT * FROM s ...

  4. 分页查询和分页缓存查询,List<Map<String, Object>>遍历和Map遍历

    分页查询 String sql = "返回所有符合条件记录的待分页SQL语句"; int start = (page - 1) * limit + 1; int end = pag ...

  5. mysql 分页查询

    mysql,; : mysql,; -last. //如果只给定一个参数,它表示返回最大的记录行数目: mysql; 个记录行 ,n. 动态传参的分页查询 SELECT * FROM table LI ...

  6. MongoDB 分页查询的方法及性能

    最近有点忙,本来有好多东西可以总结,Redis系列其实还应该有四.五.六...不过<Redis in Action>还没读完,等读完再来总结,不然太水,对不起读者. 自从上次Redis之后 ...

  7. .NET平台开源项目速览(7)关于NoSQL数据库LiteDB的分页查询解决过程

    在文章:这些.NET开源项目你知道吗?让.NET开源来得更加猛烈些吧!(第二辑) 与 .NET平台开源项目速览(3)小巧轻量级NoSQL文件数据库LiteDB中,介绍了LiteDB的基本使用情况以及部 ...

  8. SubSonic3.0插件分页查询速度测试

    使用SubSonic3.0一段时间了,一直都想找机会测试一下各种查询分页速度,对比一下插件的查询效率到底怎么样,所以昨天写好了测试程序,准备好1K.1W.10W.50W和100W记录的数据表,早上详细 ...

  9. Node.js、express、mongodb 实现分页查询、条件搜索

    前言 在上一篇Node.js.express.mongodb 入门(基于easyui datagrid增删改查) 的基础上实现了分页查询.带条件搜索. 实现效果 1.列表第一页. 2.列表第二页 3. ...

随机推荐

  1. MySQL · 引擎特性 · InnoDB 数据页解析

    前言 之前介绍的月报中,详细介绍了InnoDB Buffer Pool的实现细节,Buffer Pool主要就是用来存储数据页的,是数据页在内存中的动态存储方式,而本文介绍一下数据页在磁盘上的静态存储 ...

  2. [HNOI 2015]实验比较

    Description 小D 被邀请到实验室,做一个跟图片质量评价相关的主观实验.实验用到的图片集一共有 N 张图片,编号为 1 到 N.实验分若干轮进行,在每轮实验中,小 D会被要求观看某两张随机选 ...

  3. [HAOI 2016]食物链

    Description 如图所示为某生态系统的食物网示意图,据图回答第1小题. 1.数一数,在这个食物网中有几条食物链( ) 现在给你n个物种和m条能量流动关系,求其中的食物链条数. 物种的名称为从1 ...

  4. UVA - 11732:"strcmp()" Anyone?

    字典树问题 对于普通的字典树,可以加一个vector数组记录非空的孩子,加快速度 还可以用左孩子右兄弟来节省空间,因为普通的trie的话是 int next[MAXN][26] 而左孩子右兄弟可以把[ ...

  5. NOIP2015-D2T3运输计划

    题目背景 公元 2044 年,人类进入了宇宙纪元. 题目描述 L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 n-1 条航道连通了 L 国的所有星球. 小 P 掌管一家 ...

  6. NOIP2014-5-10模拟赛

    Problem 1 机器人(robot.cpp/c/pas) [题目描述] 早苗入手了最新的Gundam模型.最新款自然有着与以往不同的功能,那就是它能够自动行走,厉害吧. 早苗的新模型可以按照输入的 ...

  7. 12563 Jin Ge Jin Qu hao

    • Don’t sing a song more than once (including Jin Ge Jin Qu). • For each song of length t, either si ...

  8. [BZOJ]3527 力(ZJOI2014)

    第一次背出FFT模板,在此mark一道裸题. Description 给出n个数qi,给出Fj的定义如下: 令Ei=Fi/qi,求Ei. Input 第一行一个整数n. 接下来n行每行输入一个数,第i ...

  9. 简述RIP路由协议和OSPF路由协议的相同点和不同点。

    路由协议分为静态路由协议和动态路由协议.动态路由协议有很多种,如RIP.OSPF.EIGRP等. 1.RIP(路由信息协议)是路由器生产商之间使用的第一个开放标准.RIP有两个版本:RIPv1和RIP ...

  10. 基于Spark环境对比Python和Scala语言利弊

    在数据挖掘中,Python和Scala语言都是极受欢迎的,本文总结两种语言在Spark环境各自特点. 本文翻译自  https://www.dezyre.com/article/Scala-vs-Py ...