当前位置:首页 » 编程语言 » c语言中tcb
扩展阅读
webinf下怎么引入js 2023-08-31 21:54:13
堡垒机怎么打开web 2023-08-31 21:54:11

c语言中tcb

发布时间: 2022-08-20 00:59:14

㈠ 这是老司机来解答

对土壤要求不严,沙土、

等~你《 8 8 Еeγ 、 c o m~》

粘土和微酸性土壤均能

生长,但以土层深厚、肥沃疏松

的土壤最为适宜。象草再生能力强,生长迅速,

故对肥料要求较高,需施用大量有机肥和氮肥。象草在水分、温度适宜的条件下,一般种植后7--10天即可出苗,15--20天开始分蘖。肥地分蘖多,瘠薄地分蘖少;雨季分蘖多,旱季分蘖少。所以只有水肥充足,才能获得高产。

㈡ linux 线程控制块tcb中包含了哪些内容

进程控制块(Process Control Block,PCB),是操作系统内核中一种数据结构,主要表示进程状态,它是对系统的进程进行管理的重要依据,和进程管理相关的操作无一不用到PCB中的内容。虽各实际情况不尽相同,PCB通常记载进程之相关信息。

㈢ [Linux]编写一个简单的c语言程序,编写Makefile文件。

八 环境变量
8.1 查看环境变量
$ env  显示所有的环境变量设置
$ echo $ENV_VARIABLE  显示指定环境变量的设置
例:
$ echo $PATH
/bin:/etc:/usr/bin:/tcb/bin

8.2 设定环境变量
$ ENV_VARIABLE=XXX;export ENV_VARIABLE
例:
$ PATH=$PATH:$INFORMIXDIR/bin;export PATH  将环境变量PATH设定为原PATH值+$INFORMIXDIR/bin

8.3 取消环境变量设置
$ unset $ENV_VARIABLE
例:
$ set GZJ=gzj;export GZJ  设置环境变量GZJ
$ echo $GZJ
gzj  显示环境变量值
$ unset $GZJ  取消环境变量GZJ的设置
$ echo $GZJ
 已取消

一 makefile规则
makefile是一个make的规则描述脚本文件,包括四种类型行:目标行、命令行、宏定义行和make伪指令行(如“include”)。makefile文件中注释以“#”开头。当一行写不下时,可以用续行符“\”转入下一行。
1.1 目标行
目标行告诉make建立什么。它由一个目标名表后面跟冒号“:”,再跟一个依赖性表组成。
例:
example: depfile deptarget
该目标行指出目标example与depfile和deptarget有依赖关系,如果depfile或deptarget有修改,则重新生成目标。
example1 example2 example3: deptarget1 deptarget2 depfile
该目标行指出目标名表中的example1、example2、example3这三个各自独立的目标是用相同的依赖列表和规则生成的。
clean:
空的依赖列表说明目标clean没有其他依赖关系。

目标行后续的以Tab 开始的行是指出目标的生成规则,该Tab字符不能以空格代替。例如:
example.o:example.c example.h
cc –c example.c
该例子指出目标example.o依赖于example.c和example.h。如果example.c或example.h其中之一改变了,就需要执行命令cc –c example.c重新生成目标example.o。
可以用文件名模式匹配来自动为目标生成依赖表,如:
prog: *.c

以下是一个简单的makefile的例子:

图 1 最简单的makefile例
make使用makefile文件时,从第一个目标开始扫描。上例中的第一个目标为all,所以目标clean不会自动被执行,可以通过命令make clean来生成目标。

1.2 命令行
命令行用来定义生成目标的动作。
在目标行中分号“;”后面的文件都认为是一个命令,或者一行以Tab制表符开始的也是命令。
如在上面的makefile例中,第三行以Tab字符开始的cc命令即是一个命令行,说明要生成hello应执行的命令。也可以写成:hello:hello.o;cc –c hello –L…
一般情况下,命令行的命令会在标准输出中回显出来,如对上面的makefile执行make时,标准输出如下:
cc -c hello.c
cc -o hello -L/usr/X11R6/lib -L/usr/lib -lXm -lXt -lX11 hello.o
cc -c hello1.c
cc -o hello1 -L/usr/X11R6/lib -L/usr/lib -lXm -lXt -lX11 hello1.o
如果不希望命令本身回显,可在命令前加@字符,如在上例中不希望回显cc –c hello.c和cc –c hello1.c,可修改makefile文件如下:

图 2 抑制回显的makefile例
对该makefile文件执行make时,标准输出如下:
cc -o hello -L/usr/X11R6/lib -L/usr/lib -lXm -lXt -lX11 hello.o
cc -o hello1 -L/usr/X11R6/lib -L/usr/lib -lXm -lXt -lX11 hello1.o
可以看出,命令行前有@字符的不回显。

1.3 宏定义行
在makefile中,可以使用宏定义减少用户的输入,例如上例中对hello和hello1的编译选项均为“-L/usr/X11R6/lib -L/usr/lib -lXm -lXt -lX11”,此时可以用宏来代替,如:
图 3 使用宏定义的makefile例
宏定义的基本语法是:
name=value
在定义宏时,次序不重要。宏不需要在使用前定义。如果一个宏定义多次,则使用最后一次的定义值。
可以使用“$”字符和“()”或“{}”来引用宏,例如:
cc –o hello.o $(CCFLAGS) hello.o
也可以将一个宏赋值给另一个宏,但这样的定义不能循环嵌套,如:
A=value1
B=value2
C=$(A) $(B)等价于C=value1 value2

1.4 伪指令
makefile大部分由宏定义行、命令行和目标行组成。第四种类型是make伪指令行。make伪指令没有标准化,不同的make可能支持不同的伪指令集,使得makefile有一定的不兼容性。如果要考虑移植性问题,则要避免使用make伪指令。但有一些伪指令,如include,由于使用比较多,很多不同make都提供该伪指令。
1.4.1 伪指令include
该伪指令类似C语言中的#include,它允许一次编写常用的定义并包括它。include伪指令必须在一行中,第一个元素必须是include,并且跟一个要包含的文件名,如:
include default.mk

1.4.2 伪指令“#”
“#”字符也是make的伪指令,它指出“#”后面的文件是注释,如:
PROGNAME=test # define macro
#don't modify this

二 后缀规则
2.1 双后缀规则
在前面的makefile例中有许多重复内容,例如,生成hello和hello1的命令类似,生成hello.o和hello1.o的命令也类似,除了编译或链接的文件不一样外,其它均相同,这时,我们就可以使用后缀规则。首先看一个双后缀的例子:
图 4 使用双后缀规则的makefile例
后缀规则使用特殊的目标名“.SUFFIXES”。
第一行中.SUFFIXES的依赖表为空,用来清除原有的后缀规则,因为.SUFFIXES可以在makefile中多次使用,每一次都将新的后缀规则加入以前的后缀规则中。
第二行中指定后缀规则为“.c .o”,即表示将所有的.c文件转换为.o文件。
第三行指定将.c文件转换成.o文件的方法。$(CC)为make的预定义宏,其默认值为cc,$<为特殊的宏,代替当前的源文件,即所有要编译的.c文件。
第六行指定目标hello和hello1的生成方法。$@为特殊的宏,代替当前的目标名,即hello和hello1,[email protected]即为hello.o和hello1.o。

上例介绍的是双后缀规则,即它包含两个后缀,如.c.o,用来把一个C源文件编译为目标文件。双后缀规则描述如何由第一个后缀类型的文件生成第二个后缀类型的文件,例如:.c.o规则描述如何由.c文件生成.o文件。

2.2 单后缀规则
单后缀规则描述了怎样由指定后缀的文件生成由它基名为名字的文件。例如使用单后缀规则.c,可以由hello.c和hello1.c生成hello和hello1文件。例如将前面的makefile改为:
图 5 使用单后缀规则的makefile例
由于.c后缀规则为make标准后缀规则,make为其指定了相应的命令行,所以在makefile中可以不用再指定其目标生成的具体命令行。

下表是make提供的标准后缀规则。
表 1 make标准后缀规则
后缀规则 命令行
.c $(LINK.c) –o $@ $< $(LDLIBS)
.c.ln $(LINK.c) $(POUTPUT OPTPUT OPTION) –i $<
.c.o $(COMPILE.c) $(OUTPUT OPTION) $<
.c.a $(COMPILE.c) –o $% $<
$(AR) $(ARFLAGS) $@ $%
$(RM) $%

三 特殊目标
在后缀规则中使用了特殊目标.SUFFIXES,用来指定新增的后缀规则。make还提供了几个特殊目标来设置make的行为,下面为一些特殊的目标:
 .IGNORE
make在执行命令行时,如果返回的是错误码,make的缺省动作是停止并退出。增加该目标后,make将忽略命令行返回的错误码,并继续执行后续的操作。

 .SILENT
前面已经介绍过,make在执行命令行时会回显命令行内容,在命令行前增加“@”字符将抑制该命令行的回显。
如果增加该目标,所有的命令行不再回显,相当于在每个命令行前均增加了“@”字符。

 .PRECIOUS
当收到一个信号或从shell命令返回非零的错误码时,make删除它所有已建立的文件。但有些文件即使出了错误,用户也不想让make删除,这些文件可以作为.PRECIOUS目标的参数。它可以在一个makefile中出现多次,每一次都累积文件列表。

 .SUFFIXES
它为makefile指定新的后缀规则,新的后缀规则作为.SUFFIXES的依赖表给出。.SUFFIXES可以在一个makefile中多次使用,每一次都将新的后缀规则加入以前的后缀规则中,如果.SUFFIXES的依赖表为空,则设置后缀规则表为空。

四 特殊的宏
为简单使用规则,make提供了几个特殊的宏:
 $@
整个当前目标名的值可以由宏“$@”来代替。

 $<
当前的源文件由“$<”来代替。例如,在前面的例子中用到了$(CC) –c $<,其中的“$<”是所有要编译的.c文件。宏“$<”仅在后缀规则或.DEFAULT中有效。

 $*
当前目标的基名由宏“$*”来代替。例如目标的名字是hello.o,则基名就是除去了后缀.o的hello。

以上介绍的特殊宏使用了make自身的规则,用户不可以改变。下表介绍了C中预定义的宏。
用途 宏 默认值
库文档汇编命令 AR ar
ARFLAGS rv
AS as
ASFLAGS
COMPILE.s $(AS) $(ASFLAGS) $(TARGET ARCH)
C编译器命令 CC cc
CFLAGS
CPPFLAGS
COMPILE.c $(CC) $(CFLAGS) $(CPPFLAGS) $(TARGET ARCH) –c
LINK.c $(CC) $(CFLAGS) $(CPPFLAGS) $(LDFLAGS) $(TARGET ARCH)
链接编辑器命令 LD ld
LDFLAGS
rm命令 RM rm
后缀列表 SUFFIXES .o .c .c~ .s .s~ .S .S~ .ln .f .f~ .F .F~ .l .mod .mod~ .sym
.def .def~ .p .p~ .r .r~ .y .y~ .h .h~ .sh .sh~ .cps .cps~

五 makefile的应用
当调用make时,它在当前目录下搜索文件名是“makefile”或“Makefile”的文件,并执行。
如果不想使用上述缺省文件,可以使用命令行中的“-f”来指定文件,如将编写的makefile命名为mklib,则指定为“make –f mklib”。

㈣ C语言编单纯形法程序怎么写

#include<stdio.h>
#include<math.h>

intm;//记录约束条件方程组的个数
intn;//记录未知量的个数
floatM=1000000.0;
floatA[100][100];//用于记录方程组的数目和系数
floatC[100];//用于存储目标函数中各个变量的系数
floatb[100];//用于存储常约束条件中的常数
floatCB[100];//用于存储基变量的系数
floatseta[100];//存放出基与入基的变化情况
floatcn[100];//存储检验数矩阵
floatx[100];
intnum[100];//用于存放出基与进基变量的情况
floatZ=0;//记录目标函数值

voidshuru();
voidprint();
intmincz();
intfind_line(inta);
voidexchange(inta,intb);
intmain()
{
inti,j=0;
intp,q,temp;//q:换入,p:换出
shuru();
printf(" -------------------------------------------------------------------------- ");
printf(" CB XB b ");
for(i=0;i<n;i++)
printf("X(%d) ",i+1);
for(i=0;i<n;i++)
x[i]=0;
printf(" ");
while(1){
q=mincz();
if(q==-1){
print();
printf(" 所得解已经是最优解! ");
printf(" 最优解为: ");
for(j=0;j<m;j++){
temp=num[j]-1;
x[temp]=b[j];
}
for(i=0;i<n;i++){
printf("x%d=%.2f",i+1,x[i]);
Z=Z+x[i]*C[i];
}
printf("Z=%.2f",Z);
break;
}
print();
p=find_line(q);
printf(" p=%d,q=%d",p,q);
if(q==-1)break;
exchange(p,q);
}
return0;
}
intmincz()
{
inti,k=0;
intflag=0;//检验数标记
floatmin=0;
for(i=0;i<n;i++)
if(cn[i]>=0)
flag=1;
else{
flag=0;
break;
}
if(flag==1)
return-1;
//进行到此处,说明存在<0的检验数
//找到最小的检验数,作为换入变量
for(i=0;i<n;i++){
if(min>cn[i]){
min=cn[i];
k=i;
}
}
returnk;
}
intfind_line(inta)
{
inti,k,j;
intflag=0;
floatmin;
k=a;
for(i=0;i<m;i++)
if(A[i][k]<=0)
flag=1;
else{
flag=0;
break;
}
if(flag==1){
printf(" 该线性规划无最优解! ");
return-1;
}
for(i=0;i<m;i++){
if(A[i][k]>0)
seta[i]=b[i]/A[i][k];
elseseta[i]=M;
}
min=M;
for(i=0;i<m;i++){
if(min>=seta[i]){
min=seta[i];
j=i;
}
}
num[j]=k+1;
CB[j]=C[k];
returnj;
}
voidexchange(intp,intq)
{
inti,j,c,l;
floattemp1,temp2,temp3;
c=p;//行号,换出
l=q;//列号,换入
temp1=A[c][l];//A[c][l]主元
b[c]=b[c]/temp1;
for(j=0;j<n;j++)
A[c][j]=A[c][j]/temp1;//主元化为1
for(i=0;i<m;i++){
if(i!=c)
if(A[i][l]!=0){
temp2=A[i][l];
b[i]=b[i]-b[c]*temp2;
//主元所在列,其余元素化为0
for(j=0;j<n;j++)
A[i][j]=A[i][j]-A[c][j]*temp2;
}
}
temp3=cn[l];
for(i=0;i<n;i++)
cn[i]=cn[i]-A[c][i]*temp3;
}
voidprint()
{
inti,j=0;
printf(" -------------------------------------------------------------------------- ");
for(i=0;i<m;i++){
printf("%8.2f X(%d)%8.2f",CB[i],num[i],b[i]);
for(j=0;j<n;j++)
printf("%8.2f",A[i][j]);
printf(" ");
}
printf(" -------------------------------------------------------------------------- ");
printf(" ");
for(i=0;i<n;i++)
printf("%8.2f",cn[i]);
printf(" -------------------------------------------------------------------------- ");
}
voidshuru()
{
inti,j;//循环变量
intk;
printf("请输入线性规划问题的约束条件个数M:");
scanf("%d",&m);
printf("请输入线性规划问题的决策变量个数N:");
scanf("%d",&n);
printf(" 请输入方程组的系数矩阵A(%d行%d列): ",m,n);
for(i=0;i<m;i++)
for(j=0;j<n;j++)
scanf("%f",&A[i][j]);
printf(" 请输入初始基变量的数字代码矩阵: ");
for(i=0;i<m;i++)
scanf("%d",&num[i]);
printf(" 请输入方程组右边的值矩阵b: ");
for(i=0;i<m;i++)
scanf("%f",&b[i]);
printf(" 请输入目标函数各个变量的系数所构成的系数阵C: ");
for(i=0;i<n;i++)
scanf("%f",&C[i]);
for(i=0;i<n;i++)
cn[i]=-C[i];
for(i=0;i<m;i++){
k=num[i]-1;
CB[i]=C[k];
}
}

㈤ 案例3:某年8月3日,美国德克萨斯商业银行(TCB)将单据寄给信用证的开征行——中

问题不完整,请给出完整案例介绍,以便分析解答。

㈥ uC/os-ii在进行任务切换时,为什么SP(堆栈指针)不和其他寄存器一样压栈而是存放在TCB(任务控制块)中

uC/OS II的源码中,OS_TCB结构体的定义中并没有单独保存SP的成员,而是由结构体中的OSTCBStkPtr指针指向的任务堆栈保存着任务切换需要保存的CPU寄存器内容。uC/OS会给每个创建的任务分配一个TCB,TCB中包含了该任务的全部信息。


(6)c语言中tcb扩展阅读:

uC/os-ii工作原理:

1,要实现多任务机制,那么目标CPU必须具备一种在运行期更改PC的途径,否则无法做到切换。不幸的是,直接设置PC指针,还没有哪个CPU支持这样的指令。但是一般CPU都允许通过类似JMP,CALL这样的指令来间接的修改PC。

2,事实上,在使用CALL指令或者软中断指令来修改PC,主要是软中断。但在一些CPU上,并不存在软中断这样的概念,所以,我们在那些CPU上,使用几条PUSH指令加上一条CALL指令来模拟一次软中断的发生。

3,在uC/OS-II里,每个任务都有一个任务控制块(Task Control Block),这是一个比较复杂的数据结构。在任务控制块的偏移为0的地方,存储着一个指针,它记录了所属任务的专用堆栈地址。

4,事实上,在uC/OS-II内,每个任务都有自己的专用堆栈,彼此之间不能侵犯。这点要求程序员在它们的程序中保证。一般的做法是把它们申明成静态数组。而且要申明成OS_STK类型。当任务有了自己的堆栈,那么就可以将每一个任务堆栈在那里记录到前面谈到的任务控制快偏移为0的地方。

5,以后每当发生任务切换,系统必然会先进入一个中断,这一般是通过软中断或者时钟中断实现。然后系统会先把当前任务的堆栈地址保存起来,仅接着恢复要切换的任务的堆栈地址。由于哪个任务的堆栈里一定也存的是地址,这样,就达到了修改PC为下一个任务的地址的目的。

㈦ 线程控制块tcb中包含了哪些内容

进程控制块(Process
Control
Block,PCB),是操作系统内核中一种数据结构,主要表示进程状态,它是对系统的进程进行管理的重要依据,和进程管理相关的操作无一不用到PCB中的内容。虽各实际情况不尽相同,PCB通常记载进程之相关信息。
第一种描述
[1]
(1)进程状态:可以是new、ready、running、waiting或halted等。
(2)程序计数器:接着要运行的指令地址。
(3)CPU寄存器:如累加器、索引寄存器(Index
Register)、堆栈指针以及一般用途寄存器、状况代码等,主要用途在于中断时暂时存储数据,以便稍后继续利用;其数量及类因计算机架构有所差异。
(4)CPU排班法:优先级、排班队列等指针以及其他参数。
(5)存储器管理:如分页表(Page
Table)等。
(6)会计信息:如CPU与实际时间之使用数量、时限、帐号、工作或进程号码。
(7)输入输出状态:配置进程使用I/O设备,如磁带机。
第二种描述[2]
(1)进程标识符(内部,外部)
(2)处理机的信息:通用寄存器,指令计数器,PSW,用户的栈指针。
(3)进程调度信息:进程状态,进程的优先级,进程调度所需的其它信息,事件。
(4)进程控制信息:程序的数据的地址,资源清单,进程同步和通信机制,链接指针。

㈧ (OS_TCB *)0怎么理解

关于对(OS_TCB *)0是否是指向空的任务块的理解:
(OS_TCB *)0虽然可以看成指向了一个起始地址为0的任务块,但是实际一块只是作为一个标志。如果您对链表等数据结构有一些接触的话会更容易理解。在内核有关TCB构成的链表中,用(OS_TCB *)0指向该链表已经到达尾部,或者链表为空。
同样的,(OS_TCB *)1,也是作为一个标志。这个值,我想在内核中应该是用在OSPrioTable[]相关的操作代码中。将OSPrioTable[ X] 置为(OS_TCB *)1,这样就表明该任务优先级已经被占用。因为其它的代码是通过检查OSPrioTable[x] 是否等于(OS_TCB *)0来判定对应的优先级是否已经使用。