- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我构建了一个类似 printf 的可变参数函数,用于在自定义 CPU 上进行日志记录。
为了确保我没有内存错误,我编写了函数,以便它只接受指针并处理它们。预处理器总是在 va 列表的末尾添加一个 NULL 指针,因此只要函数重新识别零指针,它就会返回。这样,我就可以确定我没有阅读列表。我这样做是为了防止错误处理程序,它应该是一个安全的功能。无论用户向 va-list 中输入什么,都不应该以崩溃告终。
我现在唯一的问题是确保用户只能传递指向可变参数函数的指针,没有别的(除了不属于 va 列表的前两个参数)我的函数声明如下所示:
void LogNew( UINT LogClass, char* pMsg, ... );
和空终止:
#define LogNew(...) LogNew(__VA_ARGS__, 0)
有没有一种方法,也许是通过预处理器,来确保没有人使用按值调用而不是按引用调用。
提前致谢!
最佳答案
如果你想减少或防止错误格式的错误,我认为最好的办法是使用 fprintf
。它是一个可变参数函数,因此不安全,但它是用 C 打印内容的方式,因此程序员熟悉它及其缺点。如果您的日志记录功能只是一个宏,例如:
enum {Fatal, Error, Warning, Info};
int maxlevel = Error;
#define logmsg(L, ...) if (L > maxlevel); else { \
if (L == Fatal) fprintf(stderr, "Fatal: "); \
if (L == Error) fprintf(stderr, "Error: "); \
\
fprintf(stderr, __VA_ARGS__); \
fprintf(stderr, "\n"); \
\
if (L == Fatal) exit(1); \
} \
您可以获得静态分析的好处,它会警告您格式/类型不匹配。在 clang/gcc 中,-Wformat
是 -Wall
的一部分,将执行此操作。在 Visual Studio 中,您可以使用 /analyze
。 (当日志记录被抑制时,您不会生成 va 列表。)
如果您滚动自己的可变参数函数并调用 vfprintf
,您可以 detect bad format strings使用 clang/gcc format
属性或 Visual Studio 中的 _Printf_format_string_
注释。
就是说,如果您所有的变量参数都是同一类型,则可以使用数组。如果你的编译器支持 C99 的 compound literals ,您可以使用宏将参数列表的可变部分转换为数组。
假设所有打印的参数都应该是 const char *
。 (我知道这不是你想要的,但请耐心等待。)然后你可以实现一个打印函数,它可以接受任何正数的 C 字符串,如下所示:
void put(const char **s)
{
while (*s) {
fputs(*s++, stdout);
}
fputs("\n", stdout);
}
#define put(...) put((const char *[]){__VA_ARGS__, NULL})
然后像这样使用它:
put("Couldn't open \"", fn, "\".");
put("My name is ", (rand() % 2) ? "Igor" : "Tamara", ".");
put("");
您可以 extend this technique为 const void *
。我将代码隐藏在 Ideone 链接后面,因为使用 void
会使您丢失参数的类型信息。您的代码可能不会严重崩溃,因为指针应该指向某处,但是当您使用错误的格式时,您仍然会调用未定义的行为并获得垃圾输出。
(在对第一个答案的评论中,Lundin 指出将整数类型转换为指针是合法的。这不是所显示答案的缺陷,而是语言的缺陷。您也可以说 puts( 54)
并且只得到一个警告。)
关于c - 确保只将指针传递给可变参数函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47137818/
Github:https://github.com/jjvang/PassIntentDemo 我一直在关注有关按 Intent 传递对象的教程:https://www.javacodegeeks.c
我有一个 View ,其中包含自动生成的 text 类型的 input 框。当我单击“通过电子邮件发送结果”按钮时,代码会将您带到 CalculatedResults Controller 中的 Em
我有一个基本的docker镜像,我将以此为基础构建自己的镜像。我没有基础镜像的Dockerfile。 基本上,基本镜像使用两个--env arg,一个接受其许可证,一个选择在容器中激活哪个框架。我可以
假设我想计算 2^n 的总和,n 范围从 0 到 100。我可以编写以下内容: seq { 0 .. 100 } |> Seq.sumBy ((**) 2I) 但是,这与 (*) 或其他运算符/函数不
我有这个网址: http://www.example.com/get_url.php?ID=100&Link=http://www.test.com/page.php?l=1&m=7 当我打印 $_G
我想将 window.URL.createObjectURL(file) 创建的地址传递给 dancer.js 但我得到 GET blob:http%3A//localhost/b847c5cd-aa
我想知道如何将 typedef 传递给函数。例如: typedef int box[3][3]; box empty, *board[3][3]; 我如何将 board 传递给函数?我
我正在将一些代码从我的 Controller 移动到核心数据应用程序中的模型。 我编写了一个方法,该方法为我定期发出的特定获取请求返回 NSManagedObjectID。 + (NSManagedO
为什么我不能将类型化数组传递到采用 any[] 的函数/构造函数中? typedArray = new MyType[ ... ]; items = new ko.observableArray(ty
我是一名新的 Web 开发人员,正在学习 html5 和 javascript。 我有一个带有“选项卡”的网页,可以使网页的某些部分消失并重新出现。 链接如下: HOME 和 JavaScript 函
我试图将对函数的引用作为参数传递 很难解释 我会写一些伪代码示例 (calling function) function(hello()); function(pass) { if this =
我在尝试调用我正在创建的 C# 项目中的函数时遇到以下错误: System.Runtime.InteropServices.COMException: Operation is not allowed
使用 ksh。尝试重用当前脚本而不修改它,基本上可以归结为如下内容: `expr 5 $1 $2` 如何将乘法命令 (*) 作为参数 $1 传递? 我首先尝试使用“*”,甚至是\*,但没有用。我尝试
我一直在研究“Play for Java”这本书,这本书非常棒。我对 Java 还是很陌生,但我一直在关注这些示例,我有点卡在第 3 章上了。可以在此处找到代码:Play for Java on Gi
我知道 Javascript 中的对象是通过引用复制/传递的。但是函数呢? 当我跳到一些令人困惑的地方时,我正在尝试这段代码。这是代码片段: x = function() { console.log(
我希望能够像这样传递参数: fn(a>=b) or fn(a!=b) 我在 DjangoORM 和 SQLAlchemy 中看到了这种行为,但我不知道如何实现它。 最佳答案 ORM 使用 specia
在我的 Angular 项目中,我最近将 rxjs 升级到版本 6。现在,来自 npm 的模块(在 node_modules 文件夹内)由于一些破坏性更改而失败(旧的进口不再有效)。我为我的代码调整了
这个问题在这里已经有了答案: The issue of * in Command line argument (6 个答案) 关闭 3 年前。 我正在编写一个关于反向波兰表示法的 C 程序,它通过命
$(document).ready(function() { function GetDeals() { alert($(this).attr("id")); } $('.filter
下面是一个例子: 复制代码 代码如下: use strict; #这里是两个数组 my @i =('1','2','3'); my @j =('a','b','c'); &n
我是一名优秀的程序员,十分优秀!