精华内容
下载资源
问答
  • prepare

    2018-03-17 23:54:10
    1. 在canvas上面画矩形之后,弹出一个popup,上面从左到右依次是题号,题型,分值,标准答案等,2. 备课软件,目录结构,Hudong\beikeHudong\beike\uploadsHudong\beike\dest\pic3/18/2018, 从备课软件添加考试包到...

    1. 在canvas上面画矩形之后,弹出一个popup,上面从左到右依次是题号,题型,分值,标准答案等,

    2. 备课软件,目录结构,

    Hudong\beike

    Hudong\beike\uploads

    Hudong\beike\dest\pic

    3/18/2018, 从备课软件添加考试包到java后台成功

    3/19/2018, 解决了hdback cors问题,思考如何把显示学生列表做成一个reusable usercontrol或者custom control,没有实现

    3/20/2018, 准备解决examine的学生列表问题,已解决

    3/21/2018, 从阿里云下载文件

    3/22/2018, 解析考试包,显示学生试卷或者全班试卷, solved

    3/23/2018, 绘制笔迹,敲定谁写服务, java bean读取姓名模板文件,注入到controller当中,Iterator接口,读取xml文件。绘制没有完成

    3/24/2018, checkin,

    3/25/2018, checkin很复杂,因为下笔抬笔没有页面信息,需要设计一个stranger,抬笔时替换掉自己成为真正的Student,写字时记录x,y值。线程安全的字典,读不加锁,写部分加锁

    3/26/2018, 绘制出来了,总共花了4天,主要没有仔细考虑checkin,因为需要仔细考虑,所以费时间。

    3/27/2018, elasticsearch和项目不能配合,在本机安装elasticsearch,看文档

    3/28/2018, 产品环境下elasticsearch搞定了,版本是2.3.5,数字时钟搞定,待识别队列确定下来。

    3/29/2018, 备课软件的鼠标拖拽的矩形大小设置,能记住5个矩形大小,能定义矩形形状。

    3/30/2018, 调试识别,把笔迹整理成待识别的对象并送入队列,已解决

    3/31/2018, protobuf搞定,还是用的protobuf-net,但是可以定义.proto文件了。

    4/1/2018, 上传服务搞定

    4/2/2018, 学生的笔迹序列化成一个文件,题目区域以及批改的信息序列化成另外一个文件。

    4/3/2018, 阅卷获取到学生-考试列表,定义了学生-考试DTO,OneToMany是不能获取到Many端的数据的,比如从Examine对象获取不到StudentExamine列表的,因为jpa的lazy特征。

    4/4/2018, 有考试视图切换到批阅视图,显示学生名单, 显示试卷

    4/5/2018, 生成学生笔迹文件,待阅卷文件,xml文件没有来得及生成

    4/6/2018, 生成xml文件, 在服务中更新数据库表

    4/7/2018, 研究mongodb,新建了一个jhipster的mongodb工程,跑通

    4/8/2018, 了解mysql5.7.8之后自带json支持,看了一天文档,决定不用mongo了,商用一年18万,一天onsite费用1万8.还是用mysql,关键要集成到spring data jpa当中。

    4/9/2018, 尝试集成mysql的json类型,https://vladmihalcea.com/how-to-map-json-objects-using-generic-hibernate-types/

    4/10/2018, 保存json字符串成功,花了4天时间,我草,休息一下

    4/11/2018, 获取json列成功,可以显示序列化的stroke了

    4/12/2018, 着手显示question,但是安装2017费了不少时间,要创建新的Gadget

    4/13/2018, 每道题显示分值

    4/14/2018, docker

    4/15/2018, docker

    4/16/2018, docker

    4/17/2018, docker推送到阿里云

    4/18/2018, docker-compose启动应用成功,解决了一个template.ban的问题

    4/19/2018, docker-compose upgrade image and container

    4/20/2018, angular 5

    4/21/2018, angular 5

    4/22/2018, angular 5

    4/23/2018, angular 5

    4/24/2018, angular 5

    4/25/2018, angular5

    4/26/2018, angular5

    展开全文
  • Prepare fixes

    2020-12-01 15:39:57
    <div><p>fixed minor issues I encountered when running the prepare command <ol><li><code>cellxgene prepare <file></code> does nothing because no file is saved. I added an error for this case</li>...
  • Prepare 1.10

    2020-12-07 12:02:50
    <div><p>prepare 1.10 release</p><p>该提问来源于开源项目:sasstools/sass-lint</p></div>
  • Prepare release

    2020-12-30 12:23:23
    <div><p>Prepare release of 7.2.0.0 related to #198 #199 </p><p>该提问来源于开源项目:vvanholl/elasticsearch-prometheus-exporter</p></div>
  • <div><p>One time prepare - added $prepared attribute to indicate if prepare() was done - if $prepared prepare() exits on begin <p>Prepare in mass actions - processSessionData() and prepare() is always...
  • <div><p>If a prepare hook is defined in a platform block, it is not executed when <code>cordova prepare</code> is called without specifying a platform. <p>My expectation is that the platform prepare ...
  • MySQL prepare原理详解

    2020-09-09 13:44:05
    主要介绍了MySQL prepare的相关内容,包括prepare的产生,在服务器端的执行过程,以及jdbc对prepare的处理以及相关测试,需要的朋友可以了解下。希望对大家有所帮助。
  • MySQL官方将prepare、execute、deallocate统称为PREPARE STATEMENT,习惯称其为【预处理语句】,下面是对其详细的介绍。示例代码PREPARE stmt_name FROMpreparable_stmtEXECUTEstmt_name[USING @var_name [, @var_...

    MySQL官方将prepare、execute、deallocate统称为PREPARE STATEMENT,习惯称其为【预处理语句】,下面是对其详细的介绍。

    示例代码

    PREPARE stmt_name FROMpreparable_stmtEXECUTEstmt_name[USING @var_name [, @var_name] ...] -{DEALLOCATE | DROP} PREPARE stmt_name

    举个栗子:

    mysql> PREPARE pr1 FROM 'SELECT ?+?';

    Query OK,0 rows affected (0.01sec)

    Statement prepared

    mysql> SET @a=1, @b=10;

    Query OK,0 rows affected (0.00sec)

    mysql> EXECUTE pr1 USING @a, @b;+------+

    | ?+? |

    +------+

    | 11 |

    +------+

    1 row in set (0.00sec)

    mysql> EXECUTE pr1 USING 1, 2; --只能使用用户变量传递。

    ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version fortheright syntax to use near '1, 2' at line 1mysql> DEALLOCATE PREPAREpr1;

    Query OK,0 rows affected (0.00 sec)

    使用PAREPARE STATEMENT可以减少每次执行SQL的语法分析,比如用于执行带有WHERE条件的SELECT和DELETE,或者UPDATE,或者INSERT,只需要每次修改变量值即可。

    同样可以防止SQL注入,参数值可以包含转义符和定界符。

    适用在应用程序中,或者SQL脚本中均可。

    更多用法:

    同样PREPARE ... FROM可以直接接用户变量:

    mysql> CREATE TABLE a (a int);

    Query OK,0 rows affected (0.26sec)

    mysql> INSERT INTO a SELECT 1;

    Query OK,1 row affected (0.04sec)

    Records:1 Duplicates: 0 Warnings: 0mysql> INSERT INTO a SELECT 2;

    Query OK,1 row affected (0.04sec)

    Records:1 Duplicates: 0 Warnings: 0mysql> INSERT INTO a SELECT 3;

    Query OK,1 row affected (0.04sec)

    Records:1 Duplicates: 0 Warnings: 0mysql> SET @select_test = CONCAT('SELECT * FROM', @table_name);

    Query OK,0 rows affected (0.00sec)

    mysql> SET @table_name = 'a';

    Query OK,0 rows affected (0.00sec)

    mysql> PREPARE pr2 FROM @select_test;

    Query OK,0 rows affected (0.00sec)

    Statement prepared

    mysql> EXECUTEpr2 ;+------+

    | a |

    +------+

    | 1 |

    | 2 |

    | 3 |

    +------+

    3 rows in set (0.00sec)

    mysql> DROP PREPARE pr2; --此处DROP可以替代DEALLOCATE

    Query OK, 0 rows affected (0.00 sec)

    每一次执行完EXECUTE时,养成好习惯,须执行DEALLOCATE PREPARE … 语句,这样可以释放执行中使用的所有数据库资源(如游标)。

    不仅如此,如果一个session的预处理语句过多,可能会达到max_prepared_stmt_count的上限值。

    预处理语句只能在创建者的会话中可以使用,其他会话是无法使用的。

    而且在任意方式(正常或非正常)退出会话时,之前定义好的预处理语句将不复存在。

    如果在存储过程中使用,如果不在过程中DEALLOCATE掉,在存储过程结束之后,该预处理语句仍然会有效。

    展开全文
  • MySQL PREPARE语句简介:在本教程中,您将学习如何使用MySQL预处理语句 使您的查询执行更快,更安全。MySQL PREPARE 语句简介在MySQL 4.1版之前,查询以文本格式发送到MySQL服务器。反过来,MySQL使用文本协议将数据...

    MySQL PREPARE语句

    简介:在本教程中,您将学习如何使用MySQL预处理语句 使您的查询执行更快,更安全。

    MySQL PREPARE 语句简介

    在MySQL 4.1版之前,查询以文本格式发送到MySQL服务器。反过来,MySQL使用文本协议将数据返回给客户端。MySQL必须  完全 解析查询并将结果集转换为字符串,然后再将其返回给客户端。

    文本协议具有严重的性能影响。为了解决这个问题,MySQL从4.1版开始添加了一个名为prepared的新功能。

    准备好的语句利用客户端/服务器二进制协议。它将包含占位符(?)的查询传递给MySQL服务器,如下例所示:

    SELECT *

    FROM products

    WHERE productCode = ?;

    当MySQL使用不同的productcode值执行此查询时,它不必完全解析查询。因此,这有助于MySQL更快地执行查询,尤其是当MySQL多次执行查询时。因为预准备语句使用占位符(?),这有助于避免SQL注入的许多变体,从而使您的应用程序更安全。

    MySQL PREPARE 语句用法

    为了使用MySQL预处理语句,您需要使用其他三个MySQL语句,如下所示:

    PREPARE - 准备要执行的语句。

    EXECUTE - 执行由PREPARE语句准备的预准备语句。

    DEALLOCATE PREPARE - 发布准备好的声明。

    下图说明了如何使用预准备语句:

    MySQL-Prepared-Statement.png

    MySQL编写了语句实例

    让我们看一下使用MySQL预处理语句的示例。

    PREPARE stmt1 FROM 'SELECT productCode, productName

    FROM products

    WHERE productCode = ?';

    SET @pc = 'S10_1678';

    EXECUTE stmt1 USING @pc;

    DEALLOCATE PREPARE stmt1;

    首先,我们使用PREPARE语句准备执行语句。我们使用  SELECT语句根据指定的产品代码查询products表中的产品数据  。我们使用问号(?)作为产品代码的占位符。

    接下来,我们声明了一个产品代码变量  @pc并将其值设置为S10_1678。

    然后,我们使用EXECUTE语句用产品代码变量执行预准备语句@pc。

    最后,我们用它  DEALLOCATE PREPARE来发布准备好的声明。

    在本教程中,我们向您展示了如何使用MySQL预处理语句执行带占位符的查询,以提高查询速度并使查询更安全。

    展开全文
  • Java数据库JDBC——prepareStatement的用法和解释

    万次阅读 多人点赞 2019-01-10 17:06:11
    一、prepareStatement 的用法和解释 1.PreparedStatement是预编译的,对于批量处理可以大大提高效率. 也叫JDBC存储过程 2.使用 Statement 对象。在对数据库只执行一次性存取的时侯,用 Statement 对象进行处理。...

    转自:http://blog.csdn.net/QH_JAVA/article/details/48245945

    一、prepareStatement 的用法和解释

    1.PreparedStatement是预编译的,对于批量处理可以大大提高效率. 也叫JDBC存储过程

    2.使用 Statement 对象。在对数据库只执行一次性存取的时侯,用 Statement 对象进行处理。PreparedStatement 对象的开销比Statement大,对于一次性操作并不会带来额外的好处。
    3.statement每次执行sql语句,相关数据库都要执行sql语句的编译,preparedstatement是预编译得, preparedstatement支持批处理

    4、

    Code Fragment 1:

    String updateString = "UPDATE COFFEES SET SALES = 75 " + "WHERE COF_NAME LIKE ′Colombian′";
    stmt.executeUpdate(updateString);
    


    Code Fragment 2:

    PreparedStatement updateSales = con.prepareStatement("UPDATE COFFEES SET SALES = ? WHERE COF_NAME LIKE ? ");
    updateSales.setInt(1, 75);
    updateSales.setString(2, "Colombian");
    updateSales.executeUpdate();


    片断2和片断1的区别在于,后者使用了PreparedStatement对象,而前者是普通的Statement对象。PreparedStatement对象不仅包含了SQL语句,而且大多数情况下这个语句已经被预编译过,因而当其执行时,只需DBMS运行SQL语句,而不必先编译。当你需要执行Statement对象多次的时候,PreparedStatement对象将会大大降低运行时间,当然也加快了访问数据库的速度。
    这种转换也给你带来很大的便利,不必重复SQL语句的句法,而只需更改其中变量的值,便可重新执行SQL语句。选择PreparedStatement对象与否在于相同句法SQL语句是否执行了多次,而且两次之间的差别仅仅是变量的不同。如果仅仅执行了一次的话,它应该和普通的对象毫无差异,体现不出它预编译的优越性。

     

    5.执行许多SQL语句的JDBC程序产生大量的Statement和PreparedStatement对象。通常认为PreparedStatement对象比Statement对象更有效,特别是如果带有不同参数的同一SQL语句被多次执行的时候。PreparedStatement对象允许数据库预编译SQL语句,这样在随后的运行中可以节省时间并增加代码的可读性。


    然而,在Oracle环境中,开发人员实际上有更大的灵活性。当使用Statement或PreparedStatement对象时,Oracle数据库会缓存SQL语句以便以后使用。在一些情况下,由于驱动器自身需要额外的处理和在Java应用程序和Oracle服务器间增加的网络活动,执行PreparedStatement对象实际上会花更长的时间。

    然而,除了缓冲的问题之外,至少还有一个更好的原因使我们在企业应用程序中更喜欢使用PreparedStatement对象,那就是安全性。传递给PreparedStatement对象的参数可以被强制进行类型转换,使开发人员可以确保在插入或查询数据时与底层的数据库格式匹配。

    当处理公共Web站点上的用户传来的数据的时候,安全性的问题就变得极为重要。传递给PreparedStatement的字符串参数会自动被驱动器忽略。最简单的情况下,这就意味着当你的程序试着将字符串“D'Angelo”插入到VARCHAR2中时,该语句将不会识别第一个“,”,从而导致悲惨的失败。几乎很少有必要创建你自己的字符串忽略代码。

    在Web环境中,有恶意的用户会利用那些设计不完善的、不能正确处理字符串的应用程序。特别是在公共Web站点上,在没有首先通过PreparedStatement对象处理的情况下,所有的用户输入都不应该传递给SQL语句。此外,在用户有机会修改SQL语句的地方,如HTML的隐藏区域或一个查询字符串上,SQL语句都不应该被显示出来。
    在执行SQL命令时,我们有二种选择:可以使用PreparedStatement对象,也可以使用Statement对象。无论多少次地使用同一个SQL命令,PreparedStatement都只对它解析和编译一次。当使用Statement对象时,每次执行一个SQL命令时,都会对它进行解析和编译。 

    第一:

    prepareStatement会先初始化SQL,先把这个SQL提交到数据库中进行预处理,多次使用可提高效率。  
    Statement不会初始化,没有预处理,没次都是从0开始执行SQL

    第二:

    prepareStatement可以替换变量  
    在SQL语句中可以包含?,可以用ps=conn.prepareStatement("select * from Cust where ID=?");  
    int sid=1001;  
    ps.setInt(1, sid);  
    rs = ps.executeQuery();  
    可以把?替换成变量。  
    而Statement只能用  
    int sid=1001;  
    Statement stmt = conn.createStatement();  
    ResultSet rs = stmt.executeQuery("select * from Cust where ID="+sid);  
    来实现。

     

    二、深入理解statement 和prepareStatement

     

    1、使用Statement而不是PreparedStatement对象
    JDBC驱动的最佳化是基于使用的是什么功能. 选择PreparedStatement还是Statement取决于你要怎么使用它们. 对于只执行一次的SQL语句选择Statement是最好的. 相反, 如果SQL语句被多次执行选用PreparedStatement是最好的.

    PreparedStatement的第一次执行消耗是很高的. 它的性能体现在后面的重复执行. 例如, 假设我使用Employee ID, 使用prepared的方式来执行一个针对Employee表的查询. JDBC驱动会发送一个网络请求到数据解析和优化这个查询. 而执行时会产生另一个网络请求.在JDBC驱动中,减少网络通讯是最终的目的. 如果我的程序在运行期间只需要一次请求, 那么就使用Statement. 对于Statement, 同一个查询只会产生一次网络到数据库的通讯.

    对于使用PreparedStatement池的情况下, 本指导原则有点复杂. 当使用PreparedStatement池时, 如果一个查询很特殊, 并且不太会再次执行到, 那么可以使用Statement. 如果一个查询很少会被执行,但连接池中的Statement池可能被再次执行, 那么请使用PreparedStatement. 在不是Statement池的同样情况下, 请使用Statement.

    2、使用PreparedStatement的Batch功能
    Update大量的数据时, 先Prepare一个INSERT语句再多次的执行, 会导致很多次的网络连接. 要减少JDBC的调用次数改善性能, 你可以使用PreparedStatement的AddBatch()方法一次性发送多个查询给数据库. 例如, 让我们来比较一下下面的例子.

    例 1: 多次执行PreparedStatement,多次数据库请求(网络请求)

    PreparedStatement ps = conn.prepareStatement(  
       "INSERT into employees values (?, ?, ?)");  
      
    for (n = 0; n < 100; n++) {  
      
      ps.setString(name[n]);  
      ps.setLong(id[n]);  
      ps.setInt(salary[n]);  
      ps.executeUpdate();  
    }  

    例 2: 使用Batch,以此请求执行多条

    PreparedStatement ps = conn.prepareStatement(  
       "INSERT into employees values (?, ?, ?)");  
      
    for (n = 0; n < 100; n++) {  
      
      ps.setString(name[n]);  
      ps.setLong(id[n]);  
      ps.setInt(salary[n]);  
      ps.addBatch();  
    }  
    ps.executeBatch();  

      在例 1中, PreparedStatement被用来多次执行INSERT语句. 在这里, 执行了100次INSERT操作, 共有101次网络往返.

       其中,1次往返是预储PreparedStatement, 另外100次往返执行每个迭代.

    在例2中, 当在100次INSERT操作中使用addBatch()方法时, 只有两次网络往返.

       1次往返是预储PreparedStatement, 另一次是执行batch命令. 虽然Batch命令会用到更多的数据库的CPU周期, 但是通过减少网络往返,性能得到提高.记住, JDBC的性能最大的增进是减少JDBC驱动与数据库之间的网络通讯.次数


      注:Oracel 10G的JDBC Driver限制最大Batch size是16383条,如果addBatch超过这个限制,那么executeBatch时就会出现“无效的批值”(Invalid Batch Value) 异常。因此在如果使用的是Oracle10G,在此bug减少前,Batch size需要控制在一定的限度。

           同样mysql 5.5.28 批量执行的数据最大限度是多少不清楚,但自己试了1w,2w,3w 都没问题,记得在url 后面添加:rewriteBatchedStatements=true 表示批量插入,如果不添加的话即使使用addbatch() ,executeBatch() 在后台入库的地方还是不会一次请求入库而是多次请求入库。

     

    3、选择合适的光标类型
    的光标类型以最大限度的适用你的应用程序. 本节主要讨论三种光标类型的性能问题.
    对于从一个表中顺序读取所有记录的情况来说, Forward-Only型的光标提供了最好的性能. 获取表中的数据时, 没有哪种方法比使用Forward-Only型的光标更快. 但不管怎样, 当程序中必须按无次序的方式处理数据行时, 这种光标就无法使用了.

    对于程序中要求与数据库的数据同步以及要能够在结果集中前后移动光标, 使用JDBC的Scroll-Insensitive型光标是较理想的选择. 此类型的光标在第一次请求时就获取了所有的数据(当JDBC驱动采用'lazy'方式获取数据时或许是很多的而不是全部的数据)并且储存在客户端. 因此, 第一次请求会非常慢, 特别是请求长数据时会理严重. 而接下来的请求并不会造成任何网络往返(当使用'lazy'方法时或许只是有限的网络交通) 并且处理起来很快. 因为第一次请求速度很慢, Scroll-Insensitive型光标不应该被使用在单行数据的获取上. 当有要返回长数据时, 开发者也应避免使用Scroll-Insensitive型光标, 因为这样可能会造成内存耗尽. 有些Scroll-Insensitive型光标的实现方式是在数据库的临时表中缓存数据来避免性能问题, 但多数还是将数据缓存在应用程序中.

    Scroll-Sensitive型光标, 有时也称为Keyset-Driven光标, 使用标识符, 像数据库的ROWID之类. 当每次在结果集移动光标时, 会重新该标识符的数据. 因为每次请求都会有网络往返, 性能可能会很慢. 无论怎样, 用无序方式的返回结果行对性能的改善是没有帮助的.

    现在来解释一下这个, 来看这种情况. 一个程序要正常的返回1000行数据到程序中. 在执行时或者第一行被请求时, JDBC驱动不会执行程序提供的SELECT语句. 相反, 它会用键标识符来替换SELECT查询, 例如, ROWID. 然后修改过的查询都会被驱动程序执行,跟着会从数据库获取所有1000个键值. 每一次对一行结果的请求都会使JDBC驱动直接从本地缓存中找到相应的键值, 然后构造一个包含了'WHERE ROWID=?'子句的最佳化查询, 再接着执行这个修改过的查询, 最后从服务器取得该数据行.

    当程序无法像Scroll-Insensitive型光标一样提供足够缓存时, Scroll-Sensitive型光标可以被替代用来作为动态的可滚动的光标. 

    4、使用有效的getter方法
    JDBC提供多种方法从ResultSet中取得数据, 像getInt(), getString(), 和getObject()等等. 而getObject()方法是最泛化了的, 提供了最差的性能。 这是因为JDBC驱动必须对要取得的值的类型作额外的处理以映射为特定的对象. 所以就对特定的数据类型使用相应的方法.

    更进一步的改善性能, 应在取得数据时提供字段的索引号, 例如, getString(1), getLong(2), 和getInt(3)等来替代字段名. 如果没有指定字段索引号, 网络交通不会受影响, 但会使转换和查找的成本增加. 例如, 假设你使用getString("foo") ... JDBC驱动可能会将字段名转为大写(如果需要), 并且在到字段名列表中逐个比较来找到"foo"字段. 如果可以, 直接使用字段索引, 将为你节省大量的处理时间.

    例如, 假设你有一个100行15列的ResultSet, 字段名不包含在其中. 你感兴趣的是三个字段 EMPLOYEENAME (字串型), EMPLOYEENUMBER (长整型), 和SALARY (整型). 如果你指定getString(“EmployeeName”), getLong(“EmployeeNumber”), 和getInt(“Salary”), 查询旱每个字段名必须被转换为metadata中相对应的大小写, 然后才进行查找. 如果你使用getString(1), getLong(2), 和getInt(15). 性能就会有显著改善.

    5、获取自动生成的键值
    有许多数据库提供了隐藏列为表中的每行记录分配一个唯一键值. 很典型, 在查询中使用这些字段类型是取得记录值的最快的方式, 因为这些隐含列通常反应了数据在磁盘上的物理位置. 在JDBC3.0之前, 应用程序只可在插入数据后通过立即执行一个SELECT语句来取得隐含列的值.

    例 3: JDBC3.0之前

    //插入行  
    int rowcount = stmt.executeUpdate (  
       "insert into LocalGeniusList (name) values ('Karen')");  
    // 现在为新插入的行取得磁盘位置 - rowid  
    ResultSet rs = stmt.executeQuery (  
       "select rowid from LocalGeniusList where name = 'Karen'");  

    这种取得隐含列的方式有两个主要缺点. 第一, 取得隐含列是在一个独立的查询中, 它要透过网络送到服务器后再执行. 第二, 因为不是主键, 查询条件可能不是表中的唯一性ID. 在后面一个例子中, 可能返回了多个隐含列的值, 程序无法知道哪个是最后插入的行的值.


    (译者:由于不同的数据库支持的程度不同,返回rowid的方式各有差异。在SQL Server中,返回最后插入的记录的id可以用这样的查询语句:SELECT @IDENTITY )

    JDBC3.0规范中的一个可选特性提供了一种能力, 可以取得刚刚插入到表中的记录的自动生成的键值. 

    例 4: JDBC3.0之后

    int rowcount = stmt.executeUpdate (  
       "insert into LocalGeniusList (name) values ('Karen')",  
    // 插入行并返回键值  
    Statement.RETURN_GENERATED_KEYS);  
    ResultSet rs = stmt.getGeneratedKeys ();  
    // 得到生成的键值  

    现在, 程序中包含了一个唯一性ID, 可以用来作为查询条件来快速的存取数据行, 甚至于表中没有主键的情况也可以.

    这种取得自动生成的键值的方式给JDBC的开发者提供了灵活性, 并且使存取数据的性能得到提升.

    6、选择合适的数据类型
    接收和发送某些数据可能代价昂贵. 当你设计一个schema时, 应选择能被最有效地处理的数据类型. 例如, 整型数就比浮点数或实数处理起来要快一些. 浮点数的定义是按照数据库的内部规定的格式, 通常是一种压缩格式. 数据必须被解压和转换到另外种格式, 这样它才能被数据的协议处理.

    7、获取ResultSet
    由于数据库系统对可滚动光标的支持有限, 许多JDBC驱动程序并没有实现可滚动光标. 除非你确信数据库支持可滚动光标的结果集, 否则不要调用rs.last()和rs.getRow()方法去找出数据集的最大行数. 因为JDBC驱动程序模拟了可滚动光标, 调用rs.last()导致了驱动程序透过网络移到了数据集的最后一行. 取而代之, 你可以用ResultSet遍历一次计数或者用SELECT查询的COUNT函数来得到数据行数.

    通常情况下,请不要写那种依赖于结果集行数的代码, 因为驱动程序必须获取所有的数据集以便知道查询会返回多少行数据.
     

    三、preparestatement 防止sql注入

    在JDBC应用中,如果你已经是稍有水平开发者,你就应该始终以PreparedStatement代替Statement.也就是说,在任何时候都不要使用Statement.基于以下的原因:

    1、代码的可读性和可维护性.虽然用PreparedStatement来代替Statement会使代码多出几行,但这样的代码无论从可读性还是可维护性上来说.都比直接用Statement的代码高很多档次:
    stmt.executeUpdate("insert into tb_name (col1,col2,col2,col4) values ('"+var1+"','"+var2+"',"+var3+",'"+var4+"')");
    perstmt = con.prepareStatement("insert into tb_name (col1,col2,col2,col4) values (?,?,?,?)");

    perstmt.setString(1,var1);perstmt.setString(2,var2);perstmt.setString(3,var3);perstmt.setString(4,var4);

    perstmt.executeUpdate();
    不用我多说,对于第一种方法.别说其他人去读你的代码,就是你自己过一段时间再去读,都会觉得伤心.

    2、PreparedStatement尽最大可能提高性能.每一种数据库都会尽最大努力对预编译语句提供最大的性能优化.因为预编译语句有可能被重复调用.所以语句在被DB的编译器编译后的执行代码被缓存下来,那么下次调用时只要是相同的预编译语句就不需要编译,只要将参数直接传入编译过的语句执行代码中(相当于一个涵数)就会得到执行.这并不是说只有一个 Connection中多次执行的预编译语句被缓存,而是对于整个DB中,只要预编译的语句语法和缓存中匹配.那么在任何时候就可以不需要再次编译而可以直接执行.而statement的语句中,即使是相同一操作,而由于每次操作的数据不同所以使整个语句相匹配的机会极小,几乎不太可能匹配.比如:insert into tb_name (col1,col2) values ('11','22');insert into tb_name (col1,col2) values ('11','23');即使是相同操作但因为数据内容不一样,所以整个个语句本身不能匹配,没有缓存语句的意义.事实是没有数据库会对普通语句编译后的执行代码缓存.这样每执行一次都要对传入的语句编译一次.
    当然并不是所以预编译语句都一定会被缓存,数据库本身会用一种策略,比如使用频度等因素来决定什么时候不再缓存已有的预编译结果.以保存有更多的空间存储新的预编译语句.

    3、最重要的一点是极大地提高了安全性.
    即使到目前为止,仍有一些人连基本的恶义SQL语法都不知道.String sql = "select * from tb_name where name= '"+varname+"' and passwd='"+varpasswd+"'";如果我们把[' or '1' = '1]作为varpasswd传入进来.用户名随意,看看会成为什么?
    select * from tb_name = '随意' and passwd = '' or '1' = '1';因为'1'='1'肯定成立,所以可以任何通过验证.更有甚者:把[';drop table tb_name;]作为varpasswd传入进来,则:select * from tb_name = '随意' and passwd = '';drop table tb_name;有些数据库是不会让你成功的,但也有很多数据库就可以使这些语句得到执行.
    而如果你使用预编译语句.你传入的任何内容就不会和原来的语句发生任何匹配的关系.(前提是数据库本身支持预编译,但上前可能没有什么服务端数据库不支持编译了,只有少数的桌面数据库,就是直接文件访问的那些)只要全使用预编译语句,你就用不着对传入的数据做任何过虑.而如果使用普通的statement, 有可能要对drop,;等做费尽心机的判断和过虑.
    上面的几个原因,还不足让你在任何时候都使用PreparedStatement吗?

     

    总结: 上面是三篇文章,三篇文章详细介绍了statement 和preparestatement 两个对象的使用以及效率、安全问题。在实际项目中如果能够使用preparestatement  还是建议使用preparestatement  原因有3:

    1)、上面说了 如果sql中只有数值在变则效率高

    2)、preparestatement 具有防sql注入

    3)、代码可读性比较好

     

    实例:下面这个比喻很好,很明确的说明了批量添加,并且从中也可以看出在批量添加的时候PreparedStatement为什么比Statement快的原因~

    Statement和PreparedStatement的区别就不多废话了,直接说PreparedStatement最重要的addbatch()结构的使用.

    PreparedStatement  的addBatch和executeBatch实现批量添加

    1.建立链接   

       Connection    connection =getConnection();

    2.不自动 Commit (瓜子不是一个一个吃,全部剥开放桌子上,然后一口舔了)

    connection.setAutoCommit(false);   

    3.预编译SQL语句,只编译一回哦,效率高啊.(发明一个剥瓜子的方法,以后不要总想怎么剥瓜子好.就这样剥.)
    PreparedStatement statement = connection.prepareStatement("INSERT INTO TABLEX VALUES(?, ?)");   

    4.来一个剥一个,然后放桌子上

    //记录1
    statement.setInt(1, 1); 
    statement.setString(2, "Cujo"); 
    statement.addBatch();   

    //记录2
    statement.setInt(1, 2); 
    statement.setString(2, "Fred"); 
    statement.addBatch();   

    //记录3
    statement.setInt(1, 3); 
    statement.setString(2, "Mark"); 
    statement.addBatch();   

    //批量执行上面3条语句. 一口吞了,很爽
    int [] counts = statement.executeBatch();   

    //Commit it 咽下去,到肚子(DB)里面
    connection.commit();

     

    statement 对象的addBatch 和 executeBatch 来实现批量添加
    stmt.addBatch("update  TABLE1 set 题目="盛夏话足部保健1"   where id="3407"");
    stmt.addBatch("update  TABLE1 set 题目="夏季预防中暑膳食1" where id="3408""); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("11","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("12","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("13","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("14","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("15","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("16","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("17","12","13","","")"); 
    stmt.addBatch("INSERT INTO  TABLE1  VALUES("18","12","13","","")"); 

    int [] updateCounts=stmt.executeBatch(); 
    cn.commit();

     

    实例:批量添加

    public static void insertData(List<Map<String,String>> list,Logger log){  
        //获取的数据  
        List <Map<String,String>> nlist= list;  
        String upsql="update   hrd_staff  set position =?  where id=?";  
        Iterator<Map<String,String>> iter= nlist.iterator();  
        Connection con= Utils.getCon();  
        int count=0;  
        try {  
            //在皮脸添加的时候注意事务提交方式  
            con.setAutoCommit(false);  
            //PreparedStatement方法的使用  
            PreparedStatement pstm = con.prepareStatement(upsql);  
            while(iter.hasNext()){  
                count++;  
                Map<String,String> map= iter.next();  
                String jon_name= map.get("job_name");  
                String uid= map.get("uid");  
                pstm.setString(1,jon_name);  
                pstm.setString(2,uid);  
                //添加到缓存中  
                pstm.addBatch();  
                // 如果数据量很大,不能一次性批量添加所以我们要分批次添加,这里就是300条一次  
                if(count%300==0){  
                    //持久化  
                    int []res=pstm.executeBatch();  
                    //提交事务,持久化数据  
                    con.commit();  
                    pstm.clearBatch();  
                    log.info("300整除插入结果: "+res.length);  
                }  
            }  
            //小于300条的在这里持久化  
            int []ress= pstm.executeBatch();  
            //事务提交持久化  
            con.commit();  
            pstm.clearBatch();  
            log.info("插入数据结果:"+ress.length);  
        } catch (SQLException e) {  
            try {  
                con.rollback();  
            } catch (SQLException e1) {  
                // TODO Auto-generated catch block  
                e1.printStackTrace();  
            }  
            e.printStackTrace();  
        }finally{  
            try {  
                if(null!=con){  
                con.close();  
                con.setAutoCommit(true);  
                }  
            } catch (SQLException e) {  
                // TODO Auto-generated catch block  
                e.printStackTrace();  
            }  
        }  
    }  

    这里除了下面说的url中的批量设置外,我们也要注意事务的设置,不能设置为自动提交,要批量添加后在提交事务

     

    总结:

    addBatch() 就是把你的处理内容添加到批处理单元中。即添加到了batch中。你可以循环加入很多,数据库都不会处理,直到调用如下代码executeBatch() 此时,数据库把刚才加到batch中的命令批量处理。

     

    使用批量插入的好处: , 当在100次INSERT操作中使用addBatch()方法时, 只有两次网络往返. 1次往返是预储statement, 另一次是执行batch命令. 虽然Batch命令会用到更多的数据库的CPU周期, 但是通过减少网络往返,性能得到提高. 记住, JDBC的性能最大的增进是减少JDBC驱动与数据库之间的网络通讯. 如果没有使用批处理则网络往返101次这样会耗很多时间,自然效率也就一般

     

    这里要注意:在mysql 下使用批量执行的时候要在,url 后面添加手动设置支持批量添加 实例如下:

     String url="jdbc:mysql://localhost:3306/music?rewriteBatchedStatements=true";

    // 默认情况下rewriteBatchedStatements 的值为false 也就是批量添加功能是关闭的,如果使用则要手动开启!

    还有就是事务的设置,不能使自动提交,要批量添加后才提交!!!

    展开全文
  • 每次都看别人的,今天我自己写下来,以后就不用看别人的了语法PREPARE statement_name FROM sql_text /*定义*/EXECUTE statement_name [USING variable [,variable...]] /*执行预处理语句*/DEALLOCATE PREPARE ...
  • 每次都看别人的,今天我自己写下来,以后就不用看别人的了语法PREPARE statement_name FROM sql_text /*定义*/EXECUTE statement_name [USING variable [,variable...]] /*执行预处理语句*/DEALLOCATE PREPARE ...
  • php prepare

    2018-10-25 14:14:00
    详见...php pdo prepare真的能防止sql injection吗, 这里 php使用pdo 连接mysql , 会使用prepare预处...
  • PREPARE STATEMENT

    2017-10-12 17:03:51
    文章来自:... MySQL官方将prepare、execute、deallocate统称为PREPARE STATEMENT。 我习惯称其为【预处理语句】。 其用法十分简单, PREPARE stmt_name FROM preparable_stmt
  • Refactor prepare read

    2020-11-22 23:06:20
    ve rebased your latest version of the prepare-stream branch onto latest master and then bolted my work on top of that. Took some doing, at first I didn't notice you had considerably altered your &...
  • Codeclimate prepare

    2020-12-01 11:43:08
    <div><p>Codeclimate prepare <pre><code> $ codeclimate help prepare </code></pre>该提问来源于开源项目:codeclimate/codeclimate</p></div>
  • 原子线程调用Toast报Can'tcreatehandlerinsidethreadthathasnotcalledLooper.prepare()错误 今天用子线程调Toast报了一个Can'tcreatehandlerinsidethreadthathasnotcalledLooper.prepare()错误。 因为toast的现实...
  • PDO::preparePDO::prepare — 准备要执行的SQL语句并返回一个 PDOStatement 对象...= 0.1.0)说明语法public PDOStatement PDO::prepare ( string $statement [, array $driver_options = array() ] )为 PDO...
  • 主要帮助大家学习理解Mysql prepare预处理语句,对prepare预处理语句感兴趣的小伙伴们可以参考一下
  • 首先大家必须了解mysqli扩展类中为什么要使用prepare准备语句?这是因为prepare可以解决大访问量的网站给数据库服务器所带来的负载和开销,使用prepare准备语句可以确保创建查询的稳定性和安全性。下面向大家介绍...
  • 原标题:MySQL prepare语句的SQL语法 PREPARE语句用于预备一个语句,并指定名称statement_name,以后引用该语句MySQL prepare语法:PREPARE statement_name FROM preparable_SQL_statement; /*定义*/EXECUTE ...
  • Prepare test net

    2020-12-08 23:46:34
    <div><p>Prepare a test net for Parachains.</p><p>该提问来源于开源项目:paritytech/cumulus</p></div>
  • PDO::prepare讲解

    2021-01-20 01:35:43
    PDO::prepare PDO::prepare — 准备要执行的SQL语句并返回一个 PDOStatement 对象(PHP 5 >= 5.1.0, PECL pdo >= 0.1.0) 说明 语法 public PDOStatement PDO::prepare ( string $statement [, array $driver_options...
  • <div><p>supercedes #37 ... http://php.net/manual/en/mysqli.prepare.php and http://php.net/manual/en/mysqli.quickstart.prepared-statements.php</p><p>该提问来源于开源项目:ezSQL/ezsql</p></div>
  • MySQL prepare语法:PREPAREstatement_nameFROMpreparable_SQL_statement;/*定义*/EXECUTEstatement_name [USING@var_name[, @var_name] ...];/*执行预处理语句*/{DEALLOCATE | DROP}PREPAREstatement_name /*删除...
  • MySQL prepare语法:PREPAREstatement_nameFROMpreparable_SQL_statement;/*定义*/EXECUTEstatement_name [USING@var_name[, @var_name] ...];/*执行预处理语句*/{DEALLOCATE | DROP}PREPAREstatement_name /*删除...
  • 标签:语法PREPAREstatement_nameFROMsql_text/*定义*/EXECUTEstatement_name[USINGvariable[,variable...]]/*执行预处理语句*/DEALLOCATEPREPAREstatement_name/*删除定义*/例mysql>PREPAREprodFROM...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 48,609
精华内容 19,443
关键字:

prepare