4

我们有大量使用 sprintf 和 cstrings 来构造文件名的“演示”代码。我想用 c++ 字符串替换它以允许非常长的文件名,并且因为它提供了更简洁的语法。

所以本质上我们需要像这样转换块

ofstream some_file;
char filename[100];
sprintf(filename,"%s/soln%i.dat",a.c_str(), b);
some_file.open(filename);

变成类似的东西

ofstream some_file((a + "soln" + to_string(b) + ".dat").c_str());

然而,因为这在很多地方都使用过,所以我想使用某种自动转换(即 sed 表达式、emacs 函数/宏、Visual Studio?等)。困难(我可以看到)是:

  1. sprintf 语句的内容可以是任何内容。(所以我们需要解析一个 sprintf 语句?)

  2. some_file 和 filename 可以在任何地方声明,可以命名为任何名称,并且经常被重用于输出到多个文件。(所以我们需要解析C++?)

这可能/可行吗?

对于加分,我们是否可以避免在字符串上调用 .c_str() (不更改 ofstream 构造函数)?

4

2 回答 2

2

使用Boost 格式库。它在 C++ 中实现 printf 类型的格式,生成 C++ 字符串作为输出。调用类似,但略有不同。你的例子是:

string fname = boost::format("%s/soln%i.dat") % a % b;
ofstream some_file(fname.c_str());

所以基本上,您需要识别字符缓冲区,并将其替换为字符串;识别 sprintf 语句,将其标记为参数(同时注意引号等),并使用 % 而不是 重写它,第一个参数包含在 boost::format 中。听起来有点吓人,但并非不可能。一个更大的问题是你的块可能并不都是那样的,通常识别这些情况会非常困难。

于 2012-10-29T15:43:41.980 回答
2

这里有一些更简单的东西,没有 Yassnippet:

(defun sprintf-to-ofstream (file)
  (interactive "sFile to print to: ")
  (back-to-indentation)
  (let* ((start (point))
         arglist
         sprintf-args
         ofstream-args
         (end 
          (save-excursion
            (move-end-of-line 1)
            (point)))
         (line (buffer-substring-no-properties start end)))
    (unless (string= (substring line 0 7) "sprintf")
      (error "No `sprintf' at this line"))
    (setq arglist
          (split-string
           (substring line (1+ (position ?\( line))
                      (position ?\) line :from-end t))
           "\\s-*,\\s-*")
          sprintf-args
          (split-string (substring (cadr arglist) 1 -1)
                        "%[^[:alpha:]%#]*[[:alpha:]%#]")
          ofstream-args
          (with-output-to-string
            (princ (concat "\"" (car sprintf-args) "\""))
            (setq sprintf-args (cdr sprintf-args))
            (dotimes (i (length sprintf-args))
              (princ " + ")
              (when (< (+ i 2) (length arglist))
                (princ (nth (+ i 2) arglist))
                (princ " + "))
              (princ (concat "\"" (nth i sprintf-args) "\"")))))
    (kill-region start end)
    (insert (concat "ofstream " file "((" ofstream-args ").c_str());" ))))

您可以将其绑定到您喜欢的任何键并像这样使用:

  • 将点移动到包含sprintf您要替换的行,例如通过执行M-%sprintf,然后调用此函数。它会提示你给它你想要打印消息的文件的名称(可能我可以让它更复杂,并向上搜索声明 ofstream 的地方,但如果你说它可能已经在随意的地方,这听起来像是浪费精力)。
  • RET,在此之后,该函数会将行更改为如下所示:
ofstream foo(("" + a.c_str() + "/soln" + b + ".dat").c_str());

给出你原来的例子。显然,如果您认为添加to_string()环绕参数的可能性更大,那么这很容易做到,只要付出更多努力,也很容易避免连接空字符串,但有时它可能会改变效果 /操作员的解释+,所以我决定保持这样。今天晚些时候,我将尝试将其制成一个 yassnippet 脚本,以便可以交互地跳转到被替换字符串中的不确定位置,并且也可以进行一些预定义的替换。

于 2012-10-30T09:37:23.817 回答