在sqlserver2005或SQL2008数据库项目中,创建视图或存储过程的时候往往有以下开头语句:


SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO

………………

………………

GO

SET ANSI_NULLS OFF
GO
SET QUOTED_IDENTIFIER OFF
GO

这些是 SQL-92 设置语句,使 SQL Server 2000/2005 遵从 SQL-92 规则。
当 SET QUOTED_IDENTIFIER 为 ON 时,标识符可以由双引号分隔,而文字必须由单引号分隔。当 SET QUOTED_IDENTIFIER 为 OFF 时,标识符不可加引号,且必须符合所有 Transact-SQL 标识符规则。
SQL-92 标准要求在对空值进行等于 (=) 或不等于 (<>) 比较时取值为 FALSE。当 SET ANSI_NULLS 为 ON 时,即使 column_name 中包含空值,使用 WHERE column_name = NULL 的 SELECT 语句仍返回零行。即使 column_name 中包含非空值,使用 WHERE column_name <> NULL 的 SELECT 语句仍会返回零行。
当 SET ANSI_NULLS 为 OFF 时,等于 (=) 和不等于 (<>) 比较运算符不遵从 SQL-92 标准。使用 WHERE column_name = NULL 的 SELECT 语句返回 column_name 中包含空值的行。使用 WHERE column_name <> NULL 的 SELECT 语句返回列中包含非空值的行。此外,使用 WHERE column_name <> XYZ_value 的 SELECT 语句返回所有不为 XYZ_value 也不为 NULL 的行。

posted @ 2009-02-21 09:05 lastgame 阅读(2397) 评论(1) 编辑
实用前提:在WinForm程序中,通过主窗体的Menu打开不同的窗体
我以前的做法:在Menu的Click事件中,创建打开窗体的对象实例
这样的缺点,我不说了:)
看看利用反射动态调用窗体对象的方法:
Code


这样的好处,也自己看吧,:)
posted @ 2009-02-19 12:13 lastgame 阅读(555) 评论(3) 编辑
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;

namespace VirtualMode
{
   partial class VirtualModeForm : Form
   {
      private List<DataObject> m_Data = new List<DataObject>();
      private List<bool> m_Visited = new List<bool>();
      public VirtualModeForm()
      {
         InitializeComponent();
         m_Grid.CellValueNeeded += OnCellValueNeeded;
         m_GetVisitedCountButton.Click += OnGetVisitedCount;
         InitData();
         InitGrid();
      }

      private void InitData()
      {
         for (int i = 0; i < 1000001; i++)
         {
            m_Visited.Add(false);
            DataObject obj = new DataObject();
            obj.Id = i;
            obj.Val = 2 * i;
            m_Data.Add(obj);
         }
      }

      private void InitGrid()
      {
         m_Grid.VirtualMode = true;
         m_Grid.ReadOnly = true;
         m_Grid.AllowUserToAddRows = false;
         m_Grid.AllowUserToDeleteRows = false;
         m_Grid.ColumnCount = 3;
         m_Grid.Rows.Add();
         m_Grid.Rows.AddCopies(0, 1000000);
         // Uncomment the next line and comment out the  
         // the rest of the method to switch to data bound mode
         //m_Grid.DataSource = m_Data;
      }
      private void OnCellValueNeeded(object sender,
         DataGridViewCellValueEventArgs e)
      {
         m_Visited[e.RowIndex] = true;
         if (e.ColumnIndex == 0)
         {
            e.Value = m_Data[e.RowIndex].Id;
         }
         else if (e.ColumnIndex == 1)
         {
            e.Value = m_Data[e.RowIndex].Val;
         }
         else if (e.ColumnIndex == 2)
         {
            Random rand = new Random();
            e.Value = rand.Next();
         }
      }

      private void OnGetVisitedCount(object sender, EventArgs e)
      {
         int count = 0;
         foreach (bool b in m_Visited)
         {
            if (b) count++;
         }
         MessageBox.Show(count.ToString());
      }

       private void VirtualModeForm_Load(object sender, EventArgs e)
       {

       }
   }
   public class DataObject
   {
      private int m_Id;
      private int m_Val;

      public int Val
      {
         get { return m_Val; }
         set { m_Val = value; }
      }

      public int Id
      {
         get { return m_Id; }
         set { m_Id = value; }
      }
   }

}
posted @ 2009-02-12 17:13 lastgame 阅读(488) 评论(0) 编辑
DataGridView是.net 2.0新增的表格数据编辑和显示控件,简单的数据显示和编辑,只需直接和数据源绑定就可以了。

对于 一些特殊情况,我们需要自己填充DataGridView,这时候只需要按照行列顺序,首先获得行,然后通过行的Cells属性,得到单元格,设置其 Value属性即可。但这种模式有个问题,即对于几十行或者几百行的数据,显示效率不是问题,当数据量逐渐增大时,效率就成了一个非常重要的问题。

那 么,如何解决这样的问题呢,写过或者看过类似Grid控件的朋友一定会想到,按需加载数据。也就是当我们在数据表格中移动时,显示到屏幕上的数据只是实际 需要的数据的一部分,那么,只加载这部分数据即可。而对于DataGridView,一切都准备好了,只要我们开启VirtualMode,然后实现几个 事件(用于加载保存数据),其他的事情DataGridView会替我们做好。

在WinForm窗体中添加一个DataGridView控件,设置VirtualMode属性为True。

实现如下事件:

第一个需要实现的事件是:CellValueNeeded,该事件在控件刷新,需要为单元格填充数据时发生,其参数e返回当前单元格的行和列,根据行和列,获取需要的值,赋给e的Value属性。

如果只是显示数据,该事件已足够,从理论上,通过这样的方法,DataGridView显示的数据与数据量无关,即使数据量再大,也可以在常数时间内完成刷新。

第二个需要实现的事件是CellValuePushed,该事件在一个单元格编辑完成后触发,将数据写回数据源。其中事件的参数e会返回单元格的行、列、值。

如果要添加行和删除行,则需要实现NewRowNeeded和UserDeletingRow事件。还有其他一些事件,来控制放弃单元格、行的编辑等事件。具体可以参考帮助文档。

总之,使用虚拟模式,一方面可以很好的为DataGridView控件加载数据,另一方面,也可以达到数据和显示分离的目的。
posted @ 2009-02-12 16:56 lastgame 阅读(913) 评论(0) 编辑
DataGridView虽然好用,但是如果数据量比较大的话就会出现性能的问题。网上提供的一般做法是通过虚拟模式(Virtual Mode)来加速。但是有的时候,可以通过简单的设置来加速DataGridView。
(1)使用绑定数据。就是通过加载DataTable然后通过BindingSource来和DataGridView.DataSource进行交互。
(2) 不要使用AutoSizeColumnsMode.AllCells。如果这么设置,DataGridView只有等到所有的单元格都绘制完毕了才能显 示。因为列的宽度是要根据这一列最长的单元格来决定的。如果改为DisplayedCells的话,那么只要显示的部分绘制完毕就可以决定列宽。
(3)可能的情况下使用AutoGenerateColumns来自动设定列。
在实际应用中,一个10000*20左右的DataGridView,原来的显示效率是100s左右,使用了上述的办法之后变成了6s。效果还是比较明显的。
当然,如果要加速更多的数据量的话,还是需要使用虚拟模式的。毕竟上述的方法只是解一时之需。
posted @ 2009-02-12 16:56 lastgame 阅读(275) 评论(1) 编辑

操纵dataset

在DataSet中DataRow是其所有数据的基本存放位置,它主要是由一个值数组组成,代表DataTable单独一行。
DataRow中主要包括一下几种信息:1、行中每一列的当前值,2、行中每一列的原始值,3、行状态,4、父行与子行间的链接

初始化一个DataRow:
DataTable dataTable=dataSet.Tables[0];
DataRow newRow=dataTable.NewRow(); //用dataTable生成DataRow可以利用dataTable里面的模式
dataTable.Rows.Add(newRow);

删除行:
DataTable.Rows.Remove(行实例);
DataTable.Rows.RemoveAt(行号);
DataRow.Delete(); //行自身移除

读写DataRow的值:
row["列名"],row[列号]均可引用其中的一个属性
DataColumn a=dataTable.Columns("列名"); //可以获得一个列

对行进行批处理更改:
BeginEdit()开始更改,EndEdit()结束更改,同时将更改结果写入DataSet,CancelEdit(),取消更改
例如:
row.BeginEdit();
对row进行更改
row.EndEdit();

将数据批量加载到DataTable

dataTable.BeginLoadData();
dataTable.LoadDataRow(row1,false); //第二个参数为true时,调用dataTable.AcceptChanges()时接受更改,为false直接添加
……
dataTable.EndLoadData();
使用这种数据加载方式可以在数据加载期间屏蔽所有的数据约束,索引也不会予以维护,极大的加快了数据加载速度

行的版本:
current:当前值
default:根据操作的不同决定行的default值
original:最后一次调用AcceptChanges()之后的值
proposed:调用AcceptChanges()之前被更改的值
例如要获得行的original值:
String oldString=row("FirstName",DataRowVersion.original);

行的状态:
row.RowState获得行的状态,例如删除后变成Deleted,数据存储更新后变为unchanged

DataSet导航

在ADO.NET中每个表都保持其相对独立性,允许在行级上导航不同表之间的相关行(向下导航到子行,向上导航的父行)
如DataRow[] invoiceRows=custRow.GetChildRows("Customer_invoice"); //通过关系导航到子行

DataView

DataView就时数据视图,为数据库结构提供了外模式的实现。
同时DataView也可以为窗体控件和Web控件提供数据绑定功能,在每一个DataTable中内建了一个DataView为:DataTable.DefaultView();

创建DataView

DataView sortedView=new DataView(dataTable);

对DataView进行排序

dataTable.DefaultView.sort="lastName";
dataTable.DefaultView.sort="lastName,FirstName DESC";

对DataView进行筛选

1、通过对其中的RowFilter属性设置可以实现筛选
dataTable.DefaultView.RowFilter="Vendor='Rawlings'";
不过筛选表达式只能设置成比较简单的表达式,功能有限,不过可以满足基本的要求。
同样在DataTable里面也可以进行简单的搜索,返回一个DataRow数组,例:
DataRow[] compoundRows=dataTable.select("Vendor='wilson' AND price>20.00)
2、通过RowState来筛选
dataTable.DefaultView.RowStateFilter="DataViewRowState.originalRows"可以筛选出符合要求状态的row

对DataView进行搜索:
相对于DataView使用RowFilter进行筛选得到一个矩形数据集,使用Find、FindRows可以更准确的查找到与特定键相匹配的行
搜索的时候必须首先设置DataView的sort属性:
int found=dataTable.DefaultView.Find("wilson"); //获得行的位置
DataRowView[] rows=dataTable.DefaultView.FindRows("Rawlings") //过得一个row数组

更新DB

在DataSet中,每一个DataTable对应着一个DataAdapter,DataAdapter.Update()时,DataTable自动更新。
更新的时候可以使用CommandBuilder自动根据DataSet的变化生成更新的SQL命令
SqlCommandBuilder bldr=new SqlCommandBuilder(dataAdapter);
dataAdapter.Update(custTable);
不过Update接受DataSet参数并不更新DataSet而是更新DataSet中的一个叫"Table"的表
使用CommandBuilder进行更新的时候要注意一下几点:
1、SelectCommand必须有效
2、必须有主码
3、若SelectCommand填充DataTable后架构发生改变,应该在Update()之前调用CommandBuilder.RefreshSchema();
4、更新DB时不受关系、约束或者DataSet中其他表的影响
虽然使用CommandBuilder比较方便,不用自己写更新命令,但自动生成的命令性能不高,这时可以考虑自己编写存储过程或直接使用带参数的sql语句,例如:
String insQry="Insert into Customer(CustomerID) Values (@Customer)";
SqlCommand insCmd=conn.CreateCommand();
insCmd.CommandText=insQry;
SqlParameterCollection insParams=insCmd.Parameters;
insParams.Add("@CustomerID",SqlDbType.UniqueIdentifier,0,"CustomerID");
dataAdapter.InsertCommand=insCmd;
dataAdapter.Update();
在dataAdapter.Update()更新时还可以控制更新的范围:
dataAdapter.Update(invTable.GetChanges(DataRowState.Deleted); //只更新被删除的部分

事务

tx=conn.BeginTransaction(IsolationLevel.Serializable);
invDA.SelectCommand.Transaction=tx;
事务操作
tx.Commit();提交 //tx.Rollback();事务回滚

数据绑定

简单版本:(对文本框、标签等)
{Controls}.DataBindings.Add("{Property}",{dataSource},"{dataMember}");
其中Property为待绑定的属性,dataSource为DataView或DataTable,dataMember为dataSource其中的某个属性

复杂版本:(对ListBox、ComboBox等)
要分别设置各个属性实现绑定:
DataSource=支持IList的一个对象(DataTable或DataView)
DisplayMember:待显示的DataSource中的一个属性
ValueMember:确定在DataSource中引用哪一个数据行,即实现与DisplayMember的名值对应

DataGrid绑定:
1、可以设置DataSource属性实现静态绑定
2、可以使用SetDataBinding函数实现动态绑定

同时DataGrid支持主控/详细表显示
masterGrid.setDataBinding(customerTable,"");
detailGrid.setDataBinding(customerTable,"Customer_Invoices") //第二个属性要设置成关系约束
这样在主表中选择一行,在子表中就根据主表行中外码在子表中找到相应行

绑定之后,绑定项中就有一个CurrencyManager属性实现游标功能
BindingContext[CustomerTable]返回一个CurrencyManager对象,其中的Position属性可以更改,实现游标的移动
posted @ 2009-02-11 17:34 lastgame 阅读(802) 评论(0) 编辑
    这三个方法用于删除字符串头尾出现的某些字符。Trim()删除字符串头部及尾部出现的空格,删除的过程为从外到内,直到碰到一个非空格的字符为 止,所以不管前后有多少个连续的空格都会被删除掉。TrimStart()只删除字符串的头部的空格。TrimEnd()只删除字符串尾部的空格。

      如果这三个函数带上字符型数组的参数,则是删除字符型数组中出现的任意字符。如Trim("abcd".ToCharArray())就是删除字符串头部及尾部出现的a或b或c或d字符,删除的过程直到碰到一个既不是a也不是b也不是c也不是d的字符才结束。
      这里最容易引起的误会就是以为删除的是"abcd"字符串。如下例:
Code

      可能有人以为上面s的最终结果是"from dual",但真正的结果是"from d"。需要注意的是这种写法执行的删除对象是字符数组中出现的任意字符,而不是这些字符连在一起组成的字符串!
posted @ 2009-01-24 11:46 lastgame 阅读(399) 评论(0) 编辑
posted @ 2007-11-03 15:37 lastgame 阅读(46) 评论(0) 编辑