您正在查看: 标签 操作系统 下的文章

使用国际站点

使用国际站点

在 Internet 或 Intranet 上发布消息的一个优点是您可以建立一个用户可以从不同国家(地区)访问的国际化 Web 站点。用户可以申请已被本地化为当地语言的网页,使用浏览器的本地化版本阅读。在建立一个包含有多种语言网页的 Web 站点时,需要转换在浏览器与 Web 服务器间或 ASP 脚本与 ActiveX 组件间传递的字符串。例如,如果一个日文浏览器在 HTTP 请求中发送了表格或查询字符串值,此字符串必须从浏览器的日文字符集转化为 ASP 用于处理脚本的字符集。

如果 Web 站点上的所有页都用 Web 服务器所使用的默认字符集书写,那么 ASP 将自动进行转换。但是,如果网页是用其他的字符集写的,就必须用 ASP 命令指明如何转换字符串。例如,如果您的站点既有日文字符集的网页,又有中文字符集的网页,就必须指明 ASP 处理某一特定页时所用的字符集。

ASP 也提供了支持不同地区文化习惯的命令,如货币、时间和日期格式。同字符串转换命令一样,如果您的脚本没有使用 Web 服务器的默认现场,则要使用现场命令。

设置字符串转换代码页

代码页是一个内部表,操作系统用它将符号(字母、数字和标点符号)映射为字符编号。不同的代码页支持不同国家(地区)所使用的字符集。代码页通过编号引用,例如,代码页 932 代表日文字符集,代码页 950 代表繁体中文字符集。

Active Server Pages 和 ActiveX 脚本引擎在内部使用 Unicode,这是一种 16 位固定长度的字符编码标准。如果您编写的所有页都使用 Web 服务器的默认代码页,则 ASP 将自动转换字符串。如果您的脚本不使用 Web 服务器的默认代码页,则应该指明所用的代码页以便字符串在 ASP 和脚本引擎间传递时能被正确转换。另外,您也可以为在浏览器与脚本间或 ActiveX 组件与脚本间传递的字符串指定代码页。

要为 ASP 页指定代码页,可使用 CODEPAGE 指令。例如,设置日文代码页,可使用以下指令:

<%@ CODEPAGE= 932 %>

当 ASP 处理这一页的内容和脚本时,将使用您指定的代码页来决定如何将脚本中的字符从您的脚本的字符集转化为 Unicode。例如,在 ANSI 中代表字母 "a" 的值将被转化为在 Unicode 中代表字母 "a" 的不同的值。

Active Server Pages 假定在服务器和浏览器间或脚本和 ActiveX 组件间传递的字符串的代码页与您为脚本设置的代码页是相同的。若要指定不同的代码页,可以设置 Session.CodePage 属性,从而覆盖 CODEPAGE 设置。例如,您使用 JIS 书写脚本,但却要响应使用 UTF-8 的客户程序,(JIS 和 UTF-8 是标准日文字符集的两种不同的字符编码)。

Session.CodePage 默认设置为 CODEPAGE 指令的值;对该属性的设置将覆盖当前的 CODEPAGE 设置。例如,要将代码页改为繁体中文,可使用如下命令:

<% Session.CodePage = 950 %>

如果您只是临时更改部分脚本的代码页,在此之后一定要将 Session.CodePage 设置为原来的值。以下脚本显示如何临时更改代码页:

<!-- Welcome to my home page in Japanese, code page 932 --!>
<%@ CodePage = 932 %>
...
<% Session("OriginalCodePage") = Session.CodePage %>
<!-- Look up name in Chinese, code page 950 --!>
<% Session.CodePage = 950 %>
<% Sender = ReadMailHeader("Sender") %>
<% Found = FindFriend("Sender") %>
<!-- Restore the original code page --!>
<% Session.CodePage =  Session("OriginalCodePage") %>
<% If Found == TRUE 
    ReplyWithPersonalizedForm() else
    ReplyWithBusinessForm()
%>

设置现场标识

现场是与用户语言相关的一组用户首选项信息。现场决定如何格式化日期和时间、项目;如何按字母排序以及如何比较字符串。现场标识 (LCID) 是唯一定义现场的 32 位数值。如果您没有为脚本指定不同的现场,那么 ASP 将使用 Web 服务器的默认现场。

要为 ASP 页设置现场标识,可使用 LCID 指令。例如,要设置日文现场,可使用下列现场 ID:

<%@ LCID =  1041 %>

LCID 指令告诉 ASP 书写脚本所用的现场。如果要更改脚本的输入或输出现场,可使用 Session.LCID 属性。例如,要将现场设置为标准法语,可使用如下命令:

<% Session.LCID = 1036 %>

Session.LCID 的默认设置为 LCID 指令的设置。在脚本中设置 Session.LCID 的值将覆盖默认设置。

重启动和关闭计算机

-->



实例65 重启动和关闭计算机

本实例介绍在Windows 2000环境下如何注销、重新启动和关闭计算机。

向窗体上添加一个TButton组件和3TRadioButton组件,其中TRadioButton组件用于选择程序的功能是注销、重新启动还是关闭计算机。设计完成的主界面如图4-9所示。

4-9 界面

Windows 2000操作系统中,如果在程序中直接调用ExitWindowsEx函数来完成重新启动计算机等系统级操作时,需要提升应用程序的操作权限,在本实例中,通过窗体的初始化过程来完成这一步骤:

procedure TForm1.FormCreate(Sender: TObject);

var

hToken : THandle;

Tkp : TTokenPrivileges;

Zero:DWORD;

begin

OpenProcessToken(GetCurrentProcess(),TOKEN_ADJUST_PRIVILEGES or

TOKEN_QUERY,hToken);

LookupPrivilegeValue(nil,'SeShutdownPrivilege',tkp.Privileges[0].Luid);

Tkp.PrivilegeCount :=1;

Tkp.Privileges[0].Attributes := SE_PRIVILEGE_ENABLED;

AdjustTokenPrivileges(hToken,False,Tkp,SizeOf(TTokenPrivileges),nil,Zero);

end;

执行以上代码,应用程序就可以获得注销、重新启动或关闭计算机的权限。

如果用户选择了程序需要完成的功能,单击“确定”按钮就可以完成用户指定的功能,响应代码如下:

procedure TForm1.btnOKClick(Sender: TObject);

begin

if self.RadioButton1.Checked then

begin

ExitWindowsEx(EWX_LOGOFF,0);

end;

if self.RadioButton2.Checked then

begin

ExitWindowsEx(EWX_REBOOT,0);

end;

if self.RadioButton3.Checked then

begin

ExitWindowsEx(EWX_SHUTDOWN or EWX_POWEROFF,0);

end;

end;

程序通过判断3TRadioButton组件的选中状态来设置ExitWindowsEx函数的第1个参数,如果该参数的第1个参数值为EWX_LOGOFF,那么就会完成注销的操作;如果为EWX_REBOOT,那么就会完成重新启动的操作;如果为EWX_SHUTDOWN or EWX_POWEROFF,那么就会关闭计算机。

程序代码如下:

unit Unit1;

interface

uses

Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,

Dialogs, StdCtrls;

type

TForm1 = class(TForm)

btnOK: TButton;

RadioButton1: TRadioButton;

RadioButton2: TRadioButton;

RadioButton3: TRadioButton;

procedure FormCreate(Sender: TObject);

procedure btnOKClick(Sender: TObject);

private

{ Private declarations }

public

{ Public declarations }

end;

var

Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);

var

hToken : THandle;

Tkp : TTokenPrivileges;

Zero:DWORD;

begin

OpenProcessToken(GetCurrentProcess(),TOKEN_ADJUST_PRIVILEGES or

TOKEN_QUERY,hToken);

LookupPrivilegeValue(nil,'SeShutdownPrivilege',tkp.Privileges[0].Luid);

Tkp.PrivilegeCount :=1;

Tkp.Privileges[0].Attributes := SE_PRIVILEGE_ENABLED;

AdjustTokenPrivileges(hToken,False,Tkp,SizeOf(TTokenPrivileges),nil,Zero);

end;

procedure TForm1.btnOKClick(Sender: TObject);

begin

if self.RadioButton1.Checked then

begin

ExitWindowsEx(EWX_LOGOFF,0);

end;

if self.RadioButton2.Checked then

begin

ExitWindowsEx(EWX_REBOOT,0);

end;

if self.RadioButton3.Checked then

begin

ExitWindowsEx(EWX_SHUTDOWN or EWX_POWEROFF,0);

end;

end;

end.

保存文件,然后按F9键运行程序,程序运行的初始画面如图4-10所示。

4-10 程序运行的初始画面

读者可以通过选择3TRadioButton组件来决定注销、重新启动还是关闭计算机。

读者通过这个实例不但可以学习到如何进行关闭计算机等系统级操作,也可以学会如何提升应用程序的操作权限,这一点在Windows 等操作系统中尤为关键。


实时的透明窗体



实例1 实时的透明窗体

界面是程序的面孔,很多应用程序的界面给用户留下了深刻的印象,如FlashGet的半透明窗体。本实例说明如何利用delphi制作一个Windows 2000下的透明窗体。

在窗体上分别添加一个TImageTTrackBar组件,前者用来显示一幅图片,后者用来调节窗体的透明度。添加组件后的窗体如图1-1所示。

1-1 添加组件后的窗体

为了达到实时的透明效果,在程序运行过程中用户可以通过调节TTrackBar组件上滑块的位置来设置窗体的透明程度。为此需要设置窗体的AlphaBlend属性为True,然后添加如下处理TTrackBar组件上滑块位置变化的代码:

procedure TfrmMain.TrackBar1Change(Sender: TObject);

begin

self.AlphaBlendValue:=self.TrackBar1.Max-self.TrackBar1.Position;

end;

AlphaBlend属性为True,通过改变窗体的AlphaBlendValue属性值即可设置窗体的透明度。

程序代码如下:

unit Unit1;

interface

uses

Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,

Dialogs, ComCtrls, ExtCtrls;

type

TfrmMain = class(TForm)

Image1: TImage;

TrackBar1: TTrackBar;

procedure FormCreate(Sender: TObject);

procedure TrackBar1Change(Sender: TObject);

private

{ Private declarations }

public

{ Public declarations }

end;

var

frmMain: TfrmMain;

implementation

{$R *.dfm}

procedure TfrmMain.FormCreate(Sender: TObject);

begin

self.AlphaBlend:=true;

self.TrackBar1.Min:=0;

self.TrackBar1.Max:=255;

self.TrackBar1.Frequency:=25;

end;

procedure TfrmMain.TrackBar1Change(Sender: TObject);

begin

self.AlphaBlendValue:=self.TrackBar1.Max-self.TrackBar1.Position;

end;

end.

选择“File|Save All”选项,在弹出的对话框中选择合适的文件名保存文件。然后按F9键运行程序,运行结果如图1-2所示。

值得注意的是,本程序在Windows 环境下不能够得到透明效果,因为AlphaBlend属性和AlphaBlendValue属性只支持Windows 及以后版本的操作系统。通过这个实例,读者不但可以体验到delphi的强大功能,还可以为自己的应用程序界面加上看起来很神奇的透明效果。

1-2 程序运行结果


1. 最简单的汇编程序

1. 最简单的汇编程序

例 18.1. 最简单的汇编程序

#PURPOSE: Simple program that exits and returns a
#     status code back to the linux kernel
#
#INPUT:   none
#
#OUTPUT:  returns a status code. This can be viewed
#     by typing
#
#     echo $?
#
#     after running the program
#
#VARIABLES:
#     %eax holds the system call number
#     %ebx holds the return status
#
 .section .data

.section .text
.globl _start
_start:
movl $1, %eax # this is the linux kernel command
# number (system call) for exiting
# a program

movl $4, %ebx # this is the status number we will
# return to the operating system.
# Change this around and it will
# return different things to
# echo $?

int $0x80 # this wakes up the kernel to run
# the exit command


把这个程序保存成文件hello.s(汇编程序通常以.s作为文件名后缀),用汇编器(Assembler)as把汇编程序中的助记符翻译成机器指令,生成目标文件hello.o:

$ as hello.s -o hello.o

然后用链接器(Linker,或Link Editor)ld把目标文件hello.o链接成可执行文件hello:

$ ld hello.o -o hello

为什么用汇编器翻译成机器指令了还不行,还要有一个链接的步骤呢?链接主要有两个作用,一是修改目标文件中的信息,对地址做重定位,在第 5.2 节 “可执行文件”详细解释,二是把多个目标文件合并成一个可执行文件,在第 2 节 “main函数和启动例程”详细解释。我们这个例子虽然只有一个目标文件,但也需要经过链接才能成为可执行文件。

现在执行这个程序,它只做了一件事就是退出,退出状态是4,第 2 节 “自定义函数”讲过在shell中可以用特殊变量$?得到上一条命令的退出状态:

$ ./hello
$ echo $?
4

所以这段汇编代码相当于在C程序的main函数中return 4;。为什么会相当呢?我们在第 2 节 “main函数和启动例程”详细解释。

下面逐行分析这个汇编程序。首先,#号表示单行注释,类似于c语言的//注释。

 .section .data

汇编程序中以.开头的名称并不是指令的助记符,不会被翻译成机器指令,而是给汇编器一些特殊指示,称为汇编指示(Assembler Directive)或伪操作(Pseudo-operation),由于它不是真正的指令所以加个“伪”字。.section指示把代码划分成若干个段(Section),程序被操作系统加载执行时,每个段被加载到不同的地址,操作系统对不同的页面设置不同的读、写、执行权限。.data段保存程序的数据,是可读可写的,相当于C程序的全局变量。本程序中没有定义数据,所以.data段是空的。

 .section .text

.text段保存代码,是只读和可执行的,后面那些指令都属于.text段。

 .globl _start

_start是一个符号(Symbol),符号在汇编程序中代表一个地址,可以用在指令中,汇编程序经过汇编器的处理之后,所有的符号都被替换成它所代表的地址值。在c语言中我们通过变量名访问一个变量,其实就是读写某个地址的内存单元,我们通过函数名调用一个函数,其实就是跳转到该函数第一条指令所在的地址,所以变量名和函数名都是符号,本质上是代表内存地址的。

.globl指示告诉汇编器,_start这个符号要被链接器用到,所以要在目标文件的符号表中标记它是一个全局符号(在第 5.1 节 “目标文件”详细解释)。_start就像C程序的main函数一样特殊,是整个程序的入口,链接器在链接时会查找目标文件中的_start符号代表的地址,把它设置为整个程序的入口地址,所以每个汇编程序都要提供一个_start符号并且用.globl声明。如果一个符号没有用.globl声明,就表示这个符号不会被链接器用到。

_start:

这里定义了_start符号,汇编器在翻译汇编程序时会计算每个数据对象和每条指令的地址,当看到这样一个符号定义时,就把它后面一条指令的地址作为这个符号所代表的地址。而_start这个符号又比较特殊,它所代表的地址是整个程序的入口地址,所以下一条指令movl $1, %eax就成了程序中第一条被执行的指令。

 movl $1, %eax

这是一条数据传送指令,这条指令要求CPU内部产生一个数字1并保存到eax寄存器中。mov的后缀l表示long,说明是32位的传送指令。这条指令不要求CPU读内存,1这个数是在CPU内部产生的,称为立即数(Immediate)。在汇编程序中,立即数前面要加$,寄存器名前面要加%,以便跟符号名区分开。以后我们会看到mov指令还有另外几种形式,但数据传送方向都是一样的,第一个操作数总是源操作数,第二个操作数总是目标操作数。

 movl $4, %ebx

和上一条指令类似,生成一个立即数4并保存到ebx寄存器中。

 int $0x80

前两条指令都是为这条指令做准备的,执行这条指令时发生以下动作:

  • int指令称为软中断指令,可以用这条指令故意产生一个异常,上一章讲过,异常的处理和中断类似,CPU从用户模式切换到特权模式,然后跳转到内核代码中执行异常处理程序。

  • int指令中的立即数0x80是一个参数,在异常处理程序中要根据这个参数决定如何处理,在Linux内核中int $0x80这种异常称为系统调用(System Call)。内核提供了很多系统服务供用户程序使用,但这些系统服务不能像库函数(比如printf)那样调用,因为在执行用户程序时CPU处于用户模式,不能直接调用内核函数,所以需要通过系统调用切换CPU模式,经由异常处理程序进入内核,用户程序只能通过寄存器传几个参数,之后就要按内核设计好的代码路线走,而不能由用户程序随心所欲,想调哪个内核函数就调哪个内核函数,这样可以保证系统服务被安全地调用。在调用结束之后,CPU再切换回用户模式,继续执行int $0x80的下一条指令,在用户程序看来就像函数调用和返回一样。

  • eax和ebx的值是传递给系统调用的两个参数。eax的值是系统调用号,Linux的各种系统调用都是由int $0x80指令引发的,内核需要通过eax判断用户要调哪个系统调用,_exit的系统调用号是1。ebx的值是传给_exit的参数,表示退出状态。大多数系统调用完成之后会返回用户空间继续执行后面的指令,而_exit系统调用比较特殊,它会终止掉当前进程,而不是返回用户空间继续执行。

  • x86汇编的两种语法:intel语法和AT&T语法

    x86汇编一直存在两种不同的语法,在intel的官方文档中使用intel语法,Windows也使用intel语法,而UNIX平台的汇编器一直使用AT&T语法,所以本书使用AT&T语法。movl %edx,%eax这条指令如果用intel语法来写,就是mov eax,edx,寄存器名不加%号,源操作数和目标操作数的位置互换,字长也不是用指令的后缀l表示而是用另外的方式表示。本书不详细讨论这两种语法之间的区别,读者可以参考[AssemblyHOWTO]。

    介绍x86汇编的书很多,UNIX平台的书都采用AT&T语法,例如[GroudUp],其它书一般采用intel语法,例如[x86Assembly]。

    习题

    1、把本节例子中的int $0x80指令去掉,汇编、链接也能通过,但是执行的时候出现段错误,你能解释其原因吗?

    2. x86的寄存器

    2. x86的寄存器

    x86的通用寄存器有eax、ebx、ecx、edx、edi、esi。这些寄存器在大多数指令中是可以任意选用的,比如movl指令可以把一个立即数传送到eax中,也可传送到ebx中。但也有一些指令规定只能用其中某个寄存器做某种用途,例如除法指令idivl要求被除数在eax寄存器中,edx寄存器必须是0,而除数可以在任意寄存器中,计算结果的商数保存在eax寄存器中(覆盖原来的被除数),余数保存在edx寄存器中。也就是说,通用寄存器对于某些特殊指令来说也不是通用的。

    x86的特殊寄存器有ebp、esp、eip、eflags。eip是程序计数器,eflags保存着计算过程中产生的标志位,其中包括第 3 节 “整数的加减运算”讲过的进位标志、溢出标志、零标志和负数标志,在intel的手册中这几个标志位分别称为CF、OF、ZF、SF。ebp和esp用于维护函数调用的栈帧,在第 1 节 “函数调用”详细讨论。