SQL Server 使用C#窗体与数据库连接,制作数据库查看器
SQL Server 使用C#窗体与数据库连接,制作数据库查看器
本文中心:讨论C#对SQL Server 的增删改查,使用Treeview制作数据库查看器。
- SSMS部分:确保SQL Server 提供 SQL身份验证,同时建立一个对 名为 test 的数据库有操作权限的用户(用户名:pc,密码:123456)
--建表,插入数据
use test;
create table [User_Info](
[ID] int not null primary key,
[Username] varchar(20) not null,
[Password] varchar(10) not null
);
create table [Blog](
[ID] int not null primary key check([ID]>0),
[Title] varchar(100) not null,
[Content] varchar(100),
[Author_Id] int references [User_Info]([ID])
);
insert into [User_Info] values(001,'Kevin Li','123456');
insert into [User_Info] values(002,'Donold','123456');
insert into [User_Info] values(003,'Ki','123456');
insert into [User_Info] values(004,'Steves','123456');
insert into [User_Info] values(005,'Bill','123456');
insert into [User_Info] values(006,'Jaf','123456');
insert into [User_Info] values(007,'An Li','123456');
insert into [Blog] values(01,'Build Your Dreams','Read and Action',005);
select * from User_Info;
select * from Blog;

- C#部分:VS2022新建一个窗体应用,项目名称:”SQLServerConnection“

- V层:主窗体Form1布局以及修改属性如下:
- form1
- 事件 Load=Form1_Load
- 属性 StartPosition=CenterScreen
- 属性 FormBorderStyle = FixedSingle
- 属性 Text = SQLServer数据库查看器
- 属性 MaximizeBox = False
- treeView1
- 事件 AfterSelect = treeView1_AfterSelect
- listView1
- 属性 FullRowSelect = True
- 属性 GridLines = True
- 属性 View = Details
- form1

M层:新建数据库连接类 DB.cs。
相当于M层的角色,采用了单例的思想编写,确保数据库不会被多次连接占用系统的资源.C#对SQL Server的查询过程为:先打开数据库,查询,关闭数据库连接。只是查出来的结果,C#直接放到 **SqlDataAdapter **中,再直接 Fill 方法填充到 DataTable 里面。
这里主要提供两个方法,一是 getBySql 提供查询使用,即从SQL Server 查询数据使用;二是 setBySql 用作写数据。上述两个方法提供不同参数的重载,主要是用于应对带参数与不带参数的查询。
这里采用
string.format("{0}{1}",new Object[]{"a","b"});将a,b分别替换{0}与{1}的位置,防止任何形式的SQL注入。如补加 and 1 or 1在后面最后会形成 where xx = 'and 1 or 1'。同时由于C#规定不能在构造函数中关闭连接,这里只能实现 IDisposable 的接口。
在本程序中没用到写。
using System;
using System.Collections.Generic;
using System.Data; //DataTable 用到
using System.Data.SqlClient; //一系列的数据库操作类用到
using System.Text;
namespace SQLServerConnection
{
class DB : IDisposable
{
private SqlConnection sqlConnection;
//以下代码保证该类只能有一个实例
//在自己内部定义自己的一个实例,只供内部调用
private static DB db = null;
//这个类必须自动向整个系统提供这个实例对象
//这里提供一个供外部访问本class 的静态方法,可以直接访问
public static DB GetInstance()
{
if (db == null)
{
db = new DB();
}
return db;
}
//私有无参构造函数
public DB()
{
sqlConnection = new SqlConnection("server=.;uid=pc;pwd=123456;database=test");
sqlConnection.Open();
}
//单例化结束
//查询
public DataTable GetBySql(String sql, Object[] param)
{
sql = String.Format(sql, param);//用字符串参数替换的形式防止注入
SqlDataAdapter sqlDataAdapter = new SqlDataAdapter(new SqlCommand(sql, sqlConnection));
DataTable dataTable = new DataTable();
sqlDataAdapter.Fill(dataTable);
return dataTable;
}
//无参数的查询
public DataTable GetBySql(String sql)
{
SqlDataAdapter sqlDataAdapter = new SqlDataAdapter(new SqlCommand(sql, sqlConnection));
DataTable dataTable = new DataTable();
sqlDataAdapter.Fill(dataTable);
return dataTable;
}
//无查询结果的修改
public void setBySql(string sql, Object[] param)
{
sql = String.Format(sql, param);//用字符串参数替换的形式防止注入
new SqlCommand(sql, sqlConnection).ExecuteNonQuery();
}
//无参数,无查询结果的修改
public void setBySql(string sql)
{
new SqlCommand(sql, sqlConnection).ExecuteNonQuery();
}
//相当于析构函数
public void Dispose()
{
sqlConnection.Close();
//在C#中关闭数据库连接不能在类的析构函数中关闭,否则会抛出“内部.NET Framework 数据提供程序错误 1”的异常
//通过实现C#中 IDisposable 接口中的 Dispose() 方法主要用途是释放非托管资源
}
}
}
- C层:Form1.cs,承担将数据库的查询结果与其他组件连接起来的角色。主要使用了TreeView,对其节点TreeNode的操作,节点名同样是Text,在节点下添加节点用父节点Nodes.Add(子节点)。
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
namespace SQLServerConnection
{
public partial class Form1 : Form
{
DB db;
public Form1()
{
InitializeComponent();
}
//选择树节点触发
private void treeView1_AfterSelect(object sender, TreeViewEventArgs e)
{
//删除根节点,点击根节点是没有任何动作的!
if (e.Node.Parent != null)
{
string table_name = e.Node.Text;
groupBox2.Text = "表" + table_name;
listView1.Clear();
//查表的信息
DataTable table_info = db.GetBySql("select name from syscolumns where id=object_id('{0}')", new Object[] { table_name });
for (int i = 0; i < table_info.Rows.Count; i++)
{
for (int j = 0; j < table_info.Columns.Count; j++)
{ //生成表头
listView1.Columns.Add(table_info.Rows[i][j] + "", listView1.Width / table_info.Rows.Count - 1, HorizontalAlignment.Left);
}
}
//查表的内容
DataTable table = db.GetBySql("select * from [{0}]", new Object[] { table_name });
//数据更新,UI暂时挂起,知道EndUpdate绘制控件,可以有效避免闪烁并大大提高加载速度
listView1.BeginUpdate();
for (int i = 0; i < table.Rows.Count; i++)
{
ListViewItem listViewItem = new ListViewItem();//生成每一列
for (int j = 0; j < table.Columns.Count; j++)
{
if (j <= 0)
{
listViewItem.Text = table.Rows[i][j] + "";
}
else
{
listViewItem.SubItems.Add(table.Rows[i][j] + "");
}
}
listView1.Items.Add(listViewItem);
}
//结束数据处理,UI界面一次性绘制。
listView1.EndUpdate();
}
}
private void Form1_Load(object sender, EventArgs e)
{
try
{
TreeNode root_node = new TreeNode();//建立节点
root_node.Text = "Test 数据库";
treeView1.Nodes.Add(root_node);
db = DB.GetInstance();//初始化数据库查询单例 DB.cs
//查询test数据库有多少张表
DataTable table_name = db.GetBySql("select name from sysobjects where(xtype = 'U')");
//遍历查询出来的结果表(视图)
for (int i = 0; i < table_name.Rows.Count; i++)
{
for (int j = 0; j < table_name.Columns.Count; j++)
{
TreeNode treeNode = new TreeNode();
treeNode.Text = table_name.Rows[i][j] + "";
//一一将查询结果也就是表名添加到树节点
root_node.Nodes.Add(treeNode);
}
}
}
catch
{
MessageBox.Show(this.Text, "数据库出错!");
Environment.Exit(1);
}
}
}
}
- 效果

解决方案含有的目录:
DB.cs 是数据库连接类
Form1.cs 是主窗体

SQL Server 使用C#窗体与数据库连接,制作数据库查看器的更多相关文章
- sql server登录名、服务器角色、数据库用户、数据库角色、架构区别联系
原创链接:https://www.cnblogs.com/lxf1117/p/6762315.html sql server登录名.服务器角色.数据库用户.数据库角色.架构区别联系 1.一个数据库用户 ...
- 【转】SQL Server、Oracle、MySQL和Vertica数据库常用函数对比
SQL Server.Oracle.MySQL和Vertica数据库常用函数对比 Vertica数据库是HP公司新收购的用于BI方面的数据库. 1. 绝对值 S:select abs(-1) valu ...
- 使用SQL Server 2008的事务日志传送功能备份数据库(logshiping)
使用SQL Server 2008的事务日志传送功能备份数据库(logshiping) 使用SQL Server 2008的事务日志传送功能备份数据库(logshiping)
- 利用SQL Server Management Studio(SSMS)复制数据库
利用SQL Server Management Studio(SSMS)复制数据库 标签(空格分隔): SQLServer 前言 今天由于客户购买的软件版本确认了,而之前进行开发的本地数据库版本较低, ...
- 无法定位 Local Database Runtime 安装。请验证 SQL Server Express 是否正确安装以及本地数据库运行时功能是否已启用。
错误描述: 在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误.未找到或无法访问服务器.请验证实例名称是否正确并且 SQL Server 已配置为允许远程连接. (provide ...
- SQL Server Window Function 窗体函数读书笔记二 - A Detailed Look at Window Functions
这一章主要是介绍 窗体中的 Aggregate 函数, Rank 函数, Distribution 函数以及 Offset 函数. Window Aggregate 函数 Window Aggrega ...
- SQL Server Window Function 窗体函数读书笔记一 - SQL Windowing
SQL Server 窗体函数主要用来处理由 OVER 子句定义的行集, 主要用来分析和处理 Running totals Moving averages Gaps and islands 先看一个简 ...
- SQL Server GUID 数据迁移至MongoDB后怎样查看?
关键字:SQL Server NEWID():BSON:MongoDB UUID 1.遇到的问题和困惑 SQL Server中的NEWID数据存储到MongoDB中会是什么样子呢?发现不能简单的通过此 ...
- [转载]使用SQL Server 2008的事务日志传送功能备份数据库(logshiping)
http://www.cnblogs.com/benbenkoala/archive/2009/03/11/1407793.html 引言:SQL Server的事务日志传送备份是仅次于镜像的高可靠性 ...
- 如何使用本地账户"完整"安装 SharePoint Server 2010+解决“New-SPConfigurationDatabase : 无法连接到 SharePoint_Config 的 SQL Server 的数据 库 master。此数据库可能不存在,或当前用户没有连接权限。”
注:目前看到的解决本地账户完整安装SharePoint Server 2010的解决方案如下,但是,有但是的哦: 当我们选择了"完整"模式安装SharePointServer201 ...
随机推荐
- java中单例模式 如何使用
引用: 原文链接:https://blog.csdn.net/qq_41458550/article/details/109243456
- uniapp主题切换功能的第一种实现方式(scss变量+vuex)
随着用户端体验的不断提升,很多应用在上线的时候都要求做不同的主题,最基本的就是白天与夜间主题. 就像b站app主题切换,像这样的 uniapp因为能轻松实现多端发布而得到很多开发者的青睐,但每个端的实 ...
- 在Transformers 中使用约束波束搜索引导文本生成
引言 本文假设读者已经熟悉文本生成领域波束搜索相关的背景知识,具体可参见博文 如何生成文本: 通过 Transformers 用不同的解码方法生成文本. 与普通的波束搜索不同,约束 波束搜索允许我们控 ...
- 【技术积累】Python中的NumPy库【二】
NumPy库的主要类有哪些? NumPy库的主要类包括: ndarray:N维数组对象,是NumPy最重要的类之一.它是Python中数组的基本数据结构,可以进行高效的数学计算和数据处理操作. ufu ...
- Mybatis使用级联映射时 , 查询的结果为null
错误原因 在学习多对一映射处理中的级联方式处理映射关系时 , 发现自己查询的结果有一个为为null 于是就开始对代码进行排查 , debug ,最终发现错误 , 原来是自己映射中的 property ...
- Unity中的InitializeOnLoad特性:深入解析与实践
Unity中的InitializeOnLoad特性:深入解析与实践 在Unity开发过程中,我们经常需要在编辑器启动时或脚本重新编译后执行一些操作,例如初始化数据.注册事件等.这时,我们可以使用Ini ...
- GetX 关于报错 Null check operator used on a null value的解决
import 'package:flutter/material.dart'; import 'package:get/get.dart'; import 'logic.dart'; class Ge ...
- Java 中怎样将 bytes 转换为 long 类型?
将bytes 转换为long类型: 第一种方式: String 接收 bytes 的构造器转成 String,再 Long.parseLong: 但此种情况需要注意:字节数组中的每个字节都必须是有效的 ...
- SAP ABAP 使用GENIOS求解线性规划问题的简单例子
主要内容来自Operations Research & ABAP ,结合我遇到的需求,做了一些修改. 需求:有BOX1和BOX2两种箱子,分别能包装不同数量的A物料和B物料,给出若干数量的A, ...
- 我真的想知道,AI编译器中的IR是什么?
随着深度学习的不断发展,AI 模型结构在快速演化,底层计算硬件技术更是层出不穷,对于广大开发者来说不仅要考虑如何在复杂多变的场景下有效的将算力发挥出来,还要应对 AI 框架的持续迭代. AI 编译器就 ...