- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
所以我使用 HTML Agility 包用 C# 编写了一个网站抓取程序。这是相当简单的。即使考虑到网页格式的不一致,我仍然只花了几个小时就开始工作。
现在,我必须用 C 重新实现这个程序,以便它可以在 linux 环境中运行。这是一场重大的噩梦。
我能够拉回页面,但是当涉及到跟踪它以拉出我感兴趣的部分时 - 我画了很多空白。最初,除了使用 Tidy 和其他一些 XML 库之外,我一直试图在 C# 中实现类似于我的 HTML Agility 选项的解决方案,这样我可以或多或少地保持我的逻辑相同。
这不是很好。我有权访问的 XML 库似乎不支持 xpath,而且我无法安装支持的库。所以我求助于尝试找出一种方法来使用字符串匹配来阅读页面以找到我想要的数据。我不禁觉得必须有更好的方法来做到这一点。
这是我的:
#define HTML_PAGE "codes.html"
int extract()
{
FILE *html;
int found = 0;
char buffer[1000];
char searchFor[80], *cp;
html = fopen(HTML_PAGE, "r");
if (html)
{
// this is too error prone, if the buffer cuts off half way through a section of the string we are looking for, it will fail!
while(fgets(buffer, 999, html))
{
trim(buffer);
if (!found)
{
sprintf(searchFor, "<strong>");
cp = (char *)strstr(buffer, searchFor);
if(!cp)continue;
if (strncmp(cp + strlen(searchFor), "CO1", 3) == 0 || strncmp(cp + strlen(searchFor), "CO2", 3) == 0)
{
got_code(cp + strlen(searchFor));
}
}
}
}
fclose(html);
return 0;
}
got_code(html)
char *html;
{
char code[8];
char *endTag;
struct _code_st *currCode;
int i;
endTag = (char *)strstr(html, "</strong>");
if(!endTag)return;
sprintf(code, "%.7s", html);
for(i=0 ; i<Data.Codes ; i++)
if(strcasecmp(Data.Code[i].Code, code)==0)
return;
ADD_TO_LIST(currCode, _code_st, Data.Code, Data.Codes);
currCode->Code = (char *)strdup(code);
printf("Code: %s\n", code);
}
以上不能正常工作。我得到了很多我感兴趣的代码,但正如我上面提到的,如果缓冲区在错误的位置切断,我会错过一些。
我确实尝试过将我感兴趣的整个 html block 读入一个字符串,但我无法弄清楚如何循环它 - 我无法显示任何代码。
有谁知道我该如何解决这个问题?
编辑:我一直在考虑这个问题。有什么方法可以让我在文件中向前看并搜索我正在解析的每个文本“ block ”的末尾并将缓冲区大小设置为读取之前的大小?我需要另一个指向同一个文件的文件指针吗?这将(希望)防止缓冲区在不方便的地方切断的问题。
最佳答案
好吧,在多次用头撞墙试图想出一种方法让我的上述代码工作之后,我决定尝试一种稍微不同的方法。
因为我知道我正在抓取的页面上的数据包含在一个大行中,所以我更改了我的代码以搜索文件直到找到它。然后我继续寻找我想要的积木。这工作得非常好,一旦我让代码读取了一些 block ,就很容易进行小的修改以解决 HTML 中的不一致问题。花时间最长的部分是弄清楚一旦我到达行尾如何跳出,我通过提前达到峰值以确保还有另一个 block 要读取来解决这个问题。
这是我的代码(丑陋但实用):
#define HTML_PAGE "codes.html"
#define START_BLOCK "<strong>"
#define END_BLOCK "</strong>"
int extract()
{
FILE *html;
int found = 0;
char *line = NULL, *endTag, *startTag;
size_t len = 0;
ssize_t read;
char searchFor[80];
html = fopen(HTML_PAGE, "r");
if (html)
{
while((read = getline(&line, &len, html)) != -1)
{
if (found) // found line with codes we are interested in
{
char *ptr = line;
size_t nlen = strlen (END_BLOCK);
while (ptr != NULL)
{
sprintf(searchFor, START_BLOCK);
startTag = (char *)strstr(ptr, searchFor);
if(!startTag)
{
nlen = strlen (START_BLOCK);
ptr += nlen;
continue;
}
if (strncmp(startTag + strlen(searchFor), "CO1", 3) == 0 || strncmp(startTag + strlen(searchFor), "CO2", 3) == 0)
got_code(startTag + strlen(searchFor), code);
else {
nlen = strlen (START_BLOCK);
ptr += nlen;
continue;
}
sprintf(searchFor, END_BLOCK);
ptr = (char *)strstr(ptr, searchFor);
if (!ptr) { found = 0; break; }
nlen = strlen (END_BLOCK);
ptr += nlen;
if (ptr)
{
// look ahead to make sure we have more to pull out
sprintf(searchFor, END_BLOCK);
endTag = (char *)strstr(ptr, searchFor);
if (!endTag) { break; }
}
}
found = 0;
break;
}
// find the section of the downloaded page we care about
// the next line we read will be a blob containing the html we want
if (strstr(line, "wiki-content") != NULL)
{
found = 1;
}
}
fclose(html);
}
return 0;
}
got_code(char *html)
{
char code[8];
char *endTag;
struct _code_st *currCode;
int i;
endTag = (char *)strstr(html, "</strong>");
if(!endTag)return;
sprintf(code, "%.7s", html);
for(i=0 ; i<Data.Codes ; i++)
if(strcasecmp(Data.Code[i].Code, code)==0)
return;
ADD_TO_LIST(currCode, _code_st, Data.Code, Data.Codes);
currCode->Code = (char *)strdup(code);
printf("Code: %s\n", code);
}
虽然不如我的 C# 程序优雅或健壮,但至少它可以提取我想要的所有信息。
关于c - 如何使用 C 抓取网页?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26416673/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!