linux
linux有没有线程id确认函数?
一、linux有没有线程id确认函数?
linux C中,获取当前进程id 函数为getpid() ; 头文件:#include 函数原型:pid_t getpid(void); 函数说明:getpid ()用来取得目前进程的进程id,许多程序利用取到的此值来建立临时文件, 以避免临时文件相同带来的问题。 返回值:目前进程的进程id 范例 #include #include main() { printf("pid=%d\n", getpid()); } 执行: pid=1494 /*每次执行结果都不一定相同 */
二、linux多线程详解?
1.进程是操作系统分配资源的基本单位。而线程通俗来讲就是一个进程中一个执行流。
2.这里以串行与并行下载文件举例,如果我们使用串行的方式去下载多个文件,那么得到的结果是,将这些文件逐个按个的下载,即上一个下载完成之后才会下载接下来的文件。
3.如果使用并行的方式下载,那么这些文件就会一次同时下载多个文件,而不是等待上一个下载完后才继续下载接下来的,大大的提高了下载效率。
三、Linux多线程通信?
PIPE和FIFO用来实现进程间相互发送非常短小的、频率很高的消息;
这两种方式通常适用于两个进程间的通信。
共享内存用来实现进程间共享的、非常庞大的、读写操作频率很高的数据(配合信号量使用);这种方式通常适用于多进程间通信。
其他考虑用socket。这里的“其他情况”,其实是今天主要会碰到的情况:分布式开发。
在多进程、多线程、多模块所构成的今天最常见的分布式系统开发中,socket是第一选择
。消息队列,现在建议不要使用了 ---- 因为找不到使用它们的理由。在实际中,我个人感觉,PIPE和FIFO可以偶尔使用下,共享内存都用的不多了。在效率上说,socket有包装数据和解包数据的过程,所以理论上来说socket是没有PIPE/FIFO快,不过现在计算机上真心不计较这么一点点速度损失的。你费劲纠结半天,不如我把socket设计好了,多插一块CPU来得更划算。另外,进程间通信的数据一般来说我们都会存入数据库的,这样万一某个进程突然死掉或者整个服务器死了,也不至于丢失重要数据、便于回滚到之前的状态。从这个角度考虑,适用共享内存的情况也更少了,所以socket使用得更多。再多说一点关于共享内存的:共享内存的效率确实高,但它的重点在“共享”二字上。如果的确有好些进程共享一大块数据(如果把每个进程都看做是类的对象的话,那么共享数据就是这个类的static数据成员),那么共享内存就是一个不二的选择了。但是在面向对象的今天,我们更多的时候是多线程+锁+线程间共享数据。因此共享进程在今天使用的也越来越少了。不过,在面对一些极度追求效率的需求时,共享内存就会成为唯一的选择,比如高频交易系统。除此以外,一般是不需要特意使用共享内存的。另外,PIPE和共享内存是不能跨LAN的
(FIFO可以但FIFO只能用于两个进程通信)。
如果你的分布式系统随着需求的增加而越来越大所以你想把不同的模块放在不同机器上而你之前开发的时候用了PIPE或者共享内存,那么你将不得不对代码进行大幅修改......同时,即使FIFO可以跨越LAN,其代码的可读性、易操作性和可移植性、适应性也远没有socket大。这也就是为什么一开始说socket是第一选择的原因。最后还有个信号简单说一下。请注意,是信号,不是信号量。
信号量是用于同步线程间的对象的使用的(建议题主看我的答案,自认为比较通俗易懂:semaphore和mutex的区别? - Linux - 知乎
)。信号也是进程间通信的一种方式。比如在Linux系统下,一个进程正在执行时,你用键盘按Ctrl+c,就是给这个进程发送了一个信号。进程在捕捉到这个信号后会做相应的动作。虽然信号是可以自定义的,但这并不能改变信号的局限性:不能跨LAN、信息量极其有限
。在现代的分布式系统中,通常都是消息驱动:
即进程受到某个消息后,通过对消息的内容的分析然后做相应的动作。如果你把你的分布式系统设置成信号驱动的,这就表示你收到一个信号就要做一个动作而一个信号的本质其实就是一个数字而已。这样系统稍微大一点的话,系统将变得异常难以维护;甚至在很多时候,信号驱动是无法满足我们的需求的。因此现在我们一般也不用信号了。因此,请记住:除非你有非常有说服力的理由,否则请用socket。
顺便给你推荐个基于socket的轻量级的消息库:ZeroMQ。四、linux如何停止线程?
杀死线程 所在的进程就可以, ps aux | grep 进程名 kill -TERM 进程号 如果你指的写程序, 那就参考 man pthread_exit。
《Linux就该这么学》里有相关介绍,建议看看。
五、linux下C中怎么让才能安全关闭线程?
回答这个问题,首先得搞清楚线程关闭或者退出有哪些方式
线程的退出方式
如果进程中的任何线程调用exit,_Exit或_exit,则整个进程终止。 类似地,当信号的默认操作是终止进程时,发送到线程的信号将终止整个进程。单个线程可以有三种方式退出其控制流程,而不会终止整个进程。1线程可以简单地从线程处理程序中返回,返回值是线程的退出代码。
2该线程可以被同一进程中的另一个线程取消。
3该线程可以调用pthread_exi
线程退出的返回值
#include <pthread.h> void pthread_exit(void *rval_ptr);#include <pthread.h> int pthread_join(pthread_t thread, void **rval_ptr);
pthread_join函数的rval_ptr参数是无类型指针。进程中的其他线程可通过调用pthread_join函数来使用rval_ptr指针,调用它线程将阻塞,直到指定的线程调用pthread_exit或从其线程处理程序中返回或被取消。如果只是从其线程处理程序返回,则rval_ptr将包含返回码。如果线程被取消,则rval_ptr指定的内存位置设置为PTHREAD_CANCELED。
通过调用pthread_join,自动会将加入的线程放置在分离状态,如果线程已处于分离状态,则pthread_join可能会失败,返回EINVAL。如果我们对线程的返回值不感兴趣,我们可以将rval_ptr设置为NULL。在这种情况下,调用pthread_join允许我们等待指定的线程,但不去检索线程的终止状态。
下图显示了如何从已终止的线程中获取退出代码
运行结果:
lj@lj-PC:~$ ./ptest
thread 1 returning
thread 2 exiting
thread 1 exit code 1
thread 2 exit code 2
线程如何取消
一个线程可以通过调用pthread_cancel函数请求取消同一进程中的另一个。
#include <pthread.h> int pthread_cancel(pthread_t tid);
在默认情况下,pthread_cancel将使tid指定的线程的行为就像它使用PTHREAD_CANCELED参数调用pthread_exit一样。 但是,线程可以选择忽略或以其他方式控制取消的方式。 请注意,pthread_cancel不会等待线程终止。
线程可以安排函数在退出时被调用,这些函数称为线程清理处理程序。 可以为一个线程建立多个清理处理程序。 处理程序记录在堆栈中,这意味着它们的执行顺序与它们注册的顺序相反。
#include <pthread.h> void pthread_cleanup_push(void (*rtn)(void *), void *arg);void pthread_cleanup_pop(int execute);
当线程执行以下操作之一时,pthread_cleanup_push函数会被调用
调用pthread_exit
回复取消请求
使用非零执行参数调用pthread_cleanup_pop
如果execute参数设置为零,则不会调用cleanup函数。 在任何一种情况下,pthread_cleanup_pop都会删除最后一次调用pthread_cleanup_push所建立的清理处理程序。
下图举例如何使用线程清理处理程序。
运行结果:
lj@lj-PC:~$ ./pclean
thread 1 start
thread 1 push complete
thread 2 start
thread 1 exit code 1
thread 2 push complete
cleanup: thread 2 second handler
cleanup: thread 2 first handler
thread 2 exit code 2
从输出中,我们可以看到两个线程都正常启动并退出,但只调用了第二个线程的清理处理程序。因此,如果线程是通过其处理函数直接返回而终止,则不会调用其清理处理程序,不过此行为在具体平台实现之间会有所不同。另请注意,清理处理程序的调用顺序与安装它们的顺序相反。
如果我们在FreeBSD或Mac OS X上运行相同的程序,我们会发现该程序会导致段错误。发生这种情况是因为在这些系统上,pthread_cleanup_push实现为在堆栈上存储某些上下文的宏。当线程1在对pthread_cleanup_push的调用和对pthread_cleanup_pop的调用之间返回时,堆栈被覆盖,并且这些平台在调用清理处理程序时尝试使用此(已损坏的)上下文。在Single UNIX Specification中,在对pthread_cleanup_push和pthread_cleanup_pop的一对匹配调用之间返回会导致未定义的行为。在这两个函数之间返回的唯一可移植方法是调用pthread_exit。
线程和进程的类似操作
从上文我们可以看到线程和进程的相似之处,见如下表格:
讲了这么多,还有好多细节没有讲到,只要详细的了解了这些细节,我相信关于你的这个问题“linux下C中怎么让才能安全关闭线程”自然就有了答案。
六、linux最大线程数?
默认情况下:
主线程+辅助线程 +<253个自己的线程
含主线程和一个辅助线程,最多255个,即一个用户只能生成253个线程。
1、总结系统限制有:
/proc/sys/kernel/pid_max #查系统支持的最大线程数,一般会很大,相当于理论值
/proc/sys/kernel/thread-max
max_user_process(ulimit -u) #系统限制某用户下最多可以运行多少进程或线程
/proc/sys/vm/max_map_count
硬件内存大小
七、linux线程默认栈多大?
linux的线程栈大小可以使用ulimit -s查看,对于ubuntu 2.6的内核线程栈的默认大小为8M
八、linux怎么指定线程库?
大概的介绍一下Linux 的指定CPU运行,包括进程和线程。linux下的top命令是可以查看当前的cpu的运行状态,按1可以查看系统有多少个CPU,以及每个CPU的运行状态。 可是如何查看线程的CPU呢?
top -Hp pid,pid就是你当前程序的进程号,如果是多线程的话,是可以查看进程内所有线程的CPU和内存使用情况。
pstree可以查看主次线程,同样的pstree -p pid。可以查看进程的线程情况。
taskset这个其实才是重点,可以查看以及设置当前进程或线程运行的CPU(设置亲和力)。
taskset -pc pid,查看当前进程的cpu,当然有的时候不只是一个,taskset -pc cpu_num pid ,cpu_num就是设置的cpu。 这样的话基本的命令和操作其实大家都知道了,接下来就是在代码中完成这些操作,并通过命令去验证代码的成功率。 进程制定CPU运行:
[cpp] view plain copy #include #include #include #include #include #define __USE_GNU #include #include #include int main(int argc, char* argv[]) { //sysconf获取有几个CPU int num = sysconf(_SC_NPROCESSORS_CONF); int created_thread = 0; int myid; int i; int j = 0; //原理其实很简单,就是通过cpu_set_t进行位与操作 cpu_set_t mask; cpu_set_t get; if (argc != 2) { printf("usage : ./cpu num\n"); exit(1); } myid = atoi(argv[1])
; printf("system has %i processor(s). \n", num)
; //先进行清空,然后设置掩码 CPU_ZERO(&mask); CPU_SET(myid, &mask)
; //设置进程的亲和力 if (sched_setaffinity(0, sizeof(mask), &mask) == -1) { printf("warning: could not set CPU affinity, continuing...\n"); } while (1) { CPU_ZERO(&get); //获取当前进程的亲和力 if (sched_getaffinity(0, sizeof(get), &get) == -1) { printf("warning: cound not get cpu affinity, continuing...\n"); } for (i = 0; i < num; i++) { if (CPU_ISSET(i, &get)) { printf("this process %d is running processor : %d\n",getpid(), i); } } } return 0; } 进程设置CPU运行,其实只能是单线程。多线程设定CPU如下:
[cpp] view plain copy #define _GNU_SOURCE #include #include #include #include #include #include void *myfun(void *arg) { cpu_set_t mask; cpu_set_t get; char buf[256]; int i; int j; //同样的先去获取CPU的个数 int num = sysconf(_SC_NPROCESSORS_CONF); printf("system has %d processor(s)\n", num); for (i = 0; i < num; i++) { CPU_ZERO(&mask); CPU_SET(i, &mask); //这个其实和设置进程的亲和力基本是一样的 if (pthread_setaffinity_np(pthread_self(), sizeof(mask), &mask) < 0) { fprintf(stderr, "set thread affinity failed\n"); } CPU_ZERO(&get); if (pthread_getaffinity_np(pthread_self(), sizeof(get), &get) < 0) { fprintf(stderr, "get thread affinity failed\n"); } for (j = 0; j < num; j++) { if (CPU_ISSET(j, &get)) { printf("thread %d is running in processor %d\n", (int)pthread_self(), j); } } j = 0; while (j++ < 100000000) { memset(buf, 0, sizeof(buf)); } } pthread_exit(NULL); } int main(int argc, char *argv[]) { pthread_t tid; if (pthread_create(&tid, NULL, (void *)myfun, NULL) != 0) { fprintf(stderr, "thread create failed\n"); return -1; } pthread_join(tid, NULL); return 0; }
九、linux运行后有哪些线程?
就绪:线程分配了CPU以外的全部资源,等待获得CPU调度 执行:线程获得CPU,正在执行 阻塞:线程由于发生I/O或者其他的操作导致无法继续执行,就放弃处理机,转入线程就绪队列 挂起:由于终端请求,操作系统的要求等原因,导致挂起。
十、java 停止线程函数
在Java编程中,线程是一种关键的概念,它允许程序同时执行多项任务,提高了程序的效率和性能。然而,有时候我们需要停止一个正在运行的线程,这就需要使用停止线程函数来实现。
什么是停止线程函数?
停止线程函数是一种用于结束一个线程执行的机制。当线程完成其任务后,会自动停止,但有时候我们需要在特定条件下手动停止线程。这就需要使用停止线程函数来实现。
为什么需要停止线程函数?
在实际开发中,有时候线程可能因为某些原因进入死循环或长时间阻塞而无法自行停止。这时,我们就需要使用停止线程函数来手动终止线程以避免程序陷入不可控状态。
如何使用停止线程函数?
在Java中,停止线程函数的主要方法是使用interrupt()
方法来中断线程的执行。当调用interrupt()
方法时,线程会收到一个中断信号,然后可以在合适的时机自行结束执行。
另外,我们也可以通过设置标识位来控制线程是否继续执行,当需要停止线程时,只需要修改标识位即可。
停止线程函数的注意事项
在使用停止线程函数时,有一些注意事项需要注意:
- 确保线程处于可中断状态,即在合适的时机能够响应中断信号。
- 避免使用
stop()
方法来强制停止线程,这种方法可能会导致线程安全性问题。 - 合理处理线程停止后的清理工作,确保程序的稳定性和健壮性。
总结
停止线程函数在Java编程中扮演着重要的角色,它可以帮助我们有效地控制线程的执行,并避免程序出现不可控的情况。通过合理地运用停止线程函数,可以提高程序的稳定性和可靠性,是Java开发中不可或缺的一部分。
热点信息
-
在Python中,要查看函数的用法,可以使用以下方法: 1. 使用内置函数help():在Python交互式环境中,可以直接输入help(函数名)来获取函数的帮助文档。例如,...
-
一、java 连接数据库 在当今信息时代,Java 是一种广泛应用的编程语言,尤其在与数据库进行交互的过程中发挥着重要作用。无论是在企业级应用开发还是...
-
一、idea连接mysql数据库 php connect_error) { die("连接失败: " . $conn->connect_error);}echo "成功连接到MySQL数据库!";// 关闭连接$conn->close();?> 二、idea连接mysql数据库连...
-
要在Python中安装modbus-tk库,您可以按照以下步骤进行操作: 1. 确保您已经安装了Python解释器。您可以从Python官方网站(https://www.python.org)下载和安装最新版本...