C语言连接sql server实现增删改查

上数据库课时老师布置的一份作业,在做这个之前甚至不会sql server,硬是查了网上所能查到的还学了一点sql server才理解做出,非原创,只是整合了大佬们留下的各种代码,方便后来者看一看吧。

1.需要知道的

        首先,先了解一下sql server中如何实现插入操作

insert into 表1
values('木木','男','202100020')

这个操作是将  姓名:木木,性别:男,学号:202100020  这三个数据插入到表1中去。

 而我们c语言能做什么,就是将上面的代码每一个字符传入sql server中,这些操作都得自己完成,没有已经准备好的库函数,因此上面说看看就好,真要操作还是得用python。

2.创建库以及连接库

        2.1创建库

                打开sql使用密码登录,注意服务器名称为你的电脑设备名称

              

                右键数据库,新建库,双击新建的库,右键表新建一个表

         2.2链接库

                打开ODBC->添加sql server

                      登录数据库的id进行登录

                一直下一步,结束。

                然后打开vs进行操作。

3.完整代码

        在vs2022可以运行!

        

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<windows.h>
#include<sql.h>
#include<sqlext.h>
#include<sqltypes.h>

SQLRETURN ret;
SQLHENV henv;//环境句柄
SQLHDBC hdbc;//连接句柄
SQLHSTMT hstmt;//语句句柄
SQLRETURN retcode;//返回值

void sql_connect(SQLRETURN ret, SQLHENV& henv, SQLHDBC& hdbc, SQLHSTMT& hstmt)//设置连接参数的句柄

{
    const char* SY1 = "mytest";
    unsigned char* SY = (unsigned char*)SY1;
    const char* db21 = "sa";//数据库ODBC连接的账户
    unsigned char* db2 = (unsigned char*)db21;
    const char* pass1 = "123456";//数据库ODBC连接的密码
    unsigned char* pass = (unsigned char*)pass1;

    ret = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);//申请环境句柄
    ret = SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, (void*)SQL_OV_ODBC3, SQL_IS_INTEGER);
    ret = SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);//申请数据库连接句柄

    ret = SQLConnect(hdbc, SY, SQL_NTS, db2, SQL_NTS, pass, SQL_NTS);
    /*db2为配置的ODBC数据源名称,这里根据自己的配置进行修改*/

    if (!(ret == SQL_SUCCESS || ret == SQL_SUCCESS_WITH_INFO))
    {
        printf("连接数据库失败!\n");
        return;
    }
	else
	{
		printf("数据库连接成功!\n");
	}

    ret = SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);//用于分配句柄
}

void sql_add(SQLRETURN ret, SQLHENV& henv, SQLHDBC& hdbc, SQLHSTMT& hstmt)
{
	SQLCHAR sql1[] = "use TEXT";		//选择要进行操作的数据库
	retcode = SQLExecDirect(hstmt, sql1, SQL_NTS);

	int i = 0;
	unsigned char tablename[200];

	strcpy((char*)tablename, "Studentlnfo(Stuid,StuName,Classid,Sex,Phone,CreatTime) values (?,?,?,?,?,?)");
	printf("请根据Stuid,StuName,Classid,Sex,Phone,CreatTime依次输入数据,数据之间带回车或空格\n");


	char s[10][10];  //用于保存输入流信息
	/*long conlumnlen;*/
	SQLAllocStmt(hdbc, &hstmt);
	SQLCHAR sql2[100] = "insert into ";//要注意,sqlchar实际上就是unsigned char,这在字符串操作语句当中是不能够直接操作的,需要强转为char型后再进行操作
	strcat((char*)sql2, (char*)tablename);
	//为什么要strcat()?就是想在后面执行这样一段话:
	// "insert into Studentlnfo(Stuid,StuName,Classid,Sex,Phone,CreatTime) values (?,?,?,?,?,?)"

	SQLPrepare(hstmt, (SQLCHAR*)sql2, strlen((char*)sql2));//这里是对sql语句进行prepare操作,这一步有助于防注入,提高安全性
	
	for (i = 0; i < 6; i++)
	{
		scanf("%s", s[i]);
		SQLBindParameter(hstmt, i + 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, 10, 0, &s[i], 100, NULL);
		//sqlbindparameter函数操作用于绑定刚才的问号的位置
	}
	

	SQLExecute(hstmt);

	if (ret == SQL_SUCCESS || ret == SQL_SUCCESS_WITH_INFO)
	{
		printf("操作成功!\n");
	}
	else
	{
		UCHAR errmsg[100];
		printf("操作失败!\n");
		SQLError(henv, hdbc, hstmt, NULL, NULL, errmsg, sizeof(errmsg), NULL);
		printf("%s", errmsg);
	}

}

void sql_delete(SQLRETURN ret, SQLHENV& henv, SQLHDBC& hdbc, SQLHSTMT& hstmt)
{
	SQLCHAR sql1[] = "use TEXT";		//选择要进行操作的数据库
	retcode = SQLExecDirect(hstmt, sql1, SQL_NTS);

	int i = 0;
	unsigned char tablename[60];

	strcpy((char*)tablename, "Studentlnfo where Stuid = ?"); printf("请输入id,之间用空格隔开\n");
	unsigned char s[3][10];
	SQLCHAR sql[120] = "delete from ";
	/*long conlumnlen;*/
	SQLAllocStmt(hdbc, &hstmt);
	strcat((char*)sql, (char*)tablename);
	SQLPrepare(hstmt, (SQLCHAR*)sql, strlen((char*)sql));
	

	scanf("%s", s[0]);
	SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, 10, 0, &s[0], 100, NULL);
	
	
	SQLExecute(hstmt);//sql执行
	if (ret == SQL_SUCCESS || ret == SQL_SUCCESS_WITH_INFO)
	{
		printf("操作成功!\n");
	}
	else
	{
		UCHAR errmsg[100];
		printf("操作失败!\n");
		SQLError(henv, hdbc, hstmt, NULL, NULL, errmsg, sizeof(errmsg), NULL);
		printf("%s", errmsg);
	}
}

void sql_mod(SQLRETURN ret, SQLHENV& henv, SQLHDBC& hdbc, SQLHSTMT& hstmt)
{
	SQLCHAR sql1[] = "use TEXT";		//选择要进行操作的数据库
	retcode = SQLExecDirect(hstmt, sql1, SQL_NTS);

	int i = 0;
	unsigned char tablename1[60], tablename2[60];
	unsigned char s[3][10];

	strcpy((char*)tablename1, "Studentlnfo set "); 
	strcpy((char*)tablename2, "where Stuid=? and StuName=?");

	unsigned char s1[10];
	char part[6][20];
	strcpy(part[0], "Stuid=? ");
	strcpy(part[1], "StuName=? ");
	strcpy(part[2], "Classid=? ");
	strcpy(part[3], "Sex=? ");
	strcpy(part[4], "Phone=? ");
	strcpy(part[5], "CreatTime=? ");

	int j = 0, num;
	do {
		printf("选择修改的元素名\n");
		printf("1.id 2.name 3.class_id 4.sex 5.phone 6.CreatTime\n");
		scanf("%d", &j);
	} while (j < 1 || j>6);
	SQLCHAR sql[100] = "update ";
	/*long conlumnlen;*/
	SQLAllocStmt(hdbc, &hstmt);
	strcat((char*)sql, (char*)tablename1);
	strcat((char*)sql, part[j - 1]);
	strcat((char*)sql, (char*)tablename2);
	printf("%s\n", sql);
	SQLPrepare(hstmt, (SQLCHAR*)sql, strlen((char*)sql));
	
	
	printf("请输入id、name:");
	scanf("%s", s[0]);
	scanf("%s", s[1]);
	SQLBindParameter(hstmt, 2, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, 10, 0, &s[0], 100, NULL);
	SQLBindParameter(hstmt, 3, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, 10, 0, &s[1], 100, NULL);

	printf("输入修改元素值:");
	scanf("%s", s1);
	SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_CHAR, 10, 0, &s1, 100, NULL);

	ret = SQLExecute(hstmt);
	if (ret == SQL_SUCCESS || ret == SQL_SUCCESS_WITH_INFO)
	{
		printf("操作成功!\n");
	}
	else
	{
		UCHAR errmsg[100];
		printf("操作失败!\n");
		SQLError(henv, hdbc, hstmt, NULL, NULL, errmsg, sizeof(errmsg), NULL);
		printf("%s", errmsg);
	}
}

void sql_display(SQLRETURN ret, SQLHENV& henv, SQLHDBC& hdbc, SQLHSTMT& hstmt)
{
	retcode = SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);
	SQLCHAR sql1[] = "use TEXT";//使用名为"TEXT"数据库,读者根据自己创建的数据库名称进行修改
	retcode = SQLExecDirect(hstmt, sql1, SQL_NTS);//SQLExecDirect,此函数用于执行SQL语句,执行sql1语句,即"use TEXT"

	SQLCHAR sql2[] = "select * from Studentlnfo";//查询已创建好的名为"StudentInfo"表所有列
	retcode = SQLExecDirect(hstmt, sql2, SQL_NTS);//执行sql2语句
	if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
	{
		SQLCHAR str1[50], str2[50], str3[50], str4[50], str5[50], str6[50];
		SQLINTEGER len_str1, len_str2, len_str3, len_str4, len_str5, len_str6;
		while (SQLFetch(hstmt) != SQL_NO_DATA)
		{
			SQLGetData(hstmt, 1, SQL_C_CHAR, str1, 50, &len_str1);   //获取第一列数据
			SQLGetData(hstmt, 2, SQL_C_CHAR, str2, 50, &len_str2);   //获取第二列数据
			SQLGetData(hstmt, 3, SQL_C_CHAR, str3, 50, &len_str3);
			SQLGetData(hstmt, 4, SQL_C_CHAR, str4, 50, &len_str4);
			SQLGetData(hstmt, 5, SQL_C_CHAR, str5, 50, &len_str5);
			SQLGetData(hstmt, 6, SQL_C_CHAR, str6, 50, &len_str6);

			printf("\n%s\t%s\t%s\t%s\t%s\t%s\n", str1, str2, str3, str4, str5, str6);
		}
	}
	printf("process finish\n");
}

int main()
{
	sql_connect(ret, henv, hdbc, hstmt);
	int a;
	while (1)
	{
		printf("请输入序号进行的操作:1.增加 2.删除 3.修改 4.查询 0.退出\n");
		scanf("%d", &a);
		switch (a)
		{
		case 1:sql_add(ret, henv, hdbc, hstmt);
			system("pause");
			system("cls");
			break;
		case 2:sql_delete(ret, henv, hdbc, hstmt);
			system("pause");
			system("cls");
			break;
		case 3:sql_mod(ret, henv, hdbc, hstmt);
			system("pause");
			system("cls");
			break;
		case 4:sql_display(ret, henv, hdbc, hstmt);
			system("pause");
			system("cls");
			break;
		case 0:exit(-1);

		}

	}

}

4.运行示例

操作前的库:

  • 1
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
在 C 语言中,连接 SQL Server 数据库可以使用 Microsoft 提供的 ODBC API 接口。下面是一个简单的连接 SQL Server 数据库的示例代码: ``` #include <stdio.h> #include <stdlib.h> #include <sql.h> #include <sqlext.h> int main() { SQLHENV env; // 环境句柄 SQLHDBC dbc; // 连接句柄 SQLRETURN ret; // 返回值 // 初始化环境句柄 ret = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &env); if (ret != SQL_SUCCESS && ret != SQL_SUCCESS_WITH_INFO) { printf("Failed to allocate environment handle\n"); return 1; } // 设置环境属性 ret = SQLSetEnvAttr(env, SQL_ATTR_ODBC_VERSION, (SQLPOINTER)SQL_OV_ODBC3, 0); if (ret != SQL_SUCCESS && ret != SQL_SUCCESS_WITH_INFO) { printf("Failed to set environment attribute\n"); SQLFreeHandle(SQL_HANDLE_ENV, env); return 1; } // 分配连接句柄 ret = SQLAllocHandle(SQL_HANDLE_DBC, env, &dbc); if (ret != SQL_SUCCESS && ret != SQL_SUCCESS_WITH_INFO) { printf("Failed to allocate connection handle\n"); SQLFreeHandle(SQL_HANDLE_ENV, env); return 1; } // 连接数据库 SQLCHAR* conn_str = "Driver={SQL Server};Server=localhost;Database=mydb;Uid=myusername;Pwd=mypassword;"; ret = SQLDriverConnect(dbc, NULL, conn_str, SQL_NTS, NULL, 0, NULL, SQL_DRIVER_COMPLETE); if (ret != SQL_SUCCESS && ret != SQL_SUCCESS_WITH_INFO) { printf("Failed to connect to database\n"); SQLFreeHandle(SQL_HANDLE_DBC, dbc); SQLFreeHandle(SQL_HANDLE_ENV, env); return 1; } // 连接成功 printf("Connected to database successfully\n"); // 断开连接 SQLDisconnect(dbc); // 释放连接句柄和环境句柄 SQLFreeHandle(SQL_HANDLE_DBC, dbc); SQLFreeHandle(SQL_HANDLE_ENV, env); return 0; } ``` 需要注意的是,连接字符串 `conn_str` 中需要填写正确的 SQL Server 服务器、数据库、用户名和密码等信息。另外,连接成功后可以使用 ODBC API 提供的函数执行 SQL 询语句等操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值