RT-Thread 实时操作系统核心是一个高效的硬实时核心,它具备非常优异的实时性、稳
定性、可剪裁性,当进行最小配置时,内核体积可以到 3k ROM 占用、 1k RAM 占用。

RT-Thread 中的“线程”一般由三部分组成:线程代码(函数)、 线程控制块、 线程堆栈。

/* 指向线程控制块的指针*/

static struct rt_thread led_thread= RT_NULL;

void led_thread_entry(void* parameter)
{
rt_uint8_t count=;
rt_hw_led_init();
while ()
{
  if (++count>=) count = ;
  rt_hw_led_on(ledTable[count][]);
  rt_hw_led_off(ledTable[count][]);
  rt_hw_led_off(ledTable[count][]);
  rt_hw_led_off(ledTable[count][]);
  rt_thread_delay( RT_TICK_PER_SECOND/ );
  /* sleep 0.5 second and switch to other thread */
}
}

上面即是一个典型的线程代码结构—无限死循环,当然还有一种线程结构是顺序执行的,比如初始化线程,它执行到 return(),就会返回,当其返回后,系统会在 idle 线程中将其删除,从而使其退出调度队列。一般情况下用户线程都将是一个无限循环结构。

 线程堆栈:
static rt_uint8_t led_stack[ 512];
线程堆栈是一段连续的内存块,当线程切换后,为了满足线程切换和响应中断时保存cpu 寄存器中的内容及任务调用其它函数时的准备,每个线程都要配有自己的堆栈.

创建一个我们自己的线程
前面说了这么多,我们还是来自己建立一个线程,这样的话印象更深刻。
RT-Thread 中的线程分为静态线程—线程堆栈由编译器静态分配,使用 rt_thread_init ()
函数创建和动态线程—线程堆栈由系统动态分配,使用 rt_thread_create()函数创建。/* 静态线程的 线程堆栈*/

static rt_uint8_t led1_stack[];
/* 静态线程的 线程控制块 */
static struct rt_thread led1_thread;
void demo_thread_creat(void)
{
rt_err_t result;
/* 动态线程的 线程控制块指针 */
rt_thread_t led2_thread;
rt_hw_led_init();
/* 创建静态线程 : 优先级 20 ,时间片 2 个系统滴答 */
result = rt_thread_init( &led1_thread,
                   "led1",
           static_thread_entry,
                    RT_NULL,
      (rt_uint8_t*)&led1_stack[],
              sizeof(led1_stack),
                       20,
                       2);
if (result == RT_EOK)
  {
    rt_thread_startup(&led1_thread);
  }
  /* 创建动态线程 : 堆栈大小 512 bytes ,优先级 21 ,时间片 2 个系统滴答 */
  led2_thread = rt_thread_create("led2",
dynamic_thread_entry,
                    RT_NULL,
                      ,
                       21,
                       );
  if (led2_thread != RT_NULL)
   rt_thread_startup(led2_thread);
}

静态线程 VS 动态线程
从上例可看出,静态、动态线程在做同样的事情时,从效果上看,是没有任何差别的!
那么,我们在实际中如何抉择?使用静态线程时,必须先定义静态的线程控制块,并且定义好堆栈空间,然后调用rt_thread_init() 来完成线程的初始化工作。采用这种方式,线程控制块和堆栈占用的内存会放在 RW/ZI 段,这段空间在编译时就已经确定,它不是可以动态分配的,所以不能被释放,而只能使用 rt_thread_detach() 函数将该线程控制块从对象管理器中脱离。使用动态定义方式 rt_thread_create() 时, RT-Thread 会动态申请线程控制块和堆栈空间。在编译时, 编译器是不会感知到这段空间的,只有在程序运行时, RT-Thread 才会从系统堆中申请分配这段内存空间,当不需要使用该线程时,调用 rt_thread_delete() 函数就会将这段申请的内存空间重新释放到内存堆中。这两种方式各有利弊,静态定义方式会占用 RW/ZI 空间,但是不需要动态分配内存,运行时效率较高,实时性较好。 动态方式不会占用额外的 RW/ZI 空间,占用空间小,但是
运行时需要动态分配内存,效率没有静态方式高。 总的来说,这两种方式就是空间和时间效率的平衡,可以根据实际环境需求选择采用具体的分配方式。

static void rt_hw_led_init(void);
static void static_thread_entry(void* parameter);
static void dynamic_thread_entry(void* parameter); /* 变量分配4字节对齐 */
ALIGN(RT_ALIGN_SIZE) /* 静态线程的 线程堆栈*/
static rt_uint8_t led1_stack[]; /* 静态线程的 线程控制块 */
static struct rt_thread led1_thread; void demo_thread_creat(void)
{
rt_err_t result;
/* 动态线程的 线程控制块指针 */
rt_thread_t led2_thread; rt_hw_led_init(); /* 创建静态线程 : */
result = rt_thread_init(&led1_thread,
"led1",
static_thread_entry,
RT_NULL,
(rt_uint8_t*)&led1_stack[],
sizeof(led1_stack),
, //优先级 20 ,
);//时间片 2个系统滴答
if (result == RT_EOK)
{
rt_thread_startup(&led1_thread);
} /* 创建动态线程 : 堆栈大小512 bytes ,优先级 21 ,时间片 2个系统滴答 */
led2_thread = rt_thread_create("led2",
dynamic_thread_entry,
RT_NULL,
,
,
); if (led2_thread != RT_NULL)
rt_thread_startup(led2_thread); /*上面代码中的 20、 8 既是线程的优先级, 5 是为线程所分配的时间片。这里需要注意的
是, 当一个线程的优先级独一无二的时候,它的时间片这个参数将失去作用,我们不要认为
上面的两个线程运行完 5 个系统 ticks 后就会主动交出 cpu 使用权,当运行完 5 个 ticks
后如果它不需等待任何资源,也不主动让出 cpu 使用权的话,它还会继续运行,时间片这个
参数只在具有相同优先级的线程之间起作用,可是即便如此,这个参数也不能设为 0,因为
你不知道后续是否还会创建线程。*/ }

由于我们的线程一般都是一个无限循环,而 RT-Thread 又是一个抢占式的内核,所以为了使高优先级的线程不至于独占 CPU,可以给其他优先级较低的线程获得CPU使用权的机会,我们往往需要在线程中的合适位置调用系统函数 rt_thread_delay(),使当前线程的运行延时一段时间并进行一次任务调度,以让出 CPU 的使用权。

RT-Thread创建静态、动态线程的更多相关文章

  1. RT-thread线程创建:动态线程与静态线程

    本文介绍了如何创建一个动态线程和一个静态线程 RT-thread版本:RT-thread system 3.1.0 开发环境:MDK5 为了编程方便,创建了sample1.c文件,然后添加到工程中 话 ...

  2. C/C++ Muti-Thread多线程编程学习(之)线程Thread | 创建、运行、结束

    文章目录 前言 线程 Thread 创建线程 CreateThread _beginthread _beginthreadex pthread_create 线程运行 结束线程 前言   多线程(Mu ...

  3. Linux命令之ar - 创建静态库.a文件和动态库.so

    转自:http://blog.csdn.net/eastonwoo/article/details/8241693 用途说明 创建静态库.a文件.用C/C++开发程序时经常用到,但我很少单独在命令行中 ...

  4. 在Linux中创建静态库.a和动态库.so

    转自:http://www.cnblogs.com/laojie4321/archive/2012/03/28/2421056.html 在Linux中创建静态库.a和动态库.so 我们通常把一些公用 ...

  5. Xcode 创建静态库和动态库

    1.linux中静态库和动态库区别: 库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种. 静态库:这类库的名字一般是libxxx.a:利用静态函数库编译成的文件 ...

  6. 使用Thread类可以创建和控制线程

    1.创建线程 static void Main(string[] args) { /* Thread类 * 创建控制线程 * 其构造函数接受ThreadStart和ParameterizedThrea ...

  7. 在Linux中创建静态库和动态库

    我们通常把一些公用函数制作成函数库,供其它程序使用. 函数库分为静态库和动态库两种. 静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库. 动态库在程序编译时并不会被连接到目标代码中 ...

  8. linux中创建静态库和动态库

    1. 函数库有两种:静态库和动态库. 静态库在程序编译的时候会被连接到目标代码中,程序运行时将不再需要改静态库. 动态库中程序编译的时候并不会连接到目标代码中,而是在程序运行时才被载入,因此在程序运行 ...

  9. [非技术参考]C#基础:使用Thread创建线程(1)

    Thread类可以创建和控制线程,Thread类的构造函数重载为接受ThreadStart和ParameterizedThreadStart类型的委托参数.下面我们用一个例子来解释怎样用Thread类 ...

随机推荐

  1. C#对XML进行操作(添加、修改)

    XML文档内容如下: <?xml version="1.0" encoding="utf-8"?> <root> <first i ...

  2. hdu 1312:Red and Black(DFS搜索,入门题)

    Red and Black Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  3. Laravel框架数据库CURD操作、连贯操作

    这篇文章主要介绍了Laravel框架数据库CURD操作.连贯操作.链式操作总结,本文包含大量数据库操作常用方法,需要的朋友可以参考下 一.Selects 检索表中的所有行 $users = DB::t ...

  4. GitHub在Visual Studio 2015中获得TFS/VSO同等地位

    (此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 在Visual Studio 2015中微软为GitHub提供了扩展插件,从而让GitHub ...

  5. Java 反射机制及Annotation

    转自:http://justjavac.iteye.com/blog/714654 Java 反射是 Java 语言的一个很重要的特征. 它允许运行中的 Java 程序对自身进行检查,并能直接操作程序 ...

  6. SparkStreaming+Flume出现ERROR ReceiverTracker: Deregistered receiver for stream 0: Error starting receiver 0 - org.jboss.netty.channel.ChannelException

    文章发自http://www.cnblogs.com/hark0623/p/4204104.html ,转载请注明 我发现太多太多的坑要趟了… 向yarn提交sparkstreaming了,提交脚本如 ...

  7. Xamarin.iOS模拟器调试找不到资源文件

    Xamarin.iOS模拟器调试找不到资源文件 在Visual Studio 2015中,运行Xamarin.iOS项目,出现找不到资源文件的错误.错误信息:System.IO.FileNotFoun ...

  8. Chart系列(一):Chart的基本元素

    如何使用一个Chart,则首先必须要了解其组织结构,其次知道其API. Chart元素 首先,来看看Chart组成元素. Axis Label:坐标轴标签   Axis Title:坐标轴标题   C ...

  9. LightOJ1057 Collecting Gold(状压DP)

    这道题可以想到几点: 整个行程可以看作一次次的行走,每次行走都是用最短的路程从某一非空点到达另外一非空点: 两点间最少的步数是二者x和y坐标差的最大值: 返回原点这个过程,肯定是取完最后一个黄金后直接 ...

  10. BZOJ2707 : [SDOI2012]走迷宫

    首先求出SCC缩点,E[T]=0,按拓扑序计算 对于无边连出的块,如果不是T所在块,则称该块是死路块 对于一个块,如果其中的点连出的边是死路块,则它也是死路块 否则对于每块进行高斯消元求出期望 如果S ...