Mybatis.net与MVC入门配置及联合查询动态SQL拼接和简单事务

第一次学习Mybatis.net,在博客园也找到好多资料,但是在配置成功之后也遇到了一些问题,尤其是在动态SQl拼接时候,这里把遇到的问题还有自己写的一个Demo贴出来,希望能帮到新手,有不适合的地方也希望大家指出,大家互相学习,共同进步。

首先我们需要下载IBatisNet程序包,这里就不多说了

一、创建一个MVC4的Web应用程序,并引用我们下载的IBatisNet程序包中的IBatisNet.DataMapper.dll,这里我只引用了这一个,log4net我没有引用也没有配置使用,全部程序结构如下

Mybatis.net与MVC入门配置及联合查询动态SQL拼接和简单事务

二、添加Providers.config,并复制到项目根目录下

把下载的IBatisNet程序包打开,就能找到providers.config文件,里面定义了MyBatis.Net支持的各种数据库驱动,这里我以sqlServer为例,把其他不用的全部删除,只留下sqlServer的并且把版本号全部改为4.0,同时把enabled属性设置成true,如下:

<?xml version="1.0" encoding="utf-8"?>

<providers 

xmlns="http://ibatis.apache.org/providers" 

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">



<clear/>

  <!--设置enabled=true-->

  <provider

     name="sqlServer4.0"

     enabled="true"

     description="Microsoft SQL Server, provider V4.0.0.0 in framework .NET V4.0" 

     assemblyName="System.Data, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" 

     connectionClass="System.Data.SqlClient.SqlConnection" 

     commandClass="System.Data.SqlClient.SqlCommand"

     parameterClass="System.Data.SqlClient.SqlParameter"

     parameterDbTypeClass="System.Data.SqlDbType"

     parameterDbTypeProperty="SqlDbType"

     dataAdapterClass="System.Data.SqlClient.SqlDataAdapter"

     commandBuilderClass=" System.Data.SqlClient.SqlCommandBuilder"

     usePositionalParameters = "false"

     useParameterPrefixInSql = "true"

     useParameterPrefixInParameter = "true" 

     parameterPrefix="@"

     allowMARS="false"

    /> 

</providers>
providers.config

 三、在项目目录下添加SqlMap.config,它的作用主要是指定db连接串,告诉系统providers.config在哪? 以及db与entity的映射文件在哪?如下:

<?xml version="1.0" encoding="utf-8"?>

<sqlMapConfig 

  xmlns="http://ibatis.apache.org/dataMapper" 

  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">

  

  <settings>

    <setting useStatementNamespaces="false"/> <!--true表示statementName要使用Namespace-->

    <setting cacheModelsEnabled="true"/>

  </settings>

  

  <providers resource="providers.config"/><!--指定providers.config文件的路径-->

  

    <!-- Database connection information -->

    <database>

        <provider name="sqlServer4.0"/><!--设置数据库连接字符串-->

        <dataSource name="DB" connectionString="Data Source=.;Initial Catalog=DemoDB;uid=sa;pwd=123"/>

    </database>

  

  <!--db与Entity的映射文件-->

    <sqlMaps>

    <sqlMap resource="Maps/PersonAndCour.xml"/>

     <sqlMap resource="Maps/PersonMap.xml"/>

    </sqlMaps>



</sqlMapConfig>
SqlMap.config

四、这里在数据库建两张表一个是个人信息还有一个是课程表,如下:

Mybatis.net与MVC入门配置及联合查询动态SQL拼接和简单事务

第一张表是Person表,第二张表为Cour课程表,课程表名字是错的只敲一半,大家略过,第一张表的ID和第二张表的PerID关联,建两张表是为了后面的联合查询,这里先不多说

五、接下来我们在EntityModel文件夹下建立两个实体类一个是Person类和一个定义实体类PersonAndCour,而自定义实体类就比Person类多一个字段,就是课程名称,这里就只贴这一个代码 ,如下:

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;



namespace MyBatisNet.EntityModel

{

    public class PersonAndCour

    {

        public int ID { get; set; }

        public string Name { get; set; }

        public int Age { get; set; }

        public string Sex { get; set; }

        public string CourseName { get; set; }

    }

}
Model

六、我们在项目根目录下建Maps文件夹,并在文件夹中创建PersonMap.xml映射文件,如下

<?xml version="1.0" encoding="utf-8" ?>

<sqlMap namespace="MyBatisNet" xmlns="http://ibatis.apache.org/mapping"

         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">



  <alias>

    <!--类的别名-->

    <typeAlias alias="Person" type="MyBatisNet.EntityModel.Person,MyBatisNet"/>

  </alias>



  <resultMaps>

    <!--Person类与db表的映射-->

    <resultMap id="PersonResult" class="Person">

      <result property="ID" column="ID"/>

      <result property="Name" column="Name"/>

      <result property="Age" column="Age" />

      <result property="Sex" column="Sex"  />

    </resultMap>

  </resultMaps>



  

  <statements>

    <!--插入Sql语句-->

    <insert id="Add" parameterClass="Person" resultClass="Int32">

      insert into Person(Name,Age,Sex)values(#Name#,#Age#,#Sex#)

      <selectKey property="ID" resultClass="int" type="post" >

        SELECT @@identity AS ID

      </selectKey>

     

    </insert>

    <!--修改Sql语句-->

    <update id="Update" parameterClass="Person">

      update Person set Name=#Name#,Age=#Age#,Sex=#Sex# where ID=#ID#

    </update>



    <!--根据主键删除单条记录-->

    <delete id="Delete" parameterClass="Int32">

      delete Person where ID=#ID#

    </delete>



    <!--查询单个实体Sql语句-->

    <select id="Get" parameterClass="Int32" resultMap="PersonResult">

      select * from Person where ID=#ID#

    </select>



    <!--查询所有记录-->

    <select id="GetList" resultMap="PersonResult">

    <![CDATA[select * from Person where ID<4]]>  

    </select>

  </statements>

</sqlMap>
PersonMap.xml

我们可以看到这段代码里有一个命名空间,如下:

这个命名空间与此文件中的 statements标签中配置的SQL语句的id有关,在大型项目中,可能存在大量的 SQL 语句,这时候,为每个SQL 语句起一个唯一的标识id 就变得并不容易了。为了解决这个问题,在 mybatis 中,可以为每个映射文件起一个唯一的命名空间,这样,定义在这个映射文件中的每个 SQL 语句就成了定义在这个命名空间中的一个 id。只要我们能够保证每个命名空间是唯一的,即使在不同映射文件中的语句的 id 相同,也就不会冲突了。(这段是网上复制的),这个是在SqlMap.config配置文件中控制的,就是这个标签 <setting useStatementNamespaces="false"/>。

七、接下来我们就要在SqlMap.config文件中配置这个映射文件的路径了,上面的代码里面我们已经配好了,

八、在XML映射文件中我们已经写好的基础的增、删、改、查SQL语句,接下来我们在项目中建一个Common文件夹,里面写一个通用的BaseDA类,里面进行ISqlMapper的实例化,以及对MyBatis.Net做些基本的封装(这个是网上复制的)

using IBatisNet.DataMapper;

using IBatisNet.DataMapper.Configuration;

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;



namespace MyBatisNet.Common

{

    public class BaseDA

    {

        public static int Insert<T>(string statementName, T t)

        {

            ISqlMapper iSqlMapper = Mapper.Instance();

            if (iSqlMapper != null)

            {

                return (int)iSqlMapper.Insert(statementName, t);

            }

            return 0;

        }



        public static int Update<T>(string statementName, T t)

        {

            ISqlMapper iSqlMapper = Mapper.Instance();

            if (iSqlMapper != null)

            {

                return iSqlMapper.Update(statementName, t);

            }

            return 0;

        }



        public static int Delete(string statementName, int primaryKeyId)

        {

            ISqlMapper iSqlMapper = Mapper.Instance();

            if (iSqlMapper != null)

            {

                return iSqlMapper.Delete(statementName, primaryKeyId);

            }

            return 0;

        }



        public static T Get<T>(string statementName, int primaryKeyId) where T : class

        {

            ISqlMapper iSqlMapper = Mapper.Instance();

            if (iSqlMapper != null)

            {

                return iSqlMapper.QueryForObject<T>(statementName, primaryKeyId);

            }

            return null;

        }



        public static IList<T> QueryForList<T>(string statementName, object parameterObject = null)

        {

            ISqlMapper iSqlMapper = Mapper.Instance();

            if (iSqlMapper != null)

            {

                return iSqlMapper.QueryForList<T>(statementName, parameterObject);

            }

            return null;

        }

    }

}
BaseDA.cs

九、现在我们就可以控制器中进行操作了

这样我们就可以查询到Person类的全部信息了,这里方法中的第一个参数就是我们映射文件中SQL语句标签中的id名称,其他的增、删、改都一样,就不多说了

十,接下来我们看看我们多表联合查询,动态SQl拼接,以及SQL片段的使用,我们这里用到上面的PersonAndCour.cs这个自定义类

我们先添加PersonAndCour.xml这个映射文件

<?xml version="1.0" encoding="utf-8" ?>

<sqlMap namespace="MyBatisNet" xmlns="http://ibatis.apache.org/mapping"

         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">



  <alias>

    <!--类的别名-->

    <typeAlias alias="PersonAndCour" type="MyBatisNet.EntityModel.PersonAndCour,MyBatisNet"/>

  </alias>



  <resultMaps>

    <!--Product类与db表的映射-->

    <resultMap id="PersonAndCourResult" class="PersonAndCour">

      <result property="ID" column="ID"/>

      <result property="Name" column="Name"/>

      <result property="Age" column="Age" />

      <result property="Sex" column="Sex"  />

      <result property="CourseName" column="CourseName"  />

    </resultMap>

  </resultMaps>



  <statements>

    <!--sql片段,重复使用-->

    <sql id="testSql">

      select p.ID,p.Name,p.Age,p.Sex,c.CourseName from Person p

      inner join  cour c on p.ID= c.PerID

    </sql>

    

    <!--查询所有记录-->

    <select id="GetList1" resultMap="PersonAndCourResult">

      select p.ID,p.Name,p.Age,p.Sex,c.CourseName from Person p

      inner join  cour c on p.ID= c.PerID

    </select>

    

    <!--多条件查询-->

    <select id="GetList2"  parameterClass="PersonAndCour" resultMap="PersonAndCourResult">

      select p.ID,p.Name,p.Age,p.Sex,c.CourseName from Person p

      inner join  cour c on p.ID= c.PerID 

      

      <dynamic perpend="where">

        <!--姓名不为NUll-->

        <isNotEmpty property="Name" prepend="And" >

          p.Name=#Name#

        </isNotEmpty>

        <!--年龄大于0-->

        <isGreaterThan prepend="and" property="Age" compareValue="0">

          p.Age = #Age#

        </isGreaterThan>

      </dynamic>

    </select>

    

    <select id="GetList3" resultMap="PersonAndCourResult">

      <!--<include refid="sql片段" />-->

      <include refid="testSql" />

      

    </select>

  </statements>

</sqlMap>
PersonAndCour.xml

这里面已经把SQL写好了,现在我们在控制器里执行,如下:

a.先说下SQL片段,我们在查询分页时的查询拼接条件或都其他查询时,可能同一段代码重复使用,这时我们就可以利用SQL片段,把重利利用的SQL代码提取出来,如PersonAndCour.xml里一样,这样我们用<include refid="testSql" />这一句就可以引用上面的SQL语句,这里的refid就是SQl片段中id的名字

b.再说说多条件查询时动态拼接SQl语句,如下图

 这里我们传入的参数是PersonAndCour,输出的参数也是这个, 这个参数我们也可以自己定义,需要几个字段就定义几个字段,但在这里时要写全名比如:parameterClass="MyBatisNet.EntityModel.whereModel",还有之前在博客园找到的动态SQL拼接的都是JAVA目录,进而是直接用<where>标签嵌套<if>标签,我在项目中也使用这种方式,但是<where>标签的里的语句就是不执行,最后在网上找了好多,最后发现用<dynamic>标签之后里面的语句就可以拼接成功了,原因现在还不知道,那位大神知道可以告知下,<dynamic>标签里面的判断条件整理了如下几个类型:

 (1)、一元条件

 

  (2)、二元条件

 

(3)其他

十一、最后说一下自己尝试的事务,如果大家有更好的建议可以提出来大家一起分享

(1)、直接把事务写在SQL语句里面,如下

 

(1)映射文件的中还是单一的条插入SQL语句,我们在控制器中使用事务,如下:

好了,到这里基本上已经全部写完了,好多是在网上找的资料,有些是自己尝试的,文章也是在做完DEMO后才写的,所以有遗漏的地方还望见谅,有错误的地方,或需要改进的地方望大家提出来一起分享,共同成长。

 

你可能感兴趣的:(mybatis)