- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试调试一些关于堆栈使用的代码。我做了下面的测试程序(只是作为一个例子来弄清楚 pthread 库是如何工作的):
#include <string.h>
#include <pthread.h>
#include <stdlib.h>
#include <assert.h>
#include <limits.h>
#include <stdio.h>
static void *threadFunc1(void *arg)
{
char arr[5000];
printf("Hello fromt threadFunc1 address of arr:%p\n", &arr);
return;
}
static void *threadFunc2(void *arg)
{
char arr[10000];
printf("Hello fromt threadFunc2 adress of arr:%p\n", &arr);
return;
}
int main(int argc, char *argv[])
{
pthread_t t1,t2;
pthread_attr_t thread_attr;
void *res;
int s;
size_t tmp_size=0;
s=pthread_attr_init(&thread_attr);
assert(s==0);
s=pthread_attr_setstacksize(&thread_attr , PTHREAD_STACK_MIN );
assert(s==0);
s=pthread_attr_getstacksize(&thread_attr , &tmp_size );
assert(s==0);
printf("forced stack size of pthread is:%zd\n", tmp_size);
printf("sizeof char is %zd\n", sizeof(char));
s = pthread_create(&t1, &thread_attr, threadFunc1, NULL);
assert(s==0);
sleep(1);
s = pthread_create(&t2, &thread_attr, threadFunc2, NULL);
assert(s==0);
sleep(1);
printf("Main done()\n");
exit(0);
}
当我执行它时,我得到以下输出(在我的 x86_64 Ubuntu 上):
forced stack size of pthread is:16384
sizeof char is 1
Hello fromt threadFunc1 address of arr:0x7fef350d3b50
Segmentation fault (core dumped)
当我进入我新创建的线程时,有没有办法知道请求的 PTHREAD_STACK_MIN 还剩多少?如果我在进入线程函数时更改 char 数组的大小,它似乎限制在 7000 到 8000 之间,这不是我所期望的(在 16384 附近的某个地方)。
最佳答案
玩弄线程堆栈大小不会带来很多“好”的东西。通常当出现这种情况时,是因为有人试图创建数百(或数千)个线程并遇到每个进程限制。这通常不是使用线程的好方法。示例:为服务器的每个传入连接请求创建一个新线程。
即使这不是您提问的原因,您也应该知道一些事情。
在生产(现实世界)代码中,不是一个简单的示例,您永远无法确定每个线程的特定“缩减”堆栈大小是 100% 安全的。您可以在给定的堆栈大小下对其进行广泛测试,然后也许一旦您在自己的测试环境中确信它“足够大”,可能会稍微增加它(可能 10-15%)以供生产使用。我不知道有什么特定工具可以告诉您运行时每个线程的峰值堆栈使用情况。最好完全避免这种情况。
减少每个线程的堆栈大小不会直接减少内存消耗,也不会提高性能。您不是通过试验来优化您的代码。
尽可能使用本地实现选择的默认堆栈大小要安全得多,直到您有真正的理由更改它。
您缺少一些原型(prototype)的头文件,缺少线程函数中的返回值等,以及其他一些轻微的清理工作。
此处运行的代码略有修改:
#include <stdio.h>
#include <string.h>
#include <pthread.h>
#include <stdlib.h>
#include <assert.h>
#include <unistd.h>
#include <limits.h>
static void *threadFunc1(void *arg)
{
char arr[5000];
printf("Hello fromt threadFunc1\n");
return NULL;
}
static void *threadFunc2(void *arg)
{
char arr[10000];
printf("Hello fromt threadFunc2\n");
return NULL;
}
int main(int argc, char *argv[])
{
pthread_t t1,t2;
pthread_attr_t thread_attr;
int s = 0;
size_t tmp_size=0;
s = pthread_attr_init(&thread_attr);
assert(s==0);
s = pthread_attr_setstacksize(&thread_attr , PTHREAD_STACK_MIN + 0x1000);
assert(s==0);
s = pthread_attr_getstacksize(&thread_attr , &tmp_size );
assert(s==0);
printf("forced stack size of pthread is:%zu\n", tmp_size);
s = pthread_create(&t1, &thread_attr, threadFunc1, NULL);
assert(s==0);
sleep(1); /* not the "right" way, but ok for a quick test */
s = pthread_create(&t2, &thread_attr, threadFunc2, NULL);
assert(s==0);
sleep(1);
printf("Main done()\n");
return 0;
}
注:我用过
$ gcc -O0 -Wall stack_t.c -o stack_t -pthread
./stack_t
如果您想尝试堆栈的其他大小,请注意某些实现要求堆栈大小是系统页面文件大小的倍数。显然,您也可以尝试大于 PTHREAD_STACK_MIN 的值,但请注意该潜在问题。
编辑:上面的示例将一个值添加到最小值,作为稍微增加它的示例。您可能需要添加更多。如果 0x1000 不够,请尝试 0x2000、0x3000 等,直到找到一个可用的。正如您所发现的,减少线程堆栈大小是很棘手的,并且也可能与平台有关。这就是为什么我上面有关于它不安全的警告。
关于linux - 我如何决定调用 pthread_attr_setstacksize 后可以使用多少堆栈?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15205734/
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,
Linux 管道可以缓冲多少数据?这是可配置的吗? 如果管道的两端在同一个进程中,但线程不同,这会有什么不同吗? 请注意:这个“同一个进程,两个线程”的问题是理论上的边栏,真正的问题是关于缓冲的。 最
我找到了here [最后一页] 一种有趣的通过 Linux 启动 Linux 的方法。不幸的是,它只是被提及,我在网上找不到任何有用的链接。那么有人听说过一种避免引导加载程序而使用 Linux 的方法
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
我试图了解 ld-linux.so 如何在 Linux 上解析对版本化符号的引用。我有以下文件: 测试.c: void f(); int main() { f(); } a.c 和 b.c:
与 RetroPie 的工作原理类似,我可以使用 Linux 应用程序作为我的桌面环境吗?我实际上并不需要像实际桌面和安装应用程序这样的东西。我只需要一种干净简单的方法来在 RaspberryPi 上
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 10 年前。 Improve thi
有什么方法可以覆盖现有的源代码,我应该用 PyQt、PyGTK、Java 等从头开始构建吗? 最佳答案 如果您指的是软件本身而不是它所连接的存储库,那么自定义应用程序的方法就是 fork 项目。据我所
我的情况是:我在一个磁盘上安装了两个 linux。我将第一个安装在/dev/sda1 中,然后在/dev/sda2 中安装第二个然后我运行第一个系统,我写了一个脚本来在第一个系统运行时更新它。
我在 i2c-0 总线上使用地址为 0x3f 的系统监视器设备。该设备在设备树中配置有 pmbus 驱动程序。 问题是,加载 linux 内核时,这个“Sysmon”设备没有供电。因此,当我在总线 0
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 11 年前。 Improve thi
我正试图在 linux 模块中分配一大块内存,而 kalloc 做不到。 我知道唯一的方法是使用 alloc_bootmem(unsigned long size) 但我只能从 linux 内核而不是
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
我有 .sh 文件来运行应用程序。在该文件中,我想动态设置服务器名称,而不是每次都配置。 我尝试了以下方法,它在 CentOS 中运行良好。 nohup /voip/java/jdk1.8.0_71/
我是在 Linux 上开发嵌入式 C++ 程序的新手。我有我的 Debian 操作系统,我在其中开发和编译了我的 C++ 项目(一个简单的控制台进程)。 我想将我的应用程序放到另一个 Debian 操
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 4 年前。 Improve this ques
我使用4.19.78版本的稳定内核,我想找到带有企鹅二进制数据的C数组。系统启动时显示。我需要在哪里搜索该内容? 我在 include/linux/linux_logo.h 文件中只找到了一些 Log
我知道可以使用 gdb 的服务器模式远程调试代码,我知道可以调试针对另一种架构交叉编译的代码,但是是否可以更进一步,从远程调试 Linux 应用程序OS X 使用 gdbserver? 最佳答案 当然
是否有任何可能的方法来运行在另一个 Linux 上编译的二进制文件?我知道当然最简单的是在另一台机器上重建它,但假设我们唯一能得到的是一个二进制文件,那么这可能与否? (我知道这可能并不容易,但我只是
我是一名优秀的程序员,十分优秀!