数据库连接池(Connection pooling)是程序启动时建立足够的数据库连接,并将这些连接组成一个连接池,由程序动态地对池中的连接进行申请,使用,释放。
简单的说:创建数据库连接是一个很耗时的 *** 作,也容易对数据库造成安全隐患。所以,在程序初始化的时候,集中创建多个数据库连接,并把他们集中管理,供程序使用,可以保证较快的数据库读写速度,还更加安全可靠。
不使用数据库连接池
如果不使用数据库连接池,对于每一次SQL *** 作,都要走一遍下面完整的流程:
1TCP建立连接的三次握手(客户端与 MySQL服务器的连接基于TCP协议)
2MySQL认证的三次我收
3真正的SQL执行
4MySQL的关闭
5TCP的四次握手关闭
可以看出来,为了执行一条SQL,需要进行大量的初始化与关闭 *** 作
使用数据库连接池
如果使用数据库连接池,那么会 事先申请(初始化)好 相关的数据库连接,然后在之后的SQL *** 作中会复用这些数据库连接, *** 作结束之后数据库也不会断开连接,而是将数据库对象放回到数据库连接池中
资源重用:由于数据库连接得到重用,避免了频繁的创建、释放连接引起的性能开销,在减少系统消耗的基础上,另一方面也增进了系统运行环境的平稳性(减少内存碎片以及数据库临时进程/线程的数量)。
更快的系统响应速度:数据库连接池在初始化过程中,往往已经创建了若干数据库连接置于池中备用。 此时连接的初始化工作均已完成。对于业务请求处理而言,直接利用现有可用连接,避免了从数据库连接初始化和释放过程的开销,从而缩减了系统整体响应时间。
统一的连接管理,避免数据库连接泄露:在较为完备的数据库连接池实现中,可根据预先的连接占用超时设定,强制收回被占用连接。从而避免了常规数据库连接 *** 作中可能出现的资源泄露。
如果说你的服务器CPU是4核i7的,连接池大小应该为((42)+1)=9
相关视频推荐
90分钟搞懂数据库连接池技术|linux后台开发
《tcp/ip详解卷一》: 150行代码拉开协议栈实现的篇章
学习地址:C/C++Linux服务器开发/后台架构师零声教育-学习视频教程-腾讯课堂
需要C/C++ Linux服务器架构师学习资料加qun 812855908 获取(资料包括 C/C++,Linux,golang技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg 等),免费分享
源码下载
下载方式:>
本文将详细介绍如何使用Connection对象连接数据库。对于不同的NET数据提供者,ADONET采用不同的Connection对象连接数据库。这些Connection对象为我们屏蔽了具体的实现细节,并提供了一种统一的实现方法。
Connection类有四种:SqlConnection,OleDbConnection,OdbcConnection和OracleConnection。
SqlConnection类的对象连接SQL Server数据库;OracleConnection 类的对象连接Oracle数据库;
OleDbConnection类的对象连接支持OLE DB的数据库,如Access;而OdbcConnection类的对象连接任何支持ODBC的数据库。与数据库的所有通讯最终都是通过Connection对象来完成的。
SqlConnection类
Connection 用于与数据库“对话”,并由特定提供程序的类(如 SqlConnection)表示。尽管SqlConnection类是针对Sql Server的,但是这个类的许多属性、方法与事件和OleDbConnection及OdbcConnection等类相似。本章将重点讲解SqlConnection特定的属性与方法,其他的Connection类你可以参考相应的帮助文档。
注意:使用不同的Connection对象需要导入不同的命名空间。OleDbConnection的命名空间为SystemDataOleDb。SqlConnection的命名空间为SystemDataSqlClient。OdbcConnection的命名空间为SystemDataOdbc。OracleConnection的命名空间为SystemDataOracleClinet。
SqlConnection属性:
属性 说明
ConnectionString 其返回类型为string,获取或设置用于打开 SQL Server 数据库的字符串。
ConnectionTimeOut 其返回类型为int,获取在尝试建立连接时终止尝试并生成错误之前所等待的时间。
Database 其返回类型为string,获取当前数据库或连接打开后要使用的数据库的名称。
DataSource 其返回类型为string,获取要连接的 SQL Server 实例的名称。
State 其返回类型为ConnectionState,取得当前的连接状态:Broken、Closed、Connecting、Fetching或Open。
ServerVersion 其返回类型为string,获取包含客户端连接的 SQL Server 实例的版本的字符串。
PacketSize 获取用来与 SQL Server 的实例通信的网络数据包的大小(以字节为单位)。这个属性只适用于SqlConnection类型
SqlConnection方法:
方法 说明
Close() 其返回类型为void,关闭与数据库的连接。
CreateCommand() 其返回类型为SqlCommand,创建并返回一个与 SqlConnection 关联的 SqlCommand 对象。
Open() 其返回类型为void,用连接字符串属性指定的属性打开数据库连接
SqlConnection事件:
事件 说明
StateChange 当事件状态更改时发生。 (从 DbConnection 继承。)
InfoMessage 当 SQL Server 返回一个警告或信息性消息时发生。
提示:可以用事件让一个对象以某种方式通知另一对象产生某些事情。例如我们在Windows系统中选择“开始”菜单,一旦单击鼠标时,就发生了一个事件,通知 *** 作系统将“开始”菜单显示出来。
使用SqlConnection对象连接SQL Server数据库
我们可以用SqlConnection()构造函数生成一个新的SqlConnection对象。这个函数是重载的,即我们可以调用构造函数的不同版本。SqlConnection()的构造函数如下表所示:
构造函数 说明
SqlConnection () 初始化 SqlConnection 类的新实例。
SqlConnection (String) 如果给定包含连接字符串的字符串,则初始化 SqlConnection 类的新实例。
假设我们导入了SystemDataSqlClient命名空间,则可以用下列语句生成新的SqlConnection对象:
SqlConnection mySqlConnection = new SqlConnection();
程序代码说明:在上述语法范例的程序代码中,我们通过使用“new“关键字生成了一个新的SqlConnection对象,并且将其命名为mySqlConnection。
现在我们就可以使用如下两种方式连接数据库,即采用集成的Windows验证和使用Sql Server身份验证进行数据库的登录。
集成的Windows身份验证语法范例
string connectionString="server=localhost;database=Northwind;
integrated security=SSPI";
程序代码说明:在上述语法范例的程序代码中,我们设置了一个针对Sql Server数据库的连接字符串。其中server表示运行Sql Server的计算机名,由于在本书中,ASPNET程序和数据库系统是位于同一台计算机的,所以我们可以用localhost取代当前的计算机名。database表示所使用的数据库名,这里设置为Sql Server自带的一个示例数据库--Northwind。由于我们希望采用集成的Windows验证方式,所以设置 integrated security为SSPI即可。
Sql Server 2005中的Windows身份验证模式如下:
注意:在使用集成的Windows验证方式时,并不需要我们输入用户名和口令,而是把登录Windows时输入的用户名和口令传递到Sql Server。然后Sql Server检查用户清单,检查其是否具有访问数据库的权限。而且数据库连接字符串是不区分大小写的。
采用Sql Server身份验证的语法范例
string connectionString = "server=localhost;database=Northwind;uid=sa;pwd=sa";
程序代码说明:在上述语法范例的程序代码中,采用了使用已知的用户名和密码验证进行数据库的登录。uid为指定的数据库用户名,pwd为指定的用户口令。为了安全起见,一般不要在代码中包括用户名和口令,你可以采用前面的集成的Windows验证方式或者对WebConfig文件中的连接字符串加密的方式提高程序的安全性。
Sql Server 2005中的Sql Server身份验证模式如下:
如果你使用其他的数据提供者的话,所产生的连接字符串也具有相类似的形式。例如我们希望以OLE DB的方式连接到一个Oracle数据库,其连接字符串如下:
string connectionString = "data source=localhost;initial catalog=Sales;
use id=sa;password=;provider=MSDAORA";
程序代码说明:在上述语法范例的程序代码中,通过专门针对Oracle数据库的OLE DB提供程序,实现数据库的连接。data source 表示运行Oracle数据库的计算机名,initial catalog表示所使用的数据库名。provider表示使用的OLE DB提供程序为MSDAORA。
Access数据库的连接字符串的形式如下:
string connectionString = "provider=MicrosoftJetOLEDB40;
@”data source=c:\DataSource\Northwindmdb”;
程序代码说明:在上述语法范例的程序代码中,通过专门针对Access数据库的OLE DB提供程序,实现数据库的连接。这使用的的OLE DB提供程序为MicrosoftJetOLEDB40,并且数据库存放在c:\DataSource目录下,其数据库文件为Northwindmdb。
现在我们就可以将数据库连接字符串传人SqlConnection()构造函数,例如:
string connectionString = "server=localhost;database=Northwind;uid=sa;pwd=sa";
SqlConnection mySqlConnection = new SqlConnection(connectionString);
或者写成
SqlConnection mySqlConnection =new SqlConnection(
"server=localhost;database=Northwind;uid=sa;pwd=sa");
在前面的范例中,通过使用“new“关键字生成了一个新的SqlConnection对象。因此我们也可以设置该对象的ConnectionString属性,为其指定一个数据库连接字符串。这和将数据库连接字符串传人SqlConnection()构造函数的功能是一样的。
SqlConnection mySqlConnection = new SqlConnection();
mySqlConnectionConnectionString = "server=localhost;database=Northwind;uid=sa;pwd=sa";
注意:只能在关闭Connection对象时设置ConnectionString属性。
打开和关闭数据库连接
生成Connection对象并将其设置ConnectionString属性设置为数据库连接的相应细节之后,就可以打开数据库连接。为此可以调用Connection对象的Open()方法。其方法如下:
mySqlConnectionOpen();
完成数据库的连接之后,我们可以调用Connection对象的Close()方法关闭数据库连接。例如:
mySqlConnectionClose();
下面是一个显示如何用SqlConnection对象连接Sql Server Northwind数据库的实例程序,并且显示该SqlConnection对象的一些属性。
范例程序代码如下:
01 public partial class _Default : SystemWebUIPage
02 {
03 protected void Page_Load(object sender, EventArgs e)
04 {
05 //建立数据库连接字符串
06 string connectionString = "server=localhost;database=Northwind;
07 integrated security=SSPI";
08 //将连接字符串传入SqlConnection对象的构造函数中
09 SqlConnection mySqlConnection = new SqlConnection(connectionString);
10 try
11 {
12 //打开连接
13 mySqlConnectionOpen();
14 //利用label控件显示mySqlConnection对象的ConnectionString属性
15 lblInfoText = "<b>mySqlConnection对象的ConnectionString属性为:<b>" +
16 mySqlConnectionConnectionString + "<br>";
17 lblInfoText += "<b>mySqlConnection对象的ConnectionTimeout属性为<b>" +
18 mySqlConnectionConnectionTimeout + "<br>";
19 lblInfoText += "<b>mySqlConnection对象的Database属性为<b>" +
20 mySqlConnectionDatabase + "<br>";
21 lblInfoText += "<b>mySqlConnection对象的DataSource属性为<b>" +
22 mySqlConnectionDataSource + "<br>";
23 lblInfoText += "<b>mySqlConnection对象的PacketSize属性为<b>" +
24 mySqlConnectionPacketSize + "<br>";
25 lblInfoText += "<b>mySqlConnection对象的ServerVersion属性为<b>" +
26 mySqlConnectionServerVersion + "<br>";
27 lblInfoText += "<b>mySqlConnection对象的当前状态为<b>" +
28 mySqlConnectionState + "<br>";
29 }
30 catch (Exception err)
31 {
32 lblInfoText = "读取数据库出错";
33 lblInfoText += errMessage;
34 }
35 finally
36 {
37 //关闭与数据库的连接
38 mySqlConnectionClose();
39 lblInfoText += "<br><b>关闭连接后的mySqlConnection对象的状态为:</b>";
40 lblInfoText += mySqlConnectionStateToString();
41 }
42 }
43 }
程序代码说明:在上述范例的程序代码中,我们利用try catch finally对数据库连接进行异常处理。当无法连接数据库时将抛出异常,并显示出错信息,见catch代码块所示。在此程序中,无论是否发生异常,都可以通过finally区块关闭数据库的连接,从而节省计算机资源,提高了程序的效率和可扩展性。
执行结果:
当然,我们还可以采用一种更加简便的方法来实现上述程序的功能。这就是将SqlConnection对象包含到using区块中,这样程序会自动调用Dispose()方法释放SqlConnection对象所占用的系统资源,无需再使用SqlConnection对象的Close()方法。
范例程序代码如下:
01 public partial class _Default : SystemWebUIPage
02 {
03 protected void Page_Load(object sender, EventArgs e)
04 {
05 string connectionString = "server=localhost;database=Northwind;
06 integrated security=SSPI";
07 SqlConnection mySqlConnection = new SqlConnection(connectionString);
08 using (mySqlConnection)
09 {
10 mySqlConnectionOpen();
11 lblInfoText = "<b>mySqlConnection对象的ConnectionString属性为:<b>" +
12 mySqlConnectionConnectionString + "<br>";
13 lblInfoText += "<b>mySqlConnection对象的ConnectionTimeout属性为<b>" +
14 mySqlConnectionConnectionTimeout + "<br>";
15 lblInfoText += "<b>mySqlConnection对象的Database属性为<b>" +
16 mySqlConnectionDatabase + "<br>";
17 lblInfoText += "<b>mySqlConnection对象的DataSource属性为<b>" +
18 mySqlConnectionDataSource + "<br>";
19 lblInfoText += "<b>mySqlConnection对象的PacketSize属性为<b>" +
20 mySqlConnectionPacketSize + "<br>";
21 lblInfoText += "<b>mySqlConnection对象的ServerVersion属性为<b>" +
22 mySqlConnectionServerVersion + "<br>";
23 lblInfoText += "<b>mySqlConnection对象的当前状态为<b>"+
24 mySqlConnectionState + "<br>";
25 }
26 lblInfoText += "<br><b>关闭连接后的mySqlConnection对象的状态为:</b>";
27 lblInfoText += mySqlConnectionStateToString();
28 }
29 }
程序代码说明:在上述范例的程序代码中,采用using(mySqlConnection)的形式使得代码更加简洁,并且其最大的优点就是无需编写finally区块代码,可以自动关闭与数据库的连接。
连接池
打开与关闭数据库都是比较耗时的。为此,ADONET自动将数据库连接存放在连接池中。连接池可以大幅度提高程序的性能和效率,因为我们不必等待建立全新的数据库连接过程,而是直接利用现成的数据库连接。注意,利用Close()方法关闭连接时,并不是实际关闭连接,而是将连接标为未用,放在连接池中,准备下一次复用。
如果在连接字符串中提供相同的细节,即相同的数据库,用户名,密码等等,则可以直接取得并返回池中的连接。然后可以用这个连接访问数据库。
使用SqlConnection对象时,可以在连接字符串中指定max pool size,表示连接池允许的最大连接数(默认为100),也可以指定min pool size表示连接池允许的最小连接数(默认为0)。下面的代码指定了SqlConnection对象的max pool size为10,min pool size为5。
SqlConnection mySqlConnection = new SqlConnection("server=localhost;database=Northwind;
integrated security=SSPI;"+"max pool size=10;min pool size=5");
程序代码说明:在上述范例的程序代码中,程序最初在池中生成5个SqlConnection对象。池中可以存储最多10个SqlConnection对象。如果要打开新的SqlConnection对象时,池中的对象全部都在使用中,则请求要等待一个SqlConnection对象关闭,然后才可以使用新的SqlConnection对象。如果请求等待时间超过ConnectionTimeout属性指定的秒数,则会抛出异常。
下面通过一个程序来显示连接池的性能优势。在应用此程序过程我们要先引用SystemDataSqlClinet和SystemText命名空间。
范例程序代码如下:
01 public partial class _Default : SystemWebUIPage
02 {
03 protected void Page_Load(object sender, EventArgs e)
04 {
05 //设置连接池的最大连接数为5,最小为1
06 SqlConnection mySqlConnection =new SqlConnection(
07 "server=localhost;database=Northwind;integrated security=SSPI;"+
08 "max pool size=5;min pool size=1");
09 //新建一个StringBuilder对象
10 StringBuilder htmStr = new StringBuilder("");
11 for (int count = 1; count <= 5; count++)
12 {
13 //使用Append()方法追加字符串到StringBuilder对象的结尾处
14 htmStrAppend("连接对象 "+count);
15 htmStrAppend("<br>");
16 //设置一个连接的开始时间
17 DateTime start = DateTimeNow;
18 mySqlConnectionOpen();
19 //连接所用的时间
20 TimeSpan timeTaken = DateTimeNow - start;
21 htmStrAppend("连接时间为 "+timeTakenMilliseconds+"毫秒");
22 htmStrAppend("<br>");
23 htmStrAppend("mySqlConnection对象的状态为" + mySqlConnectionState);
24 htmStrAppend("<br>");
25 mySqlConnectionClose();
26 }
27 //将StringBuilder对象的包含的字符串在label控件中显示出来
28 lblInfoText = htmStrToString();
29 }
30 }
程序代码说明:在上述范例的程序代码中,我们将在连接池中重复5次打开一个SqlConnection对象,DateTimeNow表示当前的时间。timeTaken表示从连接开始到打开连接所用的时间间隔。可以看出,打开第一个连接的时间比打开后续连接的时间要长,因为第一个连接要实际连接数据库。被关闭之后,这个连接存放在连接池中。再次打开连接时,只要从池中直接读取即可,速度非常快。
提示:String 对象是不可改变的。每次使用 SystemString 类中的方法之一时,都要在内存中创建一个新的字符串对象,这就需要为该新对象分配新的空间。在需要对字符串执行重复修改的情况下,与创建新的 String 对象相关的系统开销可能会非常昂贵。如果要修改字符串而不创建新的对象,则可以使用 SystemTextStringBuilder 类。例如,当在一个循环中将许多字符串连接在一起时,使用 StringBuilder 类可以提升性能。Append 方法可用来将文本或对象的字符串表示形式添加到由当前 StringBuilder 对象表示的字符串的结尾处。
在ASPNET 20中,使用了一种在运行时解析为连接字符串值的新的声明性表达式语法,按名称引用数据库连接字符串。连接字符串本身存储在 Webconfig 文件中的 <connectionStrings> 配置节下面,以便易于在单个位置为应用程序中的所有页进行维护。
范例程序代码如下:
<xml version="10">
<configuration>
<connectionStrings>
<add name="Pubs" connectionString="Server=localhost;
Integrated Security=True;Database=pubs;Persist Security Info=True"
providerName="SystemDataSqlClient" />
<add name="Northwind" connectionString="Server=localhost;
Integrated Security=True;Database=Northwind;Persist Security Info=True"
providerName="SystemDataSqlClient" />
</connectionStrings>
<systemweb>
<pages styleSheetTheme="Default"/>
</systemweb>
</configuration>
程序代码说明:在上述范例的程序代码中,我们在WebConfig文件中的<connectionStrings> 配置节点下面设置了两个数据库连接字符串,分别指向pubs和Northwind两个示例数据库。注意,在20中引进了数据源控件,例如SqlDataSource 控件,我们可以将SqlDataSource 控件的 ConnectionString 属性被设置为表达式 <%$ ConnectionStrings:Pubs %>,该表达式在运行时由 ASPNET 分析器解析为连接字符串。还可以为SqlDataSource 的 ProviderName 属性指定一个表达式,例如 <%$ ConnectionStrings:PubsProviderName %>。其具体的用法和新特征将在以后的章节进行详细的介绍。现在有个基础的了解即可。
当然,我们也可以用下面的方式从配置文件直接读取数据库连接字符串。首先我们需要引用using SystemWebConfiguration命名空间,该命名空间包含用于设置 ASPNET 配置的类。
string connectionString =ConfigurationManagerConnectionStrings["Northwind"]ConnectionString;
程序代码说明:在上述范例的程序代码中,我们可以利用ConnectionStrings["Northwind"]读取相应的Northwind字符串。同理以可以利用ConnectionStrings["Pubs"]读取相应的Pubs字符串。
public static Object getInnter(Object con){
Object re=null;
Field f;
try {
f = con getClass() getDeclaredField( inner )
f setAccessible(true)
re= f get(con) //取得内部包装的Connection
f setAccessible(false)
} catch Exception e) {
}
return re;
}
以上代码运行后 输出
con Class Type is: mchange v c p impl NewProxyConnection
Inner con Class Type is: mysql jdbc JDBC Connection
Data from DB:
o and o is same object
Data from DB:
上述代码中 首先从数据库连接池获得一个连接 发现连接类型并不是mysql的数据库连接 而是 mchange v c p impl NewProxyConnection 根据类名中可以推测 从数据库连接池中获得的连接只是一个代理 接着 通过反射 取得这个对象中名为inner的属性 并打印其Class类型 发现这才是真正的mysql连接 关闭NewProxyConnection连接 再向池中请求一个新的连接 同样获取该连接内部的实际数据库连接对象 发现 第一次使用的实际数据库连接对象o 和第二次使用的对象o 是完全相同的
这说明 前后两次数据库连接的请求均返回了相同的数据库连接 关闭NewProxyConnection连接时 并没有真正关闭数据库连接 而只是将数据库连接放入连接池保存 使得数据库连接在连接池中得到了复用 而从连接池返回的NewProxyConnection对象 只是对真实数据库连接的包装
除了线程池和数据库连接池 对于普通的Java对象 在必要的时候 也可以进行池化管理 对于那些经常使用 并且创建很费时的大型对象来说 使用对象池维护 不仅可以节省获得对象实例的成本 还可以减轻GC频繁回收这些对象产生的系统压力 但对于生成对象开销很小的对象进行池化 反而可能得不偿失 维护对象池的成本可能会大于对象池带来的好处
注意 在JDK中 new *** 作的效率是相当高的 不需要担心频繁的new *** 作对系统有性能影响 但是new *** 作时所调用的类构造函数可能是非常费时的 对于这些对象 可以考虑池化
返回目录 Java程序性能优化 让你的Java程序更快 更稳定
编辑推荐
Java程序设计培训视频教程
J EE高级框架实战培训视频教程
J ME移动开发实战教学视频
Visual C++音频/视频技术开发与实战
Oracle索引技术
ORACLE G数据库开发优化指南
lishixinzhi/Article/program/Java/gj/201311/27800
日常数据管理工作中,需要处理存储在不同类型数据库系统的数据。对这些数据的管理,常见的是使用Navicat,DBeaver等管理工具。在对大量数据分析时,需要提取到Python/R中进行处理。下面 探索 Python调用MySQL,MongoDB,InfluxDB等多种类型数据库通用连接方法。实现方式是在Python中封装各类数据库接口包。
实现后的效果:1安全。接口信息封装便于保密管理;2复用。一次封装,永久复用;3上手快。方便不熟悉python和数据调用的同学,只会简单的sql即可使用,省时省力。
下面以MySQL,MongoDB,InfluxDB为例定义接口方法,然后把它们封装成1个通用方法。
mysql_get(sql,db):
mongo_get(sql,db):
influx_get(sql,db):
可以看到,以上函数共同调用的参数为sql和db。我们再增加一个参数db_type,将构造一个通用的方法对以上数据库调用。
同理,其他类型的数据库也可以加入到这个通用框架中,包括但不限于各类关系型,键值型,时序型数据库。
以上就是关于MySQL与Redis数据库连接池介绍(图示+源码+代码演示)全部的内容,包括:MySQL与Redis数据库连接池介绍(图示+源码+代码演示)、weblgic数据库连接已关闭、c# .NET设计系统,数据库的问题等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)