从Android面试题目溯源-1、创建线程有那几种方式

概念

程序执行流的最小单位,处理器调度调度和分派的基本单位。

如何理解这个概念

如下图,可以简单类比吉他,六根弦代表六个线程,每个线程独立且单独运行,且持有上一个音的状态,每根手指可类比为一个CPU的核心,在Arm架构中,大小核架构可以类比吉他的高底音弦,震动频率高的1,2,3弦为大核,震动频率相对较低的4,5,6弦为小核(之后针对Arm架构单独讨论)

Java的创建方式

  1. 继承Thread
  2. 实现Runnable
  3. 匿名内部类实现Runnable接口
  4. 使用Callable和Future创建有返回值的线程

延伸

Dart如何创建并管理线程

Dart线程模型基于事件循环(event loop)和异步编程的理念,通过Steam和Future类型来支持异步编程。真正意义上的线程被称为Isolate,对应Java意义上的线程,我们可以简单类比Android的Handler-Looper机制为展开版本的async, await,Future异步模型,两者都是基于事件循环,同样的模式,在kotlin等其他语言中可能被称为协程

Rust如何创建并管理线程

  1. 使用std::thread::spawn函数
  2. 使用std::thread::Builder

Linux编程中线程的概念

当我们将目光深入到Linux系统中,我们会发现Linux系统使用轻量级进程(LWP,Lightweight Process)的线程模型来实现多线程,怎么来理解这句话,首先是和进程类似,或者说包含进程的性质的同时,又是轻量级,从具体实现上,我们知晓轻量是指,共享进程的资源(地址空间,文件描述符和其他操作系统资源)。

  • pthread_create创建线程
  • pthread_exit退出线程
  • pthread_cancel取消线程

我们从Linux的编程的概念当中可以看出,虚拟机线程的实现是依赖于操作系统的实现,大部分情况下,语言创建线程对应操作系统的线程,他们有对应关系。

Linux操作系统如何创建线程

计算机组成原理中,我们知晓计算机基本构成,CPU负责计算,内存负责存储计算数据,在Arm架构中,RISC(精简指令集)使用的便是load/save内存操作模型,计算机数据特定的约定可以解释为特定的含义。我们带着这个观点去查看一下Linux使用Syscall(系统调用)创建线程的过程。

c 复制代码
#define _GNU_SOURCE
#include <sched.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>

void child_function(void *arg) {
    // 新线程执行的代码
    printf("Child thread: %s\n", (char *)arg);
}

int main() {
    // 创建一个新的栈空间
    char stack[4096];
    
    // 使用 clone 系统调用创建新线程
    int flags = CLONE_VM | CLONE_FS | CLONE_FILES | CLONE_SIGHAND | CLONE_THREAD | CLONE_SYSVSEM | CLONE_SETTLS;
    pid_t child_pid = clone(child_function, stack + sizeof(stack), flags, "Hello from child thread");

    if (child_pid == -1) {
        perror("clone");
        exit(EXIT_FAILURE);
    }

    // 主线程执行的代码
    printf("Main thread\n");

    // 等待新线程结束
    waitpid(child_pid, NULL, 0);

    return 0;
}

我们看到通过系统调用clone, 我们得到一个新的,栈大小为4096个字节的线程,新线程栈底部方法是child_function, 其中,创建线程的参数flag 表示共享的资源,例如虚拟内存空间CLONE_VM

总结

回到面试题,线程的创建和管理是开辟一个新的方法栈空间,执行新的函数调用。最初的原因是 IO性能远远低于CPU处理性能,在客户端上的体现是不耗时操作不阻塞主线程(UI线程)给用户提供流畅的体验,近些年来,协程的出现,本质上是Event Loop + 异步编程,轻量级的协程,优化的事件队列,比如Flutter实现的EventLoop由两部分组成。

  • 事件队列(Event Queue),存储异步事件
  • 微任务队列(Microtask Queue)存储立即执行的异步事件

这些方式都非常适合客户端提高流畅性和响应速度,可以多使用协程等类似的手段处理小并发问题,性能上会优于使用线程。

相关推荐
玩镜的码农小师兄40 分钟前
[从零开始面试算法] (04/100) LeetCode 136. 只出现一次的数字:哈希表与位运算的巅峰对决
c++·算法·leetcode·面试·位运算·hot100
绝无仅有1 小时前
面试真实经历某商银行大厂计算机网络问题和答案总结
后端·面试·github
绝无仅有1 小时前
面试真实经历某商银行大厂系统,微服务,分布式问题和答案总结
后端·面试·github
paishishaba2 小时前
JAVA面试复习笔记(待完善)
java·笔记·后端·面试
聪明的笨猪猪11 小时前
Java JVM “调优” 面试清单(含超通俗生活案例与深度理解)
java·经验分享·笔记·面试
兩尛13 小时前
Spring面试
java·spring·面试
恋红尘15 小时前
Redis面试八股
数据库·redis·面试
hanxiaozhang201817 小时前
Netty面试重点-2
面试·netty
怪兽20141 天前
请例举 Android 中常用布局类型,并简述其用法以及排版效率
android·面试
南一Nanyi1 天前
才知道 DNS 还能基于 HTTPS 实现!
网络协议·安全·面试