ADO.Net基础

一、ADO.Net简介
  1,程序要通过SQL语句自动化的操作数据库,必须要用一个类库,
    类库要提供execute("insert into ...")/executeQuery("select * from ...")类似的方法

  2,ADO.Net是.Net中提供的标准访问数据库的接口,访问不同的DBMS的底层方法是不一样的,ADO.Net把访问数据库的方法进行了统一,
    访问MYSql、Oracle、SqlServer等不同数据库的方法几乎是一样的

  3,ADO.Net是规范,被不同的数据库厂商提供ADO.Net的实现,称之为ADO.Net驱动,每个厂商提供的驱动可以用来操作自己的数据库

二、ADO.Net连接MYSQL

  1,安装MySql的.Net驱动mysql-connector-net-***.msi添加到项目的库中。
    如果安装遇到问题,则直接下载mysqlnetconnection(V4.5).zip。

  2,新建项目,添加引用——“扩展”,添加Mysql.Data;如果是直接解压版,然后直接添加对MySql.Data.dll的文件的引用
    static void Main(string[] args)
    {
      string connStr = ConfigurationManager.ConnectionStrings["connStr"].ToString();
      using (SqlConnection conn = new SqlConnection(connStr))
      using (SqlCommand cmd = conn.CreateCommand())
      {
        conn.Open();
        cmd.CommandText = "insert into t_fuxi ";
        int count = cmd.ExecuteNonQuery();
        Console.WriteLine(count+"受到影响");
        Console.ReadKey();
      }
    }

    代码解释:
    1,MySQLConnection、MySQLCommend实现了IDisposable接口,因此使用using进行资源回收

    2,"Server=localhost;Database=study1;uid=root;pwd=root;Charset=utf8"叫连接字符串,
      Server是Mysql服务器的地址,Database是连接的数据库,uid、pwd是用户名和密码,采用utf8编码

    3,conn.Open();在执行MySqlCommand之前一定要先打开数据库连接,否则会报错

    4,ExecuteNoQuery是执行Update、Insert、Delete等非查询语句,返回值为受影响的行数

  3,ExecuteScalar
    执行查询,并返回查询所返回的结果集中第一行的第一列,忽略其他行列。一般用来简单的获得自由一行一列的查询结果的值

    案例1
    cmd.CommandText = "Select count(*) from T_Users";
    long count = (long)cmd.ExecuteScalar();

    案例2
    cmd.CommandText = "Select Password from T_Users where UserName = 'admin'";
    string pwd = (string)cmd.ExecuteScalar();
    if(string.isNullOrEmpty(pwd))
    {
      Console.WriteLine("找不到admin");
    }
    else
    {
      Console.WriteLine("admin的密码:"+pwd);
    }

  4,执行查询 MySqlDataReader

    cmd.CommandText = "select * from T_Users";
    using(MySqlDataReader reader = cmd.ExecuteReader())
    {
      while(reader.Read())
      {
        long id = reader.GetInt64("Id");
        string userName = reader.GetString("UserName");
        string passWord = reader.GetString("Password");
        Console.WriteLine("id="+id+";UserName="+userName+";Password="+passWord);
      }
    }

    注意:
      Reader的遍历、读取时需要Connection保持连接,如果关闭了Connection,使用会出错
      也可以根据列序号获取列的值,效率略高,不过程序不容易读;通过reeder.GetOrdinal("Age")获取列名对应的序列号

三、SQL注入漏洞

  1,是由于Sql语句的拼接造成的,
    在程序接收用户的输入时,需要考虑用户输入的内容对数据库操作产生的影响,

  2,对应的解决方法,参数化查询

    cmd.CommandText = "select count(*) from TUsers where username=@username and password=@password";
    cmd.Parameters.Add(new SqlParameter() { ParameterName = "@UserName", Value = username });
    cmd.Parameters.Add(new SqlParameter() { ParameterName = "@Password", Value = password });

    为什么这样可以避免“SQL注入漏洞”,
    因此使用参数化查询,就可以对用户输入的内容进行判断和处理了,本质上是参数赋值

    注意:
    1,所有SQL中都可以使用参数化查询传递;表名、字段名等不能使用参数化进行替换
    2,不要用SqlParameter(string parameterName,object value)这个构造函数,
    因为("Age",0)会被匹配成Sqlparameter(string parameterName,SqlDbType dbType)这个构造函数

四、基本数据类型为空的问题

  1,把T_Persons表的Name、Age列修改为“允许为空”,插入一条Name、Age为空的行
    执行后发现,对于空数据reader.Get**方法会抛异常SqlNullValueException,
    相当于问数据库“当前行的Name”是什么,数据库告诉你“不知道”。

    怎么解决?
    使用ISDBNull获取指定序号的列的值是否为null

    int? age = null;
    if(!reader.IsDBNull(reader.GetOrdinal("Age")))
    {
      age = reader.GetInt32("Age");
    }

五、离线结果集DataSet

  DATAReader是服务器结果集游标的体现,所有查询出来的数据都在MySql服务器上。
  好处是:当查询结果数据量打的时候避免占用本地内存。

  不过大部分项目中都会避免大查询结果,因此缺点就明显了
  读取的时候必须保持Connection,不仅用起来麻烦,而且会较长时间占用MySql服务器的连接资源

  DataSet是一个离线结果集容器,它把结果数据放到本地内存中。
  因为查询结果可能会包含多个表,因此DataSet包含若干DataTable(ds.Tables),
  DataTable包含若干DataRow(dt.Rows)

  用法1:
  DataSet ds = new DataSet();
  MySqlDataAdapter adapter = new MySqlDataAdapter(cmd);
  adapter.Fill(ds);
  DataTable table = ds.Tables[0];

六、DataTable
  DataSet可以盛放多个查询结果集到DataTable

  DataAdapter还可以对结果进行傻瓜化更新、删除、修改。

  我们一般查询结果集就一个DataTable,DataAdapter的傻瓜化更新不适合正式的项目,因此有更简单的用法

  DataTable dt = new DataTable();
  dt.Load(reader);

  注意: 把DataTable声明到using外,using外再使用查询结果

  遍历DataTable

  for(int i =0;i<dt.Rows.Count;i++)
  {
    DataRow row = dt.Rows[i];
    string name = row.IsNull("name")?null:(string)row["Name"];//NULL处理
    Console.WriteLine("name="+name);
  }

七、封装一个库:分析

  1,ADO.Net的连接字符串写到配置文件中。

  2,每次操作数据库都要写一坨代码,太累,因此封装一个简化ADO.Net操作的库出来:
    配置文件的设置连接字符串;简化连接的创建;简化SQL的执行

  3,如果一个操作要执行多条SQL语句,如果每条都打开一个连接——执行——关闭连接的话,效率会非常低,而且会有“事务”的问题。
    因此应该提供“打开、执行、关闭”这样的方法,也要提供“使用后现有连接执行的方法”。

  4,参数化查询的查询参数个数不确定,可变长度参数会更方便

  5,为了方便大部分情况下的小结果集,执行查询返回DataTable

八、MySqlHelper方法规划

  1,public static MySqlConnection CreateConnection()

  2,public static int ExecuteNonQuery(MySqlConnection conn,string sql,params MySqlParameter[] parameters)

  3,public static int ExecuteNonQuery(string sql,params MySqlParameter[] parameters)

  4,public static object ExecuteScalar(MySqlConnection conn,string sql,params MySqlParameter[] parameters)

  5,public static object ExecuteScalar(string sql,params MySqlParameter[] parameters)

  6,public static DataTable ExecuteQuery(MySqlConnection conn, string sql, params MySqlParameter[] parameters)

  7,public static DataTable ExecuteQuery(string sql, params MySqlParameter[] parameters)

九、实现MySqlHelper

  1,连接字符串一般配置到App.config(网站是Web.config)中的<connectionStrings>段中
    然后使用ConfigurationManager类(需要添加对System.Configuration的引用)读取
    string connStr = ConfigurationManager.ConnectionStrings["connStr"].ConnectionString
    

    注意:
    一定要保证代码中的名字和配置文件中的名字是一致的

  2,其他方法的实现

  3,测试几个方法,调用不需要自己控制连接的,再调用公用一个连接的

十、获得自动增长字段的值

  1,不能用插入后获取最大值的方法,有并发问题

  2,要在同一个连接中:select LAST_INSERT_ID()

  3,可以Insert、LAST_INSERT_ID()在同一个连接中单独执行,也可以把LAST_INSERT_ID()放到insert语句后面用;
    分割(使用ExecuteScalar执行即可)

十一、事务基础

  1,有一个需求,类似于转账,从Tom的工资上转走10元,转到Jerry的工资上增加10元

    Update T_Employees Set Salary = Salary - 10 where Name = 'Tom'
    Update T_Employees Set Salary = Salary + 10 where Name = 'Jerry'

    如果执行从Jerry账上加10元的时候执行出错(使用SQL语法写错误模拟),那么就会出现总体丢失10元的问题,如果是转账呢?

  2,事务(Transaction)有四大特征:
    原子性、一致性、隔离性、持久性
    原子性指的是:几个操作要么都成功,要么都失败

十二、ADO.Net事务

  1,要在一个连接中(否则要涉及到分布式事务)
    MySqlTransaction tx = conn.BeginTransaction();
    操作结束后执行tx.Commit()提交事务;
    如果执行出错,则tx.Rollback()回滚(当前事务的操作全部取消)

    示例代码:
    MySqlTransaction tx = conn.BeginTransaction();
    try
    {
      MySqlHelper.ExecuteNonQuery(conn,"Update T_Accounts set Amount = Amount-1000 where Number ='0001'");
      string s = null;
      s.ToLower();//制造异常
      MySqlHelper.ExecuteNonQuery(conn, "Update t_accounts Set Amount=Amount+1000 where Number='0002'");
      tx.Commit();
    }
    catch(Exception ex)
    {
      tx.Rollback();
    }

    事务还有隔离级别、嵌套事务等问题

十三、SQLServer的使用

  1,安装,版本:2008有兼容性的问题,有bug、因此推荐安装SQLServer 2008 R2

  2,Management Studio的使用

    SQLServer的两种连接方式:
    Windows 身份验证(互相信任的局域网中);
    SQLServer身份验证(使用SQLServer用户名密码验证,密码要复杂一点)。
    Windows 身份验证还有一个用途:忘了sa密码,可以本机进去改。

  3,新建数据库、新建表
    SQLServer的数据类型
    (varchar和nvarchar;nvarchar(n)和nvarchar(MAX);
    long是bigint;
    获取前10条数据: select top 10 * from t_persons )、
    SQLServer的自动增长(是标识)、不需要特殊指定编码

  4,保存表设计修改的时候,如果报错“不允许保存更改”:
    工具→选项→Designers→把“阻止保存要求重新创建表的更高”勾选掉。
    遇到报错信息:要先仔细阅读。

  5,执行SQL语句(数据库上点右键“新建查询”,不要点【调试】)

十四、ADO.Net连接SQLServer
  1,ADO.Net如何连接SQLServer:
    SQLServer驱动内置

  2,连接字符串:
    server=ip;user id =sa;passWord = 密码;database = db1

  3,SQLHelper:
    把MySql替换成Sql就可以了

  4,获得自动增长列的值:
    Insert into t1(...) output insert.Id values(....)

  5,如果基于接口编程,只要改动CreateConnection就可以了,查询参数以Directory<string,object>传递
    如果使用Provider,连代码都不用改,改配置文件即可

如鹏网学习笔记(六)ADO.Net基础的更多相关文章

  1. 如鹏网学习笔记(十五)ASP.NET MVC核心基础笔记

    一.ASP.Net MVC简介 1,什么是ASP.NET MVC? HttpHandler是ASP.net的底层机制,如果直接使用HttpHandler进行开发难度比较大.工作量大.因此提供了ASP. ...

  2. 如鹏网学习笔记(七)HTML基础

    HTML笔记 一.HTML简介 1,HTML (Hyper Text Mark-up Language) 超文本标记语言,是一种编程语言,也可以说是一种标准.规范. 2,HTML提供了一系列标记(标签 ...

  3. 如鹏网学习笔记(四).Net常用类库

    .Net常用类库 一.String成员方法(常用) 1,bool Contains(string str) 判断字符串对象是否包含给定的内容 2,bool StartsWith(String str) ...

  4. 如鹏网学习笔记(十)DOM

    DOM笔记一.DOM简介 Document Object Model 文档对象模型 DOM的节点树模型:整个文档按照从大到小的节点划分,每一个内容都算作一个节点 DOM API 编程接口 可以用来操作 ...

  5. 如鹏网学习笔记(九)JavaScript

    JavaScript笔记 一.JavaScript简介 1,JavaScript是一种计算机编程语言,可以像等其他编程语言那样定义变量,执行循环等. 2,JavaScript代码主要执行在浏览器上,为 ...

  6. 如鹏网学习笔记(十四)ASP.NET

    Asp.net笔记 一.Socket类 进行网络编程的类,可以在两台计算机之间进行网络通讯 过程: 向服务器发送指令: GET /index.html HTTP/1.1 Host:127.0.0.1: ...

  7. 如鹏网学习笔记(十三)EasyUI

    一.EasyUI简介 是一组基于JQuery的UI插件集合 主要作用:为JQuery对象提供新的方法,实现新的功能 可以快速创建出简洁.友好.美观的页面,非常适合做网站后台管理页面(不够漂亮,不适合做 ...

  8. 如鹏网学习笔记(五)MySql基础

    MySQL基础 一.数据库概念 1,网友装备信息.论坛帖子信息.QQ好友关系信息.学籍管理系统中的学生信息等都要“持久化”的保存到一个地方, 如果通过IO写到文件中,那么会非常麻烦,而且不利于多人共享 ...

  9. 如鹏网学习笔记(十二)HTML5

    一.HTML5简介 HTML5是HTML语言第五次修改产生的新的HTML语言版本 改进主要包括: 增加新的HTML标签或者属性.新的CSS样式属性.新的JavaScript API等.同时删除了一些过 ...

随机推荐

  1. docker 存储驱动之 overlay2

    overlay2 简介 OverlayFS是一种和AUFS很类似的文件系统,与AUFS相比,OverlayFS有以下特性:  1) 更简单地设计 2) 从3.18开始,就进入了Linux内核主线 3) ...

  2. python--函数名的使用,闭包,迭代器

    1.函数名的使用和第一类对象 函数名是一个变量,但它是一个特殊的变量,与括号配合可以执行函数 函数对象可以像变量一样进行赋值,还可以作为列表的元素进行使用,可以作为返回值返回,可以作为参数进行传递 1 ...

  3. Python----初次见面,请多关照!

    1.计算机的最基本认识 CPU(大脑) 3GHZ + 内存(DDR4) + 主板 + 电源(心脏)+ 显示器 + 键盘 +鼠标+ 显卡 + 硬盘 80MB/s 操作系统分为: windows 家用 l ...

  4. kali linux之无线渗透

    无线技术变化大,难度大,既新鲜刺激,又压力山大.一半协议  一半理论 无线技术特点: 行业发展迅猛 互联网的重要入口 边界模糊 安全实施缺失而且困难 对技术不了解造成配置不当 企业网络私自接入ap破坏 ...

  5. django_jquery中使用ajax发送post请求变成get请求

    今天在进行js开发的过程中出现了一个奇怪的问题,就是使用ajax向后端发送post请求时,在浏览器network中查看response时,显示400 bad request 并且请求方式变成get,因 ...

  6. [HTML] <meta name="viewport" content="width=device-width,initial-scale=1.0">释义

    <meta name="viewport" content="width=device-width,initial-scale=1.0">这是 HT ...

  7. 爬虫2:urllib

        了解即可,不好用   一. 概述   python内置的http请求库,包括4个模块,分别如下   urllib.request   请求模块 urllib.error       异常处理模 ...

  8. 为什么说 Gumroad 是一家 “失败” 的创业公司?

    Gumroad 是一家 "失败" 的创业公司. 创立于 2012 年,Gumroad 是一个面向创造者的电商平台.创始人 Sahil Lavingia,一名 19 岁的少年,Pin ...

  9. java项目迁移

    电脑重装系统以后或者从不同MyEclipse版本迁移项目时候会出现: Project facet Java 1.5 is not supported by target runtime Apache ...

  10. IE下判断IE版本的语句...[if lte IE 6]……[endif](用户判断IE版本的如果小于6就显示)

    <!--[if lte IE 6]> <![endif]--> IE6及其以下版本可见   <!--[if lte IE 7]> <![endif]--> ...