C语言通过IXMLHTTPRequest以get或post方式发送http请求获取服务器文本或xml数据

做过网页设计的人应该都知道ajax。
Ajax即Asynchronous Javascript And XML(异步的JavaScript和XML)。使用Ajax的最大优点,就是能在不更新整个页面的前提下维护数据。这使得Web应用程序更为迅捷地回应用户动作,并避免了在网络上发送那些没有改变的信息。
在IE浏览器中,Ajax技术就是基于JavaScript里面的XMLHTTPRequest。AJAX通过XMLHttpRequest对象发出HTTP 请求,得到服务器返回的数据后,再进行处理。现在,服务器返回的都是JSON格式的数据,XML格式已经过时了,但是AJAX这个名字已经成了一个通用名词,字面含义已经消失了。
XMLHttpRequest对象是AJAX的主要接口,用于浏览器与服务器之间的通信。尽管名字里面有XML和Http,它实际上可以使用多种协议(比如file或ftp),发送任何格式的数据(包括字符串和二进制)。

其实,不仅在网页上能用JavaScript语言调用XMLHTTPRequest组件,在桌面窗口程序里面也可以用C语言或C++调用XMLHTTPRequest组件。
XMLHTTPRequest是微软msxml6.0里面的组件。msxml6.0可直接解析服务器返回的xml文档,但json数据需要在网上找cJSON库来解析。

首先我们要在自己的服务器上准备好处理ajax请求的页面,本文准备了三个示例页面:str_test.php、json_test.php和xml_test.php,分别用来产生文本回应、json回应和xml回应。xml_test.php页面支持get和post两种ajax请求方式。

PHP文件用UTF-8编码保存,但C文件要用GB2312编码保存。
这是因为在微软的简体中文Windows系统里面,以A结尾的API函数(如CreateWindowA)采用的是GB2312编码的char *字符串,以W结尾的API函数(如CreateWindowW)采用的是UTF-16编码的wchar_t *字符串,COM组件使用的BSTR字符串是在wchar_t *字符串的基础上增加了4字节的字符串长度前缀。
IXMLHttpRequest组件(一种COM组件)工作时使用的是UTF-16编码的BSTR字符串。
我们在收到IXMLHttpRequest组件提供的UTF-16编码的BSTR字符串后,如果想要用printf在控制台上打印出来,就需要用convert_bstr_to_string函数将BSTR转换为GB2312编码的char *字符串。想要直接打印BSTR字符串不能用printf函数,要用WriteConsoleW函数才行。如果是显示在窗口的文本框或者窗口的标题栏上那就简单了,直接用SetWindowTextW或SetDlgItemTextW函数,把BSTR字符串传进去就行了。
同样,在给IXMLHttpRequest组件传递字符串参数时,需要先用convert_string_to_bstr函数将GB2312编码的char *字符串转换为UTF-16编码的带长度前缀的BSTR字符串。
如果是写入txt文本文件的话,我们就可以采用UTF-8编码,用MultiByteToWideChar函数(CP_UTF8)将BSTR转换成UTF-8编码的char *字符串,再用fprintf或fwrite写入txt文件。

str_test.php:

<?php header('Content-type: text/html; charset=utf-8'); ?>
当前时间为<?=date("Y年n月j日 H:i:s")?>。

 json_test.php:

<?php
header('Content-type: application/json');
header('Pragma: no-cache');
header('Cache-control: no-cache');
header('Expires: 0');$arr = array();
$arr["date"] = date("Y-n-j H:i:s");
$arr["time"] = time();
$arr["desc"] = "abcd简体中文";
echo json_encode($arr);
?>

xml_test.php:

<?php
header('Content-type: text/xml');session_start();
header('Pragma: no-cache');
header('Cache-control: no-cache');
header('Expires: 0');echo '<?xml version="1.0" encoding="utf-8"?>';
$timestr = strftime("%Y%m%d%H%M%S");
echo "<test timestr=\"$timestr\" example=\"简体中文\">";
foreach ($_COOKIE as $name => $value) {$_name = htmlspecialchars($name);$_value = htmlspecialchars($value);echo "<cookie name=\"$_name\">$_value</cookie>";
}
foreach ($_GET as $name => $value) {$_name = htmlspecialchars($name);$_value = htmlspecialchars($value);echo "<param method=\"get\" name=\"$_name\">$_value</param>";
}
foreach ($_POST as $name => $value) {$_name = htmlspecialchars($name);$_value = htmlspecialchars($value);echo "<param method=\"post\" name=\"$_name\">$_value</param>";
}
echo '</test>';
?>

下面我们来看看C语言如何像网页里面那样用XMLHTTPRequest发送ajax请求。

/* 这个程序只能在C编译器下编译成功, 请确保源文件的扩展名为c */
#define COBJMACROS
#include <stdio.h>
#include <MsXml6.h>#pragma comment(lib, "msxml6.lib")// char *转BSTR
// 用完后调用SysFreeString释放
BSTR convert_string_to_bstr(const char *s)
{int n;wchar_t *ws;BSTR bstr = NULL;n = MultiByteToWideChar(CP_ACP, 0, s, -1, NULL, 0);ws = malloc(n * sizeof(wchar_t));if (ws != NULL){MultiByteToWideChar(CP_ACP, 0, s, -1, ws, n);bstr = SysAllocString(ws);free(ws);}return bstr;
}// BSTR转char *
// 用完后调用free释放
char *convert_bstr_to_string(BSTR bstr)
{char *s;int n;n = WideCharToMultiByte(CP_ACP, 0, bstr, -1, NULL, 0, NULL, NULL);s = malloc(n);if (s != NULL)WideCharToMultiByte(CP_ACP, 0, bstr, -1, s, n, NULL, NULL);return s;
}// 去掉字符串末尾的\r\n
void remove_last_crlf(char *str)
{int len;len = (int)strlen(str);if (len >= 2 && str[len - 2] == '\r' && str[len - 1] == '\n')str[len - 2] = '\0';
}// 示例: 读取纯文本内容 (同步方式)
void str_test(const char *url)
{char *response;long status;BSTR method_bstr, url_bstr, response_bstr;HRESULT hr;IXMLHTTPRequest *xhr;VARIANT async; // VARIANT代表一个弱类型的变量VARIANT null;hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLHTTPRequest, &xhr);if (SUCCEEDED(hr)){// 打开连接// get参数内容是放到url字符串里面的method_bstr = convert_string_to_bstr("GET");url_bstr = convert_string_to_bstr(url);async.vt = VT_BOOL;async.boolVal = VARIANT_FALSE; // 选择非异步方式 (也就是同步方式)null.vt = VT_NULL;hr = IXMLHttpRequest_open(xhr, method_bstr, url_bstr, async, null, null); // 最后两个参数是用户名和密码, 不用填SysFreeString(method_bstr);SysFreeString(url_bstr);if (SUCCEEDED(hr))printf("open() succeeded\n");// 发送请求// 由于我们选择的是同步方式, 所以send函数要等到所有数据都接收完了才返回hr = IXMLHttpRequest_send(xhr, null); // 第二个参数是post数据内容, 不用管if (SUCCEEDED(hr))printf("send() succeeded\n");// 获取请求状态// 0: open函数还没调用// 1: send函数还没调用// 2: send函数调用了, get_status()函数已经可以使用了, 也可以读http headers了, 但是还读不了response// 3: 已收到了部分数据, 可以读responseBody和responseText// 4: 所有数据都已经接收完了, responseBody和responseText已经是完整数据了hr = IXMLHttpRequest_get_readyState(xhr, &status);if (SUCCEEDED(hr))printf("ready state: %d\n", status);// 获取http返回的response codehr = IXMLHttpRequest_get_status(xhr, &status);if (SUCCEEDED(hr))printf("http status code: %d\n", status);// 读取并显示文本内容hr = IXMLHTTPRequest_get_responseText(xhr, &response_bstr);if (SUCCEEDED(hr)){// response_bstr是utf16编码, 转换后的response是gb2312编码// 把函数里面的CP_ACP改成CP_UTF8就可以转成utf8编码, 但是utf8就没办法用printf打印, 只能用fwrite写入txt文件再用记事本打开查看response = convert_bstr_to_string(response_bstr);remove_last_crlf(response);printf("response: %s\n", response);free(response);SysFreeString(response_bstr);}IXMLHttpRequest_Release(xhr);}printf("\n");
}// 直接显示xml代码
void display_xmlstr(IXMLDOMDocument *xmldoc)
{char *str;BSTR bstr;HRESULT hr;hr = IXMLDOMDocument_get_xml(xmldoc, &bstr);if (SUCCEEDED(hr)){str = convert_bstr_to_string(bstr);remove_last_crlf(str);printf("xmlstr: %s\n", str);free(str);SysFreeString(bstr);}
}// 显示xml属性值
void display_attr(IXMLDOMElement *elem, const char *name)
{char *value;BSTR bstr; // 表示一个带长度前缀的utf16编码的字符串VARIANT variant; // 表示一个弱类型的变量bstr = convert_string_to_bstr(name);IXMLDOMElement_getAttribute(elem, bstr, &variant);SysFreeString(bstr); // 使用完字符串后必须释放if (variant.vt == VT_BSTR){value = convert_bstr_to_string(variant.bstrVal);printf("%s: %s\n", name, value);free(value);SysFreeString(variant.bstrVal); // 保存在Variant中的BSTR也必须释放掉}else if (variant.vt == VT_NULL)printf("%s: (null)\n", name);
}// 检查xml节点名称是否为指定名称
int check_node_name(IXMLDOMNode *node, const char *expected_name)
{char *name;int ret;BSTR bstr;IXMLDOMNode_get_nodeName(node, &bstr);name = convert_bstr_to_string(bstr);SysFreeString(bstr);ret = (strcmp(name, expected_name) == 0);free(name);return ret;
}// 检查xml属性值是否为指定值
int check_node_attr(IXMLDOMNode *node, const char *attr, const char *expected_value)
{char *value;int ret = 0;BSTR bstr;HRESULT hr;IXMLDOMElement *elem;VARIANT variant;IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, &elem); // 先把node转成elementbstr = convert_string_to_bstr(attr);hr = IXMLDOMElement_getAttribute(elem, bstr, &variant); // 再通过element获取属性值SysFreeString(bstr);if (SUCCEEDED(hr)){if (variant.vt == VT_BSTR){value = convert_bstr_to_string(variant.bstrVal);ret = (strcmp(value, expected_value) == 0);SysFreeString(variant.bstrVal);free(value);}else if (variant.vt == VT_NULL)ret = (expected_value == NULL || expected_value[0] == '\0');}IXMLDOMElement_Release(elem);return ret;
}void display_xml(IXMLDOMDocument *xmldoc)
{char *text;long i, len;BSTR text_bstr;HRESULT hr;IXMLDOMElement *root;IXMLDOMNode *node;IXMLDOMNodeList *list;IXMLDOMDocument_get_documentElement(xmldoc, &root);display_attr(root, "timestr");display_attr(root, "example");display_attr(root, "title");hr = IXMLDOMElement_get_childNodes(root, &list);if (SUCCEEDED(hr)){IXMLDOMNodeList_get_length(list, &len);for (i = 0; i < len; i++){IXMLDOMNodeList_get_item(list, i, &node);if (check_node_name(node, "param") && check_node_attr(node, "method", "get") && check_node_attr(node, "name", "txt")){IXMLDOMNode_get_text(node, &text_bstr);text = convert_bstr_to_string(text_bstr);printf("txt: %s\n", text);free(text);SysFreeString(text_bstr);}else if (check_node_name(node, "param") && check_node_attr(node, "method", "post") && check_node_attr(node, "name", "hello")){IXMLDOMNode_get_text(node, &text_bstr);text = convert_bstr_to_string(text_bstr);printf("hello: %s\n", text);free(text);SysFreeString(text_bstr);}IXMLDOMNode_Release(node);}IXMLDOMNodeList_Release(list);}IXMLDOMElement_Release(root);
}// 示例: 读取xml内容 (同步方式)
void xml_test(const char *url, const char *post_data)
{long status;BSTR method_bstr, url_bstr, header_bstr, value_bstr;HRESULT hr;IDispatch *disp;IXMLDOMDocument *xmldoc;IXMLHTTPRequest *xhr;VARIANT async;VARIANT body;VARIANT null;hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLHTTPRequest, &xhr);if (SUCCEEDED(hr)){if (post_data == NULL)method_bstr = convert_string_to_bstr("GET");elsemethod_bstr = convert_string_to_bstr("POST");printf("method: %ls\n", method_bstr); // 不带中文的BSTR可直接用%ls输出, 带中文就不行了url_bstr = convert_string_to_bstr(url);async.vt = VT_BOOL;async.boolVal = VARIANT_FALSE;null.vt = VT_NULL;hr = IXMLHttpRequest_open(xhr, method_bstr, url_bstr, async, null, null);SysFreeString(method_bstr);SysFreeString(url_bstr);if (SUCCEEDED(hr))printf("open() succeeded\n");if (post_data == NULL)body = null;else{header_bstr = convert_string_to_bstr("Content-Type");value_bstr = convert_string_to_bstr("application/x-www-form-urlencoded");IXMLHttpRequest_setRequestHeader(xhr, header_bstr, value_bstr);SysFreeString(header_bstr);SysFreeString(value_bstr);body.vt = VT_BSTR;body.bstrVal = convert_string_to_bstr(post_data);}hr = IXMLHttpRequest_send(xhr, body);if (SUCCEEDED(hr))printf("send() succeeded\n");if (body.vt == VT_BSTR){SysFreeString(body.bstrVal);body = null;}hr = IXMLHttpRequest_get_readyState(xhr, &status);if (SUCCEEDED(hr))printf("ready state: %d\n", status);hr = IXMLHttpRequest_get_status(xhr, &status);if (SUCCEEDED(hr))printf("http status code: %d\n", status);hr = IXMLHTTPRequest_get_responseXML(xhr, &disp);if (SUCCEEDED(hr)){printf("get_responseXML() succeeded\n");hr = IDispatch_QueryInterface(disp, &IID_IXMLDOMDocument, &xmldoc);if (SUCCEEDED(hr)){printf("IDispatch_QueryInterface() succeeded\n");display_xmlstr(xmldoc);display_xml(xmldoc);IXMLDOMDocument_Release(xmldoc);}elseprintf("IDispatch_QueryInterface() failed\n");IDispatch_Release(disp);}elseprintf("get_responseXML() failed");IXMLHttpRequest_Release(xhr);}printf("\n");
}int main()
{CoInitializeEx(NULL, COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE); // 初始化COM组件对象模型str_test("http://adv.purasbar.com/mcu/test/str_test.php");str_test("http://adv.purasbar.com/mcu/test/json_test.php");xml_test("http://adv.purasbar.com/mcu/test/xml_test.php?txt=xp%E7%B3%BB%E7%BB%9F%E4%B8%8B%E8%BD%BD&yaya=%E5%96%9D%E5%AE%8C%E8%8D%AF%E6%84%9F%E8%A7%89%E6%B8%85%E7%88%BD%E4%BA%86%E8%AE%B8%E5%A4%9A", NULL); // get方式xml_test("http://adv.purasbar.com/mcu/test/xml_test.php?txt=xp%E7%B3%BB%E7%BB%9F%E4%B8%8B%E8%BD%BD", "hello=%E6%B5%8B%E8%AF%95%E7%8E%AF%E5%A2%83%E5%90%8E%E5%8F%B0%E7%AE%A1%E7%90%86%E7%B3%BB%E7%BB%9F&yaya=%E9%9B%BE%E9%9B%A8%20%E8%BF%9E%E9%98%B4%E5%A4%A9%20%E7%BE%BD%E7%BE%BD%20%E6%B5%85%E6%B5%85%E5%81%A5%E5%81%A5%E5%BA%B7%E5%BA%B7%20%E9%BB%9B%E8%A1%8D%20%E9%9F%B3%E9%9F%B3%E9%94%A6%E9%B2%A4%E9%99%84"); // post方式CoUninitialize();return 0;
}

get参数的提交方法:
将参数直接附加到URL网址的后面。如http://XXX//xml_test.php?A=B&C=D&E=F。汉字和符号必须用urlencode函数编码成%XX的形式。
post数据的提交方法:
通过VARIANT弱类型变量传入IXMLHttpRequest_send函数的第二个参数。没有post数据时要传入VT_NULL值。
汉字和符号同样也必须用urlencode函数编码成%XX的形式。只是最前面不需要以问号开头。

关于urlencode/urldecode、htmlspecialchars、trim和str_replace函数的C语言实现,请参考这篇文章:

https://blog.csdn.net/ZLK1214/article/details/131748124icon-default.png?t=N7T8https://blog.csdn.net/ZLK1214/article/details/131748124

程序运行结果:

open() succeeded
send() succeeded
ready state: 4
http status code: 200
response: 当前时间为2024年1月24日 20:37:39。open() succeeded
send() succeeded
ready state: 4
http status code: 200
response: {"date":"2024-1-24 20:37:40","time":1706099860,"desc":"abcd\u7b80\u4f5
3\u4e2d\u6587"}method: GET
open() succeeded
send() succeeded
ready state: 4
http status code: 200
get_responseXML() succeeded
IDispatch_QueryInterface() succeeded
xmlstr: <?xml version="1.0"?>
<test timestr="20240124203740" example="简体中文"><param method="get" name="txt"
>xp系统下载</param><param method="get" name="yaya">喝完药感觉清爽了许多</param><
/test>
timestr: 20240124203740
example: 简体中文
title: (null)
txt: xp系统下载method: POST
open() succeeded
send() succeeded
ready state: 4
http status code: 200
get_responseXML() succeeded
IDispatch_QueryInterface() succeeded
xmlstr: <?xml version="1.0"?>
<test timestr="20240124203740" example="简体中文"><cookie name="PHPSESSID">ktvth
t5ebhkssu0k5srknmpic3</cookie><param method="get" name="txt">xp系统下载</param><
param method="post" name="hello">测试环境后台管理系统</param><param method="post
" name="yaya">雾雨 连阴天 羽羽 浅浅健健康康 黛衍 音音锦鲤附</param></test>
timestr: 20240124203740
example: 简体中文
title: (null)
txt: xp系统下载
hello: 测试环境后台管理系统请按任意键继续. . .

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/427603.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JS进阶-深入对象(二)

拓展&#xff1a;深入对象主要介绍的是Js的构造函数&#xff0c;实例成员&#xff0c;静态成员&#xff0c;其中构造函数和Java种的构造函数用法相似&#xff0c;思想是一样的&#xff0c;但静态成员和实例成员和java种的有比较大的差别&#xff0c;需要认真理解 • 创建对象三…

el-select选择之后值不显示在文本框的问题解决

问题场景如下图&#xff1a; 在el-collapse-item中使用子组件&#xff0c;子组件里是el-form-item代码。el-select在for循环中&#xff0c;可以有多个。 查了一下博客&#xff0c;有的说这种场景需要给el-select添加change事件&#xff0c;加上 this.$forceUpdate() 强制刷新即…

【单例模式】保证线程安全实现单例模式

&#x1f4c4;前言&#xff1a;本文是对经典设计模式之一——单例模式的介绍并讨论单例模式的具体实现方法。 文章目录 一. 什么是单例模式二. 实现单例模式1. 饿汉式2. 懒汉式2.1 懒汉式实现单例模式的优化&#xff08;一&#xff09;2.2 懒汉式实现单例模式的优化&#xff08…

仿真机器人-深度学习CV和激光雷达感知(项目2)day04【简单例程】

文章目录 前言简单例程运行小海龟仿真启动节点查看计算图发布 Topic调用 Serviece 用 Python 发布和接收 Topic创建工作空间创建功能包&#xff0c;编译编写 Topic Publisher 节点编写 Topic Subscriber 节点运行节点 自定义消息类型用 Python 注册和调用 Serviece新建功能包在…

【若依】关于对象查询list返回,进行业务处理以后的分页问题

1、查询对象Jglkq返回 list&#xff0c;对 list 进行业务处理后返回&#xff0c;但分页出现问题。 /*** 嫁功率考勤查询*/RequiresPermissions("hr:kq:list")PostMapping("/list")ResponseBodypublic TableDataInfo list(Jglkq jglkq) throws ParseExcepti…

Spring基于dynamic-datasource实现MySQL多数据源

目录 多数据源实现 引入依赖 yml配置文件 业务代码 案例演示 多数据源实现 引入依赖 <dependency><groupId>com.baomidou</groupId><artifactId>dynamicdatasourcespringbootstarter</artifactId><version>3.5.0</version> &…

[Python] glob内置模块介绍和使用场景(案例)

Unix glob是一种用于匹配文件路径的模式&#xff0c;它可以帮助我们快速地找到符合特定规则的文件。在本文中&#xff0c;我们将介绍glob的基本概念、使用方法以及一些实际应用案例。 glob介绍 Glob(Global Match)是Unix和类Unix系统中的一种文件名扩展功能&#xff0c;它可以…

【python文件】生成的csv文件没两行数据之间有一个空行

问题描述 用python代码将数据写入csv文件&#xff0c;但生成的csv文件没两行数据之间有一个空行&#xff0c;如下图所示&#xff1a; 解决办法 在open函数中添加newline&#xff0c;如以下代码所示&#xff0c;即可解决这一问题。 with open(r"C:\Users\xxx\Desktop\DR…

未来已来:AI引领智能时代的多领域巨变

大家好&#xff0c;今天我们将深入探讨人工智能如何彻底改变我们的生活方式&#xff0c;领略未来的无限可能性。 1. 医疗革新&#xff1a;AI担任超级医生 医疗领域是AI最引人注目的战场之一。智能医学影像诊断系统&#xff0c;不仅能够精准识别病变&#xff0c;还能辅助医生提…

使用人工智能助手 Github Copilot 进行编程 02

本章涵盖了 在您的系统上设置 Python、VS Code 和 Copilot引⼊ Copilot 设计流程Copilot 的价值在于基本的数据处理任务本章将帮助您在自己的计算机上开始使用 Copilot,并熟悉与其的交互方式。在设置好Copilot 后,我们将要求您尽可能跟随我们的示例进行操作。实践是最好的学习…

从零开始训练 YOLOv8最新8.1版本教程说明(包含Mac、Windows、Linux端 )同之前的项目版本代码有区别

从零开始训练 YOLOv8 - 最新8.1版本教程说明 本文适用Windows/Linux/Mac:从零开始使用Windows/Linux/Mac训练 YOLOv8 算法项目 《芒果 YOLOv8 目标检测算法 改进》 适用于芒果专栏改进 YOLOv8 算法 文章目录 官方 YOLOv8 算法介绍改进网络代码汇总第一步 配置环境1.1 系列配…

动态IP代理与静态IP代理:详细区别与比较全析

动态代理IP和静态代理IP在跨境业务中具有非常广泛的实用性&#xff0c;但仍然有非常多小白选手并不清楚什么场景适合用哪一类IP&#xff0c;哪一中代理IP类型更适合你&#xff1f;其实他们各有其优点和缺点&#xff0c;为了使您的网络营销、社媒推广、跨境电商运营、网络抓取尽…