(数据库的一览,有图有真相)
第一步 解决读取的问题
跟数据库打交道的方式有很多,我来列举下吧:
1. 【重武器-坦克大炮】使用重型ORM框架,比如EF,NHibernat 这样的框架。
2. 【轻武器-AK47】 使用Dapper,PetaPoco之类,单cs文件。灵活高效,使用简单。居家越货必备(我更喜欢PetaPoco :))
3. 【冷兵器?匕首?】使用原生的Connection、Command。 然后写原生的SQL语句。。
分析:
【重武器】在我们这里肯定直接被PASS, 他们应该被用在大型项目中。
【轻武器】Dapper,PetaPoco 看过源码你会发现用到了反射,虽然使用IL和缓存技术,但是还是会影响读取效率,PASS
好吧那就只有使用匕首,原生SQL走起, 利用DataReader 进行高效读取,并且使用索引取数据(更快),而不是列名。
大概的代码如下:
using (var conn = new MySqlConnection("Connection String...")){conn.Open();//此处设置读取的超时,不然在海量数据时很容易超时var c = new MySqlCommand("set net_write_timeout=9999999; set net_read_timeout=9999999", conn);c.ExecuteNonQuery();MySqlCommand rcmd = new MySqlCommand();rcmd.Connection = conn;rcmd.CommandText = @"SELECT `f1`,`f2` FROM `table1`";//设置命令的执行超时rcmd.CommandTimeout = 99999999;var myData = rcmd.ExecuteReader();while (myData.Read()){var f1= myData.GetInt32(0);var f2= myData.GetString(1);//这里做数据处理....}}哈哈,怎么样,代码非常原始,还是使用索引来取数据,很容易出错。 当然一切为了性能咱都忍了
insert into table (f1,f2) values(1,"sss"),values(2,"bbbb"),values(3,"cccc");就是把values后面的全部用逗号,链接起来,然后一次性执行 。
//使用StringBuilder高效拼接字符串 var sqlBuilder = new StringBuilder(); //添加insert 语句的头 string sqlHeader = "insert into table1 (`f1`,`f2`) values"; sqlBuilder.Append(sqlHeader); using (var conn = new MySqlConnection("Connection String...")) { conn.Open(); //此处设置读取的超时,不然在海量数据时很容易超时 var c = new MySqlCommand("set net_write_timeout=9999999; set net_read_timeout=9999999", conn); c.ExecuteNonQuery(); MySqlCommand rcmd = new MySqlCommand(); rcmd.Connection = conn; rcmd.CommandText = @"SELECT `f1`,`f2` FROM `table1`"; //设置命令的执行超时 rcmd.CommandTimeout = 99999999; var myData = rcmd.ExecuteReader(); while (myData.Read()) { var f1 = myData.GetInt32(0); var f2 = myData.GetString(1); //这里做数据处理.... sqlBuilder.AppendFormat("({0},"{1}"),", f1,AddSlash(f2)); if (sqlBuilder.Length >= 1024 * 1024)//当然这里的1MB length的字符串并不等于 1MB的Packet。。。我知道:) { insertCmd.Execute(sqlBuilder.Remove(sqlBuilder.Length-1,1).ToString())//移除逗号,然后执行 sqlBuilder.Clear();//清空 sqlBuilder.Append(sqlHeader);//在加上insert 头 } }}好了,到这里 大概的优化后的高效查询、插入就完成了。