mysql 數(shù)據(jù)庫鏈接狀態(tài)確認實驗(推薦)
1.起因:
在做一個小系統(tǒng)的時候,我想數(shù)據(jù)量交互不大,就都用一個鏈接執(zhí)行算了,還能節(jié)省點資源,但是實際情況遇到了如下問題,在使用的過程中,有些數(shù)據(jù)操作會被轉(zhuǎn)移到其他線程,這樣;我這個簡單的想法遇到了問題,因為一個現(xiàn)場在使用這個鏈接的時候,其他線程也會同步插入執(zhí)行數(shù)據(jù)操作,這樣一個鏈接就會面臨共用的沖突,怎么辦呢,有如下三種方案:
1.1.數(shù)據(jù)兩次一次一聯(lián),一用,一釋放。
1.2.強制是數(shù)據(jù)庫的執(zhí)行放到一個現(xiàn)場,那么得把所有的執(zhí)行參數(shù)放到隊列中,有數(shù)據(jù)支持的線程按照隊列的順序執(zhí)行。也可以在使用的時候把鏈接索起來。這樣強制的使數(shù)據(jù)的處理串行。
1.3.做一個內(nèi)部的鏈接對象使用池,池中的對象可供服用,解決重復(fù)鏈接的問題,提供多個對象解決現(xiàn)場使用沖突的問題。
我選擇了方案三,做了如下實驗,記錄如下
2.實驗
2.1 確認數(shù)據(jù)鏈接狀態(tài),使用完的數(shù)據(jù)鏈接是什么狀態(tài)呢?
2.1.1 目的:我想根據(jù)鏈接的狀態(tài)判斷鏈接是在使用中,還是可以給別人使用;就是在可以使用的情況下,我就用該鏈接執(zhí)行,否則使用或者創(chuàng)建其他的鏈接。
代碼。
using System;
using System.Data;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
/// <summary>
/// 確認數(shù)據(jù)庫通常執(zhí)行完成是什么狀態(tài)
/// </summary>
public class Test1:Singleton<Test1>
{
public void main()
{
test1();
}
private void test1()
{
DBmsqlSub dBmsqlSub = new DBmsqlSub();
string sql = "insert into tb1(v1) values(2)";
dBmsqlSub.ExecuteNonQuery(sql);
ConnectionState connectionState = dBmsqlSub.getState();
Console.WriteLine(connectionState);
dBmsqlSub.Close();
}
}
}結(jié)果,整理使用后的鏈接狀態(tài)是open,那是不是open的時候就可以供別人使用了呢,要是在執(zhí)行的時候也是open狀態(tài),那我就沒有判定已經(jīng)了。
數(shù)據(jù)庫鏈接狀態(tài)確認 id0 index:1 isuser:False setuser:False Open
2.2 數(shù)據(jù)庫在執(zhí)行的時候,是不是有一個執(zhí)行中的狀態(tài)呢?
代碼
using System;
using System.Threading.Tasks;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
/// <summary>
/// 目的:確認執(zhí)行中的數(shù)據(jù)狀態(tài)
/// </summary>
class Test2:Singleton<Test2>
{
public void main() {
DBmsqlSub dBmsqlSub = new DBmsqlSub();
Task.Factory.StartNew(test2Exe, (Object)dBmsqlSub);
Task.Factory.StartNew(test2Exe, (Object)dBmsqlSub);
Task.Factory.StartNew(test2State, dBmsqlSub);
}
private void test2Exe(object dBmsqlSub)
{
int index = 1;
while (true)
{
index++;
string sql = "insert into tb1(v1) values(" + index + ")";
((DBmsqlSub)dBmsqlSub).ExecuteNonQuery(sql);
Task.Delay(200);
}
}
private void test2State(object dBmsqlSub)
{
while (true)
{
Console.WriteLine(((DBmsqlSub)dBmsqlSub).getState());
}
}
}
}行結(jié)果執(zhí):即使執(zhí)行中,對象的鏈接狀態(tài)依然是open,那么用鏈接狀態(tài)作為鏈接是否可用的計劃泡湯了,那就只能用自添加的對象來控制鏈接是否可用了。
Open
Open
Open
Open
id0
Open
Open
Open
Open
Open
Open
2.3 測試3:添加對象的狀態(tài),控制鏈接是否可用
2.3.1 代碼
using MySqlConnector;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
class Test3:Singleton<Test3>
{
public void main() {
test1();
//test1();
}
private void test1() {
Task.Factory.StartNew(test2Exe);
Task.Factory.StartNew(test2Exe);
Task.Factory.StartNew(test2Exe);
}
private void test2Exe()
{
int index = 1;
while (true)
{
index++;
string sql = "insert into tb1(v1) values(" + index + ")";
DBmsql.getMy().ExecuteNonQuery(sql);
}
}
private object MySqlDataReader(DBmsqlSub dBmsqlSub)
{
throw new NotImplementedException();
}
private void test2State(object dBmsqlSub)
{
while (true)
{
Console.WriteLine(((DBmsqlSub)dBmsqlSub).getState());
}
}
}
}2.3.2
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
class DBmsql : Singleton<DBmsql>
{
List<DBmsqlSub> dBmsqlSubs = new List<DBmsqlSub>();
/// <summary>
/// 執(zhí)行sql命令
/// </summary>
/// <param name="CommandText"></param>
public void ExecuteNonQuery(String CommandText)
{
getDBmsqlSub().ExecuteNonQuery(CommandText);
}
/// <summary>
/// 插入數(shù)據(jù),并返回插入數(shù)據(jù)的id
/// </summary>
/// <param name="CommandText"></param>
/// <returns></returns>
public int insertReturn(string CommandText)
{
int ret = getDBmsqlSub().insertReturn(CommandText);
return ret;
}
/// <summary>
/// 執(zhí)行并返回一個對象
/// </summary>
/// <param name="CommandText"></param>
/// <returns></returns>
public object ExecuteScalar(string CommandText)
{
object o = getDBmsqlSub().ExecuteScalar(CommandText);
return o;
}
/// <summary>
/// 獲取數(shù)據(jù)處理對象
/// </summary>
/// <returns></returns>
private DBmsqlSub getDBmsqlSub()
{
DBmsqlSub ret = null;
lock (dBmsqlSubs)
{
//避免兩個同時取到允許的狀態(tài)
foreach (DBmsqlSub dBmsqlSub in dBmsqlSubs)
{
if (!dBmsqlSub.IsUrse())
{
ret = dBmsqlSub;
dBmsqlSub.setIsUser(true);
Console.WriteLine("get:" + ret.id);
break;
}
}
//避免兩個同時創(chuàng)建對象,產(chǎn)生結(jié)果列表的錯誤
if (ret == null&& dBmsqlSubs.Count<90)
{
DBmsqlSub dBmsqlSub = new DBmsqlSub();
dBmsqlSubs.Add(dBmsqlSub);
dBmsqlSub.setIsUser(true);
ret = dBmsqlSub;
Console.WriteLine("get:" + ret.id);
}
}
return ret;
}
}
}2.3.3
using MySqlConnector;
using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
public class Constand {
public static string mesConnStr = "server=localhost;port=3306;database=db1;user id=root;password=123456;Charset=utf8;";
}
class DBmsqlSub
{
static int index = 0;
public int id = 0;
private bool isUser = false;
private static NLog.Logger logger = NLog.LogManager.GetCurrentClassLogger();
MySqlConnection mConn;
MySqlCommand mCmd;
public void setIsUser(bool value) {
lock (this) {
Console.WriteLine("index:" + index + " isuser:" + isUser+" setuser:"+ value);
isUser = value;
}
}
public MySqlDataReader MySqlDataReader { get; private set; }
public DBmsqlSub()
{
id = index++;
try
{
mConn = new MySqlConnection(Constand.mesConnStr);
mConn.Open();
mCmd = new MySqlCommand();
mCmd.Connection = mConn;
}
catch (Exception e)
{
logger.Error(e.ToString());
}
}
~DBmsqlSub()
{
mConn.Close();
}
public void Close()
{
mConn.Close();
}
public bool isOpen()
{
if (mConn.State == ConnectionState.Closed)
{
mConn.Open();
}
if (mConn.State == ConnectionState.Open)
{
return true;
}
else
{
return false;
}
}
public MySqlCommand getCmd()
{
return mCmd;
}
/// <summary>
/// 如果沒有鏈接,就直接鏈接
/// </summary>
private void conn()
{
if (mConn.State != ConnectionState.Open)
{
mConn.Open();
}
}
/// <summary>
/// 執(zhí)行sql命令
/// </summary>
/// <param name="CommandText"></param>
public void ExecuteNonQuery(String CommandText)
{
//setIsUser(true);
mCmd.CommandText = CommandText;
try
{
Console.WriteLine("id"+id);
mCmd.ExecuteNonQuery();
}
catch (Exception ex)
{
logger.Error(ex.ToString());
}
finally {
setIsUser(false);
}
}
/// <summary>
/// 插入數(shù)據(jù),并返回插入數(shù)據(jù)的id
/// </summary>
/// <param name="CommandText"></param>
/// <returns></returns>
public int insertReturn(string CommandText)
{
setIsUser(true);
int ret = 0;
MySqlTransaction sqlTransaction = mConn.BeginTransaction();
try
{
mCmd.CommandText = CommandText;
object o = mCmd.ExecuteScalar();
sqlTransaction.Commit();
ret = int.Parse(o.ToString());
}
catch (Exception e)
{
logger.Error(e.ToString());
sqlTransaction.Rollback();
}
finally
{
setIsUser(false);
}
return ret;
}
/// <summary>
/// 執(zhí)行并返回一個對象
/// </summary>
/// <param name="CommandText"></param>
/// <returns></returns>
public object ExecuteScalar(string CommandText)
{
setIsUser(true);
object o = null;
mCmd.CommandText = CommandText;
try
{
o = mCmd.ExecuteScalar();
}
catch (Exception ex)
{
logger.Error(ex.ToString());
}
finally
{
setIsUser(false);
}
return o;
}
public MySqlDataReader ExecuteReader(string CommandText)
{
setIsUser(true);
MySqlDataReader mySqlDataReader = null;
mCmd.CommandText = CommandText;
try
{
mySqlDataReader = mCmd.ExecuteReader();
//mConn.Close();
}
catch (Exception ex)
{
logger.Error(ex.ToString());
}
finally
{
setIsUser(false);
}
return mySqlDataReader;
}
public ConnectionState getState() {
return mConn.State;
}
public bool IsUrse() {
return isUser;
}
}
}2.3.4
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace 數(shù)據(jù)庫鏈接狀態(tài)確認
{
/// <summary>
/// 單件構(gòu)象基類
/// </summary>
/// <typeparam name="T"></typeparam>
public class Singleton<T> where T : new()
{
static T t = default(T);
public static T getMy()
{
if (t == null)
{
t = new T();
}
return t;
}
}
}運行結(jié)果:可用看出是幾個鏈接對象在被循環(huán)的使用,也基本達到了直接的初衷,用盡可能少的鏈接,完成多線程的調(diào)用情景。
id2 index:5 isuser:True setuser:False index:5 isuser:False setuser:True index:5 isuser:True setuser:False get:3 id3 index:5 isuser:False setuser:True get:4 id4 index:5 isuser:True setuser:False index:5 isuser:False setuser:True get:2 id2 index:5 isuser:True setuser:False index:5 isuser:False setuser:True get:3 id3
到此這篇關(guān)于mysql 數(shù)據(jù)庫鏈接狀態(tài)確認實驗的文章就介紹到這了,更多相關(guān)mysql 數(shù)據(jù)庫鏈接狀態(tài)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Windows10下mysql 8.0.19 winx64安裝教程及修改初始密碼
這篇文章主要為大家詳細介紹了Windows10下mysql 8.0.19 winx64安裝教程及修改初始密碼,文中安裝步驟介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-02-02
MySQL數(shù)據(jù)更新操作的兩種辦法(數(shù)據(jù)可視化工具和SQL語句)
MySQL是最常用的數(shù)據(jù)庫,在數(shù)據(jù)庫操作中,基本都是增刪改查操作,簡稱CRUD,下面這篇文章主要給大家介紹了關(guān)于MySQL數(shù)據(jù)更新操作的兩種辦法,需要的朋友可以參考下2023-03-03
MySQL之導(dǎo)出整個及單個表數(shù)據(jù)的操作
這篇文章主要介紹了MySQL之導(dǎo)出整個及單個表數(shù)據(jù)的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-11-11
結(jié)合PHP腳本添加和查詢MySQL數(shù)據(jù)的基本教程
這篇文章主要介紹了結(jié)合PHP腳本添加和查詢MySQL數(shù)據(jù)的基本教程,即在PHP程序中使用基本的SELECT FROM和INSERT INTO語句,需要的朋友可以參考下2015-12-12
MySQL服務(wù)啟動與關(guān)閉如何操作圖文詳解
這篇文章主要為大家介紹了MySQL服務(wù)啟動與關(guān)閉如何操作圖文詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪<BR>2023-10-10

