Chúng ta muốn tạo lưới và hiển thị trên chương trình hiển thị, cần lựa chọn tệp
tin tạo lưới trong thưmục Simulation. Tùy thuộc vào nội dung mà chúng ta tạo lưới sẽ
được tạo khác nhau.
93 trang |
Chia sẻ: lylyngoc | Lượt xem: 2380 | Lượt tải: 0
Bạn đang xem trước 20 trang tài liệu Luận văn Xây dựng trình biên dịch cho ngôn ngữ wave, để xem tài liệu hoàn chỉnh bạn click vào nút DOWNLOAD ở trên
ao trong hai từ khóa
"PARSER_BEGIN(parser_name)" và "PARSER_END(parser_name)". Tiếp
đến là danh sách các cú pháp luật sinh. Các tùy chọn và luật sinh sẽ được trình
bày dưới đây.
Tham số name trong cặp từ khóa PARSER_BEGIN và PARSER_END là
tên của bộ phân tích cú pháp được sinh ra. Đối với đề tài name có tên là
WaveParser thì JavaCC tự động chèn các tên này vào sau các file tùy chọn
được sinh ra như WaveParserConstants.java, WaveParserTokenManager.java
…
PARSER_BEGIN(parser_name)
. . .
class parser_name . . . {
. . .
}
. . .
PARSER_END(parser_name)
a. Các từ khóa và tùy chọn của JavaCC
JavaCC cung cấp một tập các từ khóa như bảng sau:
EOF IGNORE_CASE JAVACODE LOOKAHEAD
MORE options PARSER_BEGIN PARSER_END
SKIP SPECIAL_TOKEN TOKEN TOKEN_MGR_DECLS
Ngoài ra JavaCC còn cung cấp một số từ khóa thuộc loại tùy chọn trong
một file đặc tả cú pháp của JavaCC.
LOOKAHEAD: Số token được nhìn trước trước khi quyết định lựa chọn
một điểm trong quá trình phân tích. Mặc định có giá trị 1. Số nhỏ hơn, quá trình
phân tích nhanh hơn. Số này có thể định nghĩa lại cho các luật sinh cụ thể trong
phạm vi ngữ pháp để định rõ tính chất giai đoạn sau.
CHOICE_AMBIGUITY_CHECK: Đây là một tùy chọn số nguyên có giá
trị mặc định là 2. Đây là số token có được trong việc kiểm tra sự lựa chọn của
hình thức "A | B | ..." cho tình trạng có nhiều nghĩa.
- 47 -
OTHER_AMBIGUITY_CHECK: Đây là một tùy chọn số nguyên có giá trị
mặc định là 1. Đây là số token có được trong việc kiểm tra tất cả các hình thức
khác nhau của sự lựa chọn (ví dụ của hình thức "(A)*", "(A)+", and "(A)?") cho
tình trạng có nhiều nghĩa. Việc này mất nhiều thời gian hơn sự kiểm tra lựa
chọn, do đó giá trị mặc định được thiết lập là 1 hơn là 2.
STATIC: Đây là một tùy chọn kiểu boolean có giá trị mặc định là true. Nếu
true tất cả các phương thức và biến của lớp theo lý thuyết đều là static trong sự
phát sinh cú pháp và quản lý token.
DEBUG_PARSER: Đây là một tùy chọn kiểu boolean có giá trị mặc định là
false. Tùy chọn này được sử dụng để gỡ lỗi từ việc phát sinh cú pháp. Thiết lập
tùy chọn này là true là nguyên nhân quá trình phân tích cú pháp phát sinh dấu
hiệu của hành động.
DEBUG_LOOKAHEAD: Đây là tùy chọn kiểu boolean có giá trị mặc định
là false. Thiết lập tùy chọn này là true là nguyên nhân quá trình phân tích cú
pháp phát sinh tất cả dấu hiệu thông tin nó thực hiện khi tùy chọn
DEBUG_PARSER là true, đó cũng là nguyên nhân để nó phát sinh ra dấu hiệu
hành động đã thi hành trong thao tác lookahead.
DEBUG_TOKEN_MANAGER: Tùy chọn boolean có giá trị mặc định là
false. Tùy chọn này được sử dụng để gỡ lỗi thông tin từ các token đã phát sinh.
Thiết đặt tùy chọn này là true là nguyên nhân để quản lý token phát sinh dấu
hiệu của hành động. Dấu hiệu này khá rộng và chỉ nên sử dụng nó khi có một
lỗi từ vựng, những lỗi thông báo đến bạn và bạn không biết tại sao. Điển hình
trong trường hợp này, bạn có thể giải quyết vấn đề bằng cách chú ý đến vài
dòng trước đó của dấu hiệu.
ERROR_REPORTING: Tùy chọn boolean có giá trị mặc định là true. Thiết
lập tùy chọn này là false là nguyên nhân các lỗi mắc phải từ các lỗi phân tích cú
pháp có báo cáo thiếu chi tiết. Lý do để thiết lập tùy chọn này là false là để cải
tiến sự thực thi.
JAVA_UNICODE_ESCAPE: Đây là tùy chọn boolean có giá trị mặc định
là false. Khi thiết lập true, sự phân tích cú pháp đã phát sinh sử dụng một luồng
vào xử lý bởi Java Unicode escapes(\u...) trước khi gửi ký tự tới token manager.
Mặc định Java Unicode escapes không xử lý.
UNICODE_INPUT: Đây là tùy chọn kiểu boolean giá trị mặc định là false.
Khi thiết lập true, sự phân tích cú pháp đã phát sinh sử dụng một luồng nhập để
đọc file Unicode. Mặc định file ASCII được thừa nhận.
Tùy chọn này được bỏ qua nếu một trong hai tùy chọn
USER_TOKEN_MANAGER, USER_CHAR_STREAM được thiết lập là true.
IGNORE_CASE: Tùy chọn kiểu boolean có giá trị mặc định là false. Thiết
lập tùy chọn này là true là nguyên nhân phát sinh ra token manager để bác bỏ
trong trường hợp token được chỉ định rõ và những hồ sơ đưa vào. Tùy chọn này
hữu dụng trong việc viết ngữ pháp cho ngôn ngữ như HTML.
- 48 -
USER_TOKEN_MANAGER: Tùy chọn kiểu boolean, giá trị ngầm định là
false. Hành động mặc định là phát sinh ra một token manager làm việc trên
những token ngữ pháp đã chỉ rõ. Nếu nó được thiết lập true, bộ phân tích cú
pháp được phát sinh để thừa nhận những token từ bất kỳ token quản lý nào
thuộc kiểu "TokenManager"- giao diện được phát sinh vào trong thư mục bộ
phân tích cú pháp.
b. Bộ quản lý Token
Phần đặc tả từ vựng của JavaCC được tổ chức thành tập các trạng thái từ
vựng “lexical states”. Mỗi trạng thái từ vựng có một định danh. Trạng thái từ
vựng chuẩn là DEFAULT. Trình quản lý token được sinh ra tại mỗi thời điểm
sẽ có một trạng thái từ vựng trong tập các trạng thái từ vựng. Khi trình quản lý
token khởi tạo nó sẽ có trạng thái từ vựng mặc định là DEFAULT. Trạng thái
từ vựng bắt đầu có thể được xác định giống như là tham số khi xây dựng một
đối tượng quản lý token.
Có 4 loại biểu thức chính qui: SKIP, MORE, TOKEN, và SPECIAL_TOKEN.
o SKIP: Bỏ qua chuỗi từ vựng khi gặp khai báo SKIP.
o MORE: Tiếp tục cho đến khi gặp trạng thái kế tiếp, đưa chuỗi từ
vựng khi gặp khai báo MORE vào trạng thái chờ. Chuỗi này sẽ là
tiền tố cho chuỗi kế tiếp.
o TOKEN: Tạo ra một token sử dụng chuỗi và gửi nó vào bộ phân tích
cú pháp.
o SPECIAL_TOKEN: Tạo ra một token đặc biệt không tham gia vào
quá trình phân tích.
c. Luật sinh
Có 4 loại luật sinh trong JavaCC gồm javacode_production ,
bnf_production, regular_expr_production, token_manager_decls.
javacode_production và bnf_production được sử dụng để định nghĩa ngữ
pháp trực tiếp cho bộ phân tích cú pháp. regular_expr_production được sử
dụng để định nghĩa ngữ pháp của các Token, trình quản lý Token sử dụng
các ngữ pháp này để tạo ra Token manager, có thể hiểu đây các đặc tả token
trong bộ phân tích cú pháp. token_manager_decls được sử dụng để giới
thiệu các khai báo dùng để chèn vào bên trong trình quản lý token.
• javacode_production
javacode_production là một cách để viết mã Java cho một số các luật
sinh thay vì sử dụng EBNF. Điều này hữu ích khi mà việc định nghĩa
một cú pháp bằng EBNF quá khó khăn. Ví dụ sau minh họa cho việc
định nghĩa xác định luật sinh của một block.
JAVACODE
void skip_to_matching_brace() {
Token tok;
- 49 -
int nesting = 1;
while (true) {
tok = getToken(1);
if (tok.kind == LBRACE) nesting++;
if (tok.kind == RBRACE) {
nesting--;
if (nesting == 0) break;
}
tok = getNextToken();
}
}
• bnf_production
bnf_production ::= java_access_modifier java_return_type
java_identifier "(" java_parameter_list ")" ":"
java_block
"{" expansion_choices "}"
bnf_production là luật sinh chuẩn được sử dụng trong JavaCC.
Mỗi luật sinh BNF có vế trái là một đặc tả không kết thúc. Luật sinh
BNF sẽ định nghĩa đặc tả không kết thúc này theo cách thức của BNF
mở rộng hay EBNF. Điều này được thực hiện giống như các khai báo
của các phương thức trong Java.
Có 2 phần chính ở vế phải của luật sinh BNF :
- Phần thứ nhất là tập các khai báo và mã của Java(Java block). Các mã
java này sẽ được sinh ra ở phần đầu của các phương thức cho luật
sinh BNF. Mỗi khi phương thức luật sinh này được thực thi trong quá
trình phân tích cú pháp thì những khai báo và mã java này sẽ được
thực thi. Khai báo trong phần này sẽ có mặt ở tất cả các hành động
trong EBNF. JavaCC không xử lý bất kỳ một khai báo hoặc đoạn mã
java nào, ngoại trừ việc bỏ qua các dấu ngoặc đơn và thu thập tất cả
mã lệnh để làm cơ sở trong các phương thức được sinh ra sau này.
- Phần thứ hai của vế phải là các lựa chọn mở rộng expansion_choices
có cú pháp như sau:
expansion_choices ::= expansion ( "|" expansion )*
Các expansion_choices được viết giống như một danh sách của một hoặc
nhiều các mở rộng (expansion) được ngăn cách nhau bởi các dấu hoặc | .
expansion ::= ( expansion_unit )*
- 50 -
Một mở rộng(expansion) được viết thành một tập các đơn vị mở
rộng(expansion units). Một expansion đúng khi tất cả các expansion
units đều đúng. Ví dụ, mở rộng (expansion) "{" decls() "}" bao gồm 3
đơn vị mở rộng(expansion units) “{” , decls(), và “}”. Một cú pháp đúng
cho mở rộng này là phải bắt đầu bằng một dấu “{” kết thúc bởi dấu “}”
và thỏa mãn hàm decls() ở phần thân.
Một đơn vị mở rộng(expansion_unit) có thể là một lookahead, một tập
các khai báo và mã java (java_block), Các lựa chọn mở
rộng(expansion_choices), ..
expansion_unit ::= local_lookahead
| java_block
| "(" expansion_choices ")" [ "+" | "*" | "?" ]
| "[" expansion_choices "]"
| [ java_assignment_lhs "=" ] regular_expression
| [ java_assignment_lhs "=" ] java_identifier "("
java_expression_list ")"
Một đơn vị mở rộng có thể gồm các lựa chọn mở rộng có các tùy
chọn “+”, “*”, “?”.
lookahead ::= "LOOKAHEAD" "(" [ java_integer_literal ] [ "," ] [
expansion_choices ] [ "," ] [ "{" java_expression "}" ] ")"
Số token được nhìn trước trước khi quyết định lựa chọn một điểm trong
quá trình phân tích. Mặc định có giá trị 1. Số nhỏ hơn, quá trình phân
tích nhanh hơn. Số này có thể định nghĩa lại cho các luật sinh cụ thể
trong phạm vi ngữ pháp để định rõ tính chất giai đoạn sau.
Ví dụ của luật sinh BNF:
void WriteStatement() :
{ Token t; }
{
"printf" t =
{ jjtThis.name = t.image; }
}
• regular_expr_production
regular_expr_production ::= [ lexical_state_list ]
regexpr_kind [ "[" "IGNORE_CASE" "]" ] ":"
"{" regexpr_spec ( "|" regexpr_spec )* "}"
- 51 -
Một luật sinh biểu thức chính qui (regular_expr_production) được sử
dụng để định nghĩa các thực thể từ vựng sẽ được phân tích bởi trình quản
lý token.
Một luật sinh biểu thức chính qui (regular_expr_production) bắt đầu với
một đặc tả của các tình trạng cho ứng dụng của nó. Tình trạng chuẩn là
DEFAULT, đây là tình trạng mặc định cho các thực thể token.
• token_manager_decls
token_manager_decls ::= "TOKEN_MGR_DECLS" ":" java_block
Phần khai báo của trình quản lý token ( token_manager_decls ) bắt đầu
bằng từ khóa "TOKEN_MGR_DECLS" theo sau là dấu “:” và tập các
khai báo và câu lệnh java(Java Block).
3. 5. 5 Excution Processor
Excution Processor (EP) là thành phần đảm nhiệm chức năng xử lý các phép toán
và các lệnh trong wave. EP sẽ được gọi đến để xử lý sau quá trình phân tích để lấy xác
định phép toán cần được xử lý trong Wave Head. Các phép toán trong Wave được xử
lý ở mức đơn giản là chỉ gồm một toán tử và hai toán hạng. Cấu trúc của EP bao gồm
3 thành phần chính đó là: Execution Control Unit (ECU), Operand Fetch Unit (OFU),
Data Processing Unit (DPU) và Matching Unit (MU).
• Execution Control Unit sẽ lấy ra chỉ lệnh và các tên của các toán hạng bên trái,
bên phải. ECU chịu trách nhiệm điều khiển quá trình xử lý.
• Operand Fetch Unit sẽ lấy ra giá trị của các biến bên trái, bên phải của toán hạng
và điều khiển quá trình lưu trữ dữ liệu.
• Data Processing Unit là thành phần xử lý sau khi đã xác định được toán tử và dữ
liệu sẵn sàng.
• Matching Unit đảm nhận công việc xử lý tìm kiếm trong KN.
- 52 -
Hình 3-11: Excution Processor
3. 5. 6 TrackProcessor
Track Processor (TP) quản lý Track Forest của WI. TP xử lý tất cả các echo và
suspended tail thông qua track. Track Processor nhận thông điệp điều khiển từ các
thành phần khác thông qua Track Queue (TQ).
Các thông điệp đi vào Track Queue gồm có:
• CREATE: Thông điệp yêu cầu tạo track.
• EXPANDH, EXPANDS: Tạo các nhánh con (Track con) của track hiện tại.
EXPANDH dùng trong trường hợp các nhánh được sinh ra bởi lệnh hop, còn
EXPANDS là trường hợp các nhánh sinh ra bởi các sector của wave head.
• ACTIVATE: Thông điệp này sẽ được gửi ngay sau khi gửi hết thông điệp
EXPAND để kích hoạt TN hiện tại.
• ECHO: Thông điệp echo từ các nhánh con được gửi lên TN cha.
• TAIL: Thông điệp gửi chuỗi wave tail đến các nhánh con để tiếp tục phát triển.
Dưới đây là một quá trình nhận và xử lý các thông điệp của Track Processor
- 53 -
PROP
Parent /
Children /
NONE
0
Wave
Env
#.C==b
b c
a
PROP
Parent
Children /
NONE
0
Wave
Env
b c
a
PROP
Parent
Children /
NONE
0
Wave
Env
PROP
Parent
Children
NONE
2
Wave
Env
C==b C==b
TRR2 TRR3
TRR1
Hình 3-12: Sau khi nhận và xử lý CREATE
Hình 3-13: Sau khi nhận và xử lý EXPANDH
- 54 -
PROP
Parent
Children /
NONE
0
Wave
Env
b c
a
PROP
Parent
Children /
NONE
0
Wave
Env
PROP
Parent /
Children
NONE
2
Wave
Env
C==b C==b
TRR2 TRR3
TRR1
Hình 3-14: Sau khi nhận và xử lý ACTIVATE
- 55 -
PROP
Parent
Children /
NONE
0
Wave
Env
b c
a
PROP
Parent
Children /
NONE
0
Wave
Env
PROP
Parent /
Children
NONE
0
Wave
Env
C==b C==b
TRR2 TRR3
TRR1
-1 2x
Hình 3-15: Sau khi nhận ECHO từ các nhánh con
- 56 -
3. 5. 7 Communication Processor
Communication Processor đóng vai trò liên lạc giữa các WI trên các máy tính
trong mạng. Khi chuỗi wave di chuyển trong KN bằng các lệnh hop, chúng có thể lan
truyền từ WI trong máy tính này sang WI trên máy tính khác. Quá trình di chuyển của
các chuỗi wave đó đều phải thông qua CP. Để đảm bảo dữ liệu giữa các máy tính là an
toàn nên CP xử dụng giao thức TCP/IP của tầng giao vận để truyền và nhận các thông
điệp, kết nối giữa các máy tính sẽ được giữ nguyên từ lúc các WI được thiết lập cho
đến khi WI rời khỏi mạng. CP quản lý Communication Queue, đây là một hàng đợi
chứa các thông điệp sẽ được gửi đi.
PROP
Parent
Children /
NONE
0
Wave
Env
b c
a
PROP
Parent /
Children
NONE
0
Wave
Env
C==b
TRR2
TRR1
-1 2x
Hình 3-16: Sau khi xử lý ECHO nhận được
- 57 -
Cấu trúc chung của các thông điệp:
• Type: kiểu của thông điệp bao gồm :
o WAVE: thông điệp gửi wave đi
o CREATE TRACK: thông điệp yêu cầu tạo track trên máy khác,
o CREATE NODE: thông điệp yêu cầu tạo một node trong KN,
o CREATE LINK: thông điệp yêu cầu tạo link giữa 2 node trong KN,
o ECHO: thông điệp gửi echo đi, TAIL: thông điệp gửi wave tail đi)
• Destination Address: địa chỉ của máy nhận thông điệp.
• Source Address: địa chỉ của máy gửi thông điệp.
• Data: Thành phần dữ liệu của thông điệp.
3. 6 Quan hệ giữa các thành phần trong Wave Interpreter
Tổ chức các thành phần trong WI và quan hệ giữa chúng được thể hiện trong
Hình 3-18: Quan hệ giữa các thành phần trong Wave Interpreter. Các thông điệp đến
WI đều thông qua Communication Processor với 3 kiểu thông điệp chính là Wave,
Echo và Tail.
Type
Destination Address
Source Address
Data
Communication Queue
Communication Message
Hình 3-17: Communication Processor
- 58 -
Wave nhận được từ CP sẽ được gửi vào WQ để phân tích và xử lý, ngoài ra wave
có thể được chèn vào WQ một cách trực tiếp từ WI, hoặc từ TP khi kích hoạt các
nhánh để gửi các suspended wave. Thành phần EP cũng có thể gửi các wave vào WQ
khi xử lý các sector, các toán tử code injection của ngôn ngữ Wave. Sự lan truyền của
wave phụ thuộc vào TP và EP. Nếu trạng thái track của wave được kích hoạt, sự lan
truyền của wave phụ thuộc gián tiếp vào EP, đầu tiên EP sẽ gửi thông điệp điều khiển
vào TQ, thông điệp sẽ được TP xử lý và chuỗi wave được lan truyền khi track tương
ứng được kích hoạt.
Thông điệp ECHO và TAIL được CP và TP xử lý. Mỗi khi chuỗi wave kết thúc
trong chế độ track EP sẽ tạo ra thông điệp ECHO và gửi vào TQ. Các suspended tail sẽ
được TP gửi đi sau khi xử lý các ECHO mà điều kiện của rule được thỏa mãn để kích
hoạt track.
Các thành phần xử lý trong WI độc lập với nhau, mỗi thành phần sẽ xử lý dữ liệu
trong các hàng đợi riêng. Tuy nhiên, EP có thể coi là thành phần xử lý chính, khi EP
chịu trách nhiệm điều khiển toàn bộ quá trình xử lý wave và khởi tạo các thông điệp
điều khiển tới các thành phần xử lý khác.
PARSING
UNIT (PU)
TRACK
PROCESSOR
(TP)
EXECUTION
PROCESSOR
(EP)
COMMUNICATION
PROCESSOR (CP)
trackless
sectors,
code
injection,
hop, action
track
operations TQ
resumed
tails,
activated
sectors
propagations,
echoes, tails
CQ
wave,
create KN
echoes,
tails
ƽÄ!
waves
echoes
tails
WQ
KN TN
Hình 3-18: Quan hệ giữa các thành phần trong Wave Interpreter
- 59 -
WI là một hệ thống mở hoàn toàn khi 3 thông điệp WAVE, ECHO và TAIL có
thể gửi trực tiếp từ bên ngoài vào trong hệ thống tới các node và link hoặc giữa các
track. Người dùng có thể tương tác với WI bằng cách gửi các thông điệp tới CP hoặc
trực tiếp gửi vào WQ thông qua các giao diện được cung cấp trên máy có cài đặt WI.
Tương ứng với 3 loại thông điệp, WI có thể được tổ chức thành 3 luồng điều
khiển chính cho từng thông điệp và thực thi một cách hoàn toàn độc lập. Các luồng
điều khiền đó chính là đầu vào WI từ bên ngoài hệ thống, không những thế chúng còn
tương tác qua lại lẫn nhau trong WI tạo thành chu trình xử lý các đoạn mã wave trên
các node trong KN. Các hoạt động cơ bản của WI và tương tác giữa chúng được mô tả
như trong Hình 3-19.Trước tiên, chúng tôi sẽ đưa ra cái nhìn tổng quát về các luồng
chính này, sau đó sẽ mô tả chi tiết chúng trong phần tiếp theo.
• Luồng xử lý Wave: Khi nhận được một wave từ bên ngoài, chuỗi wave thông
thường được xử lý theo các bước: phân tích và xử lý wave head (parse,
process_head), tạo ra các nhánh (spread_branches), sao chép wave tail cho các
nhánh (replicate_tail) và lan tỏa chuỗi wave thu được đến các node đích
(propagate). Quá trình này sẽ được tiếp tục cho đến khi chuỗi wave kết thúc
hoặc được gửi ra ngoài hệ thống.
• Luồng xử lý Echo: mỗi echo nhận được sẽ được gửi cho TP xử lý. Quá trình xử
lý echo trải qua các bước: kích hoạt echo (activate_echo), gửi echo lên track
cha (propagate_echo), và kết hợp echo nhận được với echo đã có ở track cha.
Nếu tất cả các nhánh con của track cha đều đã gửi echo lên, thì quá trình lại
được tiếp tục đối với track cha. Vòng lặp cứ tiếp diễn cho đến khi echo được
gửi ra bên ngoài hệ thống.
• Luồng xử lý Tail: tail từ các rule sẽ được gửi ngược lại thông qua track xuống
các track con. Một wave tail từ bên ngoài đến sẽ được gửi tới track tương ứng
và gửi xuống các track con thông qua hệ thống track hiện tại. Quá trình gửi
wave tail sẽ được lặp lại cho đến khi tới các node đích hoặc wave tail được gửi
đến các track con ở bên ngoài hệ thống.
- 60 -
Hình 3-19: Luồng xử lý giữa các thành phần trong Wave Interpreter
3. 6. 1 Luồng xử lý Wave
Các wave nằm trong WQ độc lập với nhau, do đó sẽ được xử lý đồng thời. Đầu
tiên, khi có wave nằm trong WQ, WI sẽ lần lượt lấy ra các wave, tách thành phần
wave string và wave envrironment. Chuỗi wave thu được sẽ trải qua 2 giai đoạn: phân
tích bởi PU và xử lý bởi EU. Ở giai đoạn phân tích, chuỗi wave được tách ra thành
wave head và wave tail. Trong giai đoạn xử lý, wave head sẽ được xử lý ở node hiện
tại và phụ thuộc vào kết quả xử lý mà wave tail sẽ được lan tỏa đến các node đích để
tiếp tục quá trình thông dịch.
Quá trình xử lý wave head được chia ra làm 7 trường hợp xử lý bao gồm: hop,
action, sectors, control_rule, echo_dot_halt, code_injection, và external_call. Chi tiết
về các trường hợp này được mô tả dưới đây:
- 61 -
• Hop
Lệnh hop giúp cho wave di chuyển từ node này sang node khác cũng như từ WI
này sang WI khác trong mạng. Quá trình xử lý hop như sau:
9 parser: wave string được phân tích thành wave head và wave tail. Wave
head được phân tích thành 2 toán hạng và toán tử.
9 process_head: Toán tử xác định được từ bước trên dùng để xác định cấu trúc
điều khiển trong 7 trường hợp xử lý. Trong trường hợp này là hop được xác
định bằng toán tử ‘#’
9 hop: xử lý lệnh hop. Hai toán hạng được xác định giá trị ở vị trí hiện tại
trong KN.
9 match_KN: hai toán hạng sẽ được khớp với link và node của KN nối với
node hiện tại. Nếu wave ở trong chế độ CREATE thì các link và node mới
được tạo ra. Kết quả của bước này là danh sách các node thỏa mãn điều
kiện.
9 spread_branch: Với mỗi node đích, một wave mới được tạo ra cùng với các
biến frontal và biến environment ở node hiện tại.
9 replicate_tail: wave tail được gán làm wave string cho mỗi wave mới tạo ra.
9 activate_branch: kích hoạt xử lý các wave.
9 propagate: các wave tạo ra được gửi đến các node đích. Nếu node đích nằm
trên cùng KN thì quá trình lan tỏa kết thúc và wave tail được gửi đến các
node đích để tiếp tục được xử lý. Ngược lại wave sẽ được gửi ra ngoài mạng
để tới node đích trên KN của máy khác.
Độ dài của chuỗi wave giảm dần do wave head bị mất đi sau mỗi chu trình. Chu
trình liên tục được diễn ra cho đến khi chuỗi wave di chuyển sang WI khác hoặc
wave tail rỗng. Chuỗi wave rỗng nghĩa là đã được xử lý xong và có thể echo sẽ
được gửi đi thông báo trạng thái kết thúc của wave
• Action
Các action được thực hiện ngay trên node hiện tại, bao gồm các toán tử gán và
các toán tử so sánh. Các action được thực hiện dựa trên hai toán hạng bên trái và
bên phải. Trong phép gán, kết quả sẽ được lưu trữ vào toán hạng bên trái. Trong
các phép so sánh, trạng thái thực hiện của phép toán sẽ được ước lượng và quá
trình xử lý chuỗi wave sẽ dừng lại nếu phép so sanh trả về kết quả là ‘fail’.
Luồng xử lý các action sẽ bao gồm các bước sau:
9 parse: wave head và wave tail được tách ra từ chuỗi wave. Trong trường hợp
này toán tử được xác định là các phép gán hoặc các phép so sánh.
9 process_head: quá trình xử lý chuyển sang xử lý các action
9 action_sequence: xử lý toán tử thu được. Việc xử lý các phép gán và các
phép so sánh là khác nhau nên được chuyển thành 2 bước xử lý nhỏ hơn:
− process_assign: Đây là bước xử lý cho các phép gán. Kết quả của phép
gán sẽ được lưu lại vào toán hạng bên trái.
- 62 -
− process_filter: Các phép so sánh được xử lý. Nếu điều kiện so sánh được
thỏa mãn thì quá trình xử lý sẽ tiếp tục, ngược lại việc xử lý sẽ ngừng lại
và một echo thể hiện trạng thái thất bại được gửi đi (start_echo)
− resume_tail: Khi action được xử lý xong, quá trình xử lý wave sẽ được
tiếp tục lặp lại ở node hiện tại.
• Sectors
Nếu trong wave head chứa nhiều sector (mỗi sector phân cách nhau bởi dấu
phẩy) thì chuỗi wave sẽ được phân chia ở node hiện tại thành các nhánh khác nhau.
Mỗi nhánh nhận một wave string khác nhau, mà mỗi wave string là một sector
được nối với wave tail. Việc xử lý sector cũng gần giống với hop nhưng khác một
chút đó là các wave do quá trình xử lý sector sinh ra vẫn tiếp tục được xử lý ở node
hiện tại.
Quá trình xử lý các sector sẽ như sau:
9 parse: Wave head và wave tail được chia ra từ chuỗi wave, trong wave head
chứa cú pháp của sector (chứa dấu phẩy phân cách). Trong trường hợp này,
thay vì xác định các toán từ thì chuỗi wave giữa các sector sẽ được tách ra
và trả lại cho bước xử lý tiếp theo.
9 process_head: quá trình xử lý chuyển sang xử lý các sector.
9 sectors: các nhánh ứng với mỗi sector bắt đầu được tạo ra.
9 spread_branches: mỗi nhánh ứng với một sectors được tạo ra trên cùng node
hiện tại. Các biến frontal, environment được sao chép cho mỗi nhánh.
9 replicate_tail: wave tail sẽ được sao chép cho mỗi nhánh. Khác với việc sao
chép wave tail trong hop, ở đây wave tail được nỗi vào chuỗi wave nằm
trong từng nhánh được tách ra từ các sector trước đó. Các dấu ngoặc thừa bị
xóa bỏ.
9 activate_branch: kích hoạt nhánh để xử lý (khác với hop, việc xử lý các
sector không có bước propagation do các nhánh vẫn nằm trên node hiện tại)
9 resume_tail: wave string thu được sẽ tiếp tục được xử lý từ bước đầu tiên
• Code injection
Ngôn ngữ wave cho phép chèn vào các đoạn chương trình ngay trong quá trình
xử lý từ các biến đi cùng wave trong quá trình đi chuyển hay từ các biến nằm trong
node mà chuỗi wave đi qua. Đoạn chương trình được đưa vào sẽ nối vào đầu của
wave tail và tiếp tục được xử lý trên cùng node với chuỗi wave trước đó.
9 parser: wave head và wave tail được tách ra. Wave head là toán tử ‘^’ đứng
trước một biến báo hiệu đoạn mã được chèn thêm vào chính là giá trị của
biến đó.
9 process_head: quá trình xử lý chuyển sang xử lý lệnh code injection
9 code_injection: Giá trị của biến được lấy ra và chuyển sang dạng xâu (nếu
cần thiết). String thu được sẽ nối vào đầu của wave tail, quá trình xử lý với
chuỗi wave tổng hợp lại tiếp tục từ đầu trên cùng node hiện tại.
• External call
- 63 -
Toán tử external call là cách giao tiếp của WI với các hệ thống khác thông qua hệ
điều hành. Một lời gọi external call sẽ tạm dừng nhánh hiện tại và tiếp tục quay lại
xử lý sau khi lời gọi kết thúc.
9 parser: Quá trình phân tích gặp toán tử external call (cú pháp của ‘?’)
9 process_head: quá trình xử lý chuyển sang xử lý external call
9 external_call: hai toán tử được lấy giá trị và kết hợp thành lệnh để gọi
chương trình. Lệnh này sẽ được gửi đi, nhánh hiện tại sẽ tạm treo cho đến
khi chương trình được gọi thực hiện xong và trả về kết quả.
9 resume_tail: chuỗi wave sẽ tiếp tục quay lại quá trình xử lý
• Control rule
Chuỗi wave có 2 trạng thái lan tỏa, một là trong trạng thái không có track và một
là trong trạng thái track. Khi di chuyển trong trạng thái track, sau mỗi lệnh hop,
hoặc xử lý các sector thì sẽ tạo ra track mới nối với track hiện tại.
Một control rule sẽ thiết lập một điểm điều khiển vào node hiện tại, việc điều
khiển này được gắn với chuỗi wave nằm trong cặp mở đóng ngoặc của cú pháp
rule. Chuỗi wave này gọi là enclosed wave. Quá trình xử lý được chia làm 2 phần:
9 Các biến frontal và environment của nhánh hiện tại sẽ được nhân bản cho
enclosed wave, và một track mới được tạo ra.
9 Rule được gán cho track vừa tạo ra để xử lý các echo và wave tail sẽ được
treo tại track mới này.
Các luật tuần tự (SQ, OQ, AQ) đảm bảo các nhánh được thực hiện một cách lần
lượt. Sau khi kích hoạt các nhánh thì chỉ duy nhất một nhánh được thực hiện và các
nhánh khác sẽ tạm treo lại.
Các bước xử lý như sau:
9 parser: Wave head và wave tail được tách ra. Wave head chứa một trong số
các Rule của ngôn ngữ Wave và chuỗi wave nằm trong cặp mở đóng ngoặc.
9 process_head: Quá trình xử lý chuyển sang xử lý rule
9 control_rule: Xứ lý rule trong wave head
9 extract_head: tách wave string nằm trong cặp mở đóng ngoặc của rule, và
tách các biến frontal, environment khỏi wave.
9 create_track: tạo ra track mới và track hiện tại (nếu có) sẽ là track cha của
track vừa tạo
9 assign_rule: track mới tạo ra sẽ được gán với rule để xử lý các echo về sau.
9 suspend_tail: wave tail sẽ được treo ở track vừa tạo ra.
• Echo dot halt
Echo dot halt là các toán từ đặc biệt được xử dụng để kết thúc chuỗi wave đang
thực thi trên node hiện tại và gửi lại các echo cho track. Sau đó, wave tail được treo
tại track có thể được tiếp tục lan tỏa trong node hiện tại.
Các bước xử lý như sau:
- 64 -
9 parse: wave head và wave tail được tách ra, wave head chứa toán tử echo
dot halt (cú pháp của ‘!’).
9 process_head: chuyển quá trình xủ lý sang echo dot halt
9 echo_dot_halt: giá trị của echo được xác định
9 start_echo: echo được gán cho track hiện tại và tiếp tục được lan tỏa lên trên
theo track.
9 cancel_tail: wave tail sẽ được xóa bỏ sau khi gửi echo.
Không chỉ có lệnh echo dot halt gửi echo sau khi xử lý mà lệnh hop và các lệnh
so sánh cũng gửi echo. Lệnh hop sẽ gửi echo nếu việc khớp các toán hạng với
KN thất bại. Các lệnh so sánh cũng gửi echo nếu điều kiện so sánh không được
thỏa mãn.
3. 6. 2 Luồng xử lý các echo và điều khiển các rule
Các echo sẽ lan truyền ngược lên gốc của hệ thống track và kết hợp với nhau tại
các TN. Các echo sẽ được đồng bộ với nhau bằng cách track cha sẽ chờ các track con
sau khi kết thúc gửi echo lên. Sau khi tất cả các echo đã được nhận đủ, track cha lại
tiếp tục kích hoạt việc gửi echo tổng hợp lên trên. Quá trình trên sẽ tiếp tục cho đến
khi gặp track có kèm theo rule hoặc echo được gửi tới gốc của hệ thống track. Sau khi
xử lý xong chuỗi wave lan tỏa trong chế độ track, chương trình sẽ gửi thông điệp xóa
track, để xóa toàn bộ dữ liệu lưu trữ trong track và giải phóng bộ nhớ phục vụ cho
những đoạn mã wave tiếp theo.
Trong track có thể gắn thêm rule hoặc không. Nếu track có gắn thêm rule thì việc
kết hợp các echo sẽ không giống với khác track thông thường mà chịu sự qui định
khác nhau của mỗi rule. Thêm vào đó, đối với các rule tuần tự có thể kích hoạt các
track bị treo sau khi nhận được echo từ track con. Các rule còn thay đổi hành động của
track sau khi nhận được các echo từ track con. Trong trường hợp này, thay vì gửi echo
lên trên, chuỗi wave tail bị treo trong track sẽ được gửi đến các track con nếu điều kiện
của rule được thỏa mãn.
WT
tail 2
1
3
7
4
5
6
Node
đích
Hình 3-20: Lan truyền echo lên trên
- 65 -
Các bước xử lý echo như trong ta thấy, ban đầu khi các track đã phát triển tới
được node đích, quá trình echo sẽ được thực hiện từ các node đích ngược trở lên trên
về track cha. Khi echo lên tới track có gán rule thì dựa vào echo tổng hợp được và rule
thì wave tail sẽ được gửi trở lại các track lá.
Các bước xử lý echo được thể hiện trong luồng thứ 2 của Hình 3-19: Luồng xử lý
giữa các thành phần trong Wave Interpreter. Các echo sẽ bắt đầu lan tỏa từ các node
đích khi chuỗi wave kết thúc hoặc bị hủy bỏ bởi các điều kiện không thỏa mãn trong
phép so sánh hay khi xử lý lệnh echo dot halt. Echo được kích hoạt tại các track lá, sau
đó lan truyền và đồng bộ trong hệ thống track. Luồng xử lý echo có thể chia làm 3
phần đó là: gửi các echo đi, xóa track và tổng hợp echo tại track chứa rule.
• Gửi các echo
Đây là phần chính trong luồng xử lý echo, bắt đầu khi chuỗi wave kết thúc xử lý.
Các echo có thể được nhận từ ngoài mạng hoặc từ trong cùng WI, khi nhận được
echo các bước xử lý bắt đầu sau bước start_echo:
9 activate_echo: echo được kích hoạt trong track và bắt đầu lan tỏa.
9 propagate_echo: echo được gửi lên track cha. Nếu track cha nằm trên máy
khác thì echo sẽ được gửi ra bên ngoài mạng.
9 merge_echo: bước này do track cha xử lý khi nhận được echo, echo sẽ được
tổng hợp lại với echo đang lưu trữ trong track. Nếu track cha nhận đủ echo
từ các track con, quá trình này lại được tiếp tục đối với track cha.
• Xóa track
Cũng gần giống với việc lan tỏa các echo trong quá trình xóa track cũng có một
thông báo được gửi đi, nhưng thay vì chỉ gửi echo lên trên, công việc xóa track lại
bắt đầu từ track cha.Thông điệp xóa track sẽ được gửi đi sau khi chuỗi wave thực
hiện xong, hoặc chuỗi wave chuyển từ chế độ track sang chế độ thông thường. Các
bước xóa track diễn ra như sau:
9 activate_echo: quá trình xóa track được kích hoạt.
tail
2
1
3
7
4
5
6
Node
đích
Hình 3-21: Gửi tail cho các track con
- 66 -
9 delete_track: track được xóa và tất cả các dữ liệu lưu trữ trong track được
giải phóng.
9 propagate_echo: thông điệp xóa track được gửi xuống các track con
9 merge_echo: nếu track con nằm trên cùng WI với track cha, quá trình xóa
track được lặp lại. Nếu không cùng WI thì thông điệp xóa track sẽ được gửi
ra ngoài mạng.
• Tổng hợp echo tại Track có rule
Nếu rule được gán vào track thì quá trình xử lý echo sẽ thay đổi tùy thuộc vào ý
nghĩa của từng rule khác nhau. Mỗi rule định nghĩa những điều kiện logic làm thay
đổi việc tổng hợp các echo. Ví dụ như đối với luật AS nếu gặp ít nhất một echo
mang giá trị FALSE thì giá trị của echo tại track là FALSE. Các rule tuần tự còn
bao gồm các nhánh bị treo lại, và các nhánh sẽ lần lượt được kích hoạt sau khi nhận
được echo và điều kiện của rule được thỏa mãn.
Thông thường đối với các rule, các wave tail bị treo lại sẽ được gửi đến tất cả các
track con sau khi track nhận đủ echo và điều kiện của rule được thỏa mãn. Sau khi
gửi tail đi thì rule cũng sẽ bị xóa khỏi track và tính chất của rule cũng sẽ mất đi.
Các rule trong track sẽ được kích hoạt mỗi khi nhận được echo từ một trong số
các track con.
9 activate_echo: echo được kích hoạt trong track. Trong trường hợp này track
được kích hoạt là một trong số các track con của track có gắn rule.
9 propagate_echo: echo được gửi lên track cha có gắn rule. Tại đây có 3
trường hợp có thể xảy ra.
− activate_branch: nếu track hiện tại được gán rule tuần tự (SQ, OS, AS)
thì nhánh tiếp theo sẽ được kích hoạt và xử lý.
− delete_rule: nếu điều kiện của rule được thỏa mãn, rule được xóa khỏi
track và wave tail được gửi đến các track con.
− forward_tail: wave tail bắt đầu được truyền đi từ track hiện tại
• Luồng xử lý Tail
Quá trình gửi tail thông qua hệ thống track là luồng xử lý cuối cùng của WI. Quá
trình này được kích hoạt bởi các rule hoặc có thông điệp tail đến từ mạng. Trên
mỗi track nhận được tail, tail sẽ tiếp tục được được gửi đến các track con của nó
cho đến khi track hiện tại là các track lá hoặc track được gửi nằm bên ngoài hệ
thống.
Các bước xử lý tail diễn ra như sau:
9 forward_tail: tail được nhân ra và gửi đến tất cả các track con, lặp lại cho
đến khi tới được các track cuối cùng.
9 delete_track: bước này chỉ sử dụng khi track gửi tail đi không có track cha
và chuỗi wave sau quá trình này sẽ chuyển sang trạng thái lan tỏa không có
track. Tại bước này track sẽ bị xóa đi sau khi gửi tail đến các track con và
giải phóng toàn bộ vùng nhớ mà track chiếm dụng.
- 67 -
9 resume_tail: Khi các track cuối cùng nhận được tail. Tại các track này wave
tail cùng với các biến frontal, environment được lưu trữ kết hợp tạo thành
wave và tiếp tục quá trình xử lý.
3. 6. 3 Xây dựng trình biên dịch Wave trên ngôn ngữ Java
Phần dưới đây tôi xin giới thiệu sơ qua về các lớp chính và quan hệ giữa chúng
khi được viết trên ngôn ngữ Java.
• Hệ thống Knowledge Network
• Hệ thống Track Forest
• Wave và waveEnvironment
- 68 -
• Các thành phần xử lý
- 69 -
CHƯƠNG 4. THỰC HIỆN VÀ KẾT QUẢ
Ở các chương trước chúng ta đã tìm hiểu về công nghệ Wave và từng bước xây
dựng trình biên dịch cho ngôn ngữ này. Trong chương này chúng ta sẽ sử dụng chính
công nghệ Wave để mô phỏng một số vấn đề, bài toán như: tạo lưới đơn hướng (có thể
tạo ở trên máy khác), đa hướng, di chuyển để tránh hoặc vòng quanh chứng ngại vật,
3D và thực tại ảo…
4. 1 Cài đặt
4. 1. 1 Các yêu cầu về phần cứng
Muốn cài đặt và chạy các chương trình mô phỏng và hiển thị tệp tin VRML,
chúng ta cần phải có những máy tính với phần cứng tối thiểu như sau:
- CPU Pentium 4
- Bộ nhớ màn hình 64MB
- RAM 128MB ( phụ thuộc vào yêu cầu tối thiểu của hệ điều hành được cài
đặt)
- Màn hình, chuột, bàn phím
- Ổ đĩa cứng trống tối thiểu 30MB
Để chạy được các chương trình trên nhiều máy tính, chúng ta cần phải có từ hai
máy tính trở lên với cấu hình mỗi máy tối thiểu theo yêu cầu trên và mỗi máy phải có
một card mạng và kết nối mạng với nhau.
4. 1. 2 Các yêu cầu về phần mềm
Chương trình mô phỏng và hiển thị tệp tin VRML đòi hỏi từng phần mềm khác
nhau được cài đặt sẵn, để chạy được tất cả các chương trình, cần phải có tối thiểu các
phần mềm sau:
- Hệ điều hành Windows, Linux, Unix, … (Bắt buộc phải sử dụng Windows để
chạy chương trình GnuPlot và hiển thị tệp tin VRML tại các phiên bản hiện
tại)
- JRE phiên bản 6
- Java 3D phiên bản 1.5.2
- Netbeans 6.1 (Dùng để chạy chương trình dễ dàng hơn)
- Bộ thư viện X3D
- GnuPlot phiên bản 4.2.5
- 70 -
4. 2 Thử nghiệm
4. 2. 1 Sử dụng chương trình
Chương trình WAVE có thể được đóng gói để chạy như một ứng dụng riêng biệt,
tuy nhiên, vì trong chương trình chúng ta sử dụng các thư viện của WAVE để chạy
nên chúng ta sẽ gộp chung chương trình hiển thị và WAVE vào chung một project của
Netbeans cho thuận tiện hơn khi chạy.
Để chạy chương trình mô phỏng các bài toán ở dạng 2D, chúng ta cần phải chạy
chương trình là lớp Sim trong gói (package) sim của project WAVE. Khi mới được
chạy, chương trình sẽ xuất hiện cửa sổ chưa có nội dung.
Hình 4-1. Chương trình hiển thị khi mới được chạy
Sau khi chương trình hiển thị đã chạy, chúng ta chạy cả project WAVE (RUN
project) để chạy chương trình WAVE. Lúc này, chương trình WAVE cũng xuất hiện
một cửa sổ.
- 71 -
Hình 4-2. Chương trình WAVE khi bắt đầu chạy
Sau đó, để thực hiện mô phỏng cho một bài toán cụ thể, chúng ta chọn nút
Browse… của cửa sổ chương trình WAVE sau đó chọn các tệp tin tương ứng trong thư
mục Simulation để chạy tiếp chương trình.
4. 2. 2 Tạo lưới thực địa
Chúng ta muốn tạo lưới và hiển thị trên chương trình hiển thị, cần lựa chọn tệp
tin tạo lưới trong thư mục Simulation. Tùy thuộc vào nội dung mà chúng ta tạo lưới sẽ
được tạo khác nhau. Giả sử, ở đây, chúng ta tạo ra một lưới 5x5, chương trình hiển thị
sẽ hiển thị lưới ra màn hình:
Hình 4-3. Lưới 5x5
- 72 -
Chương trình WAVE cũng đưa ra thông báo, chúng ta có thể xem được tại cửa sổ
Output của Netbeans:
Hình 4-4. Cửa sổ output của Netbeans
4. 3 Di chuyển tự do
Bài toán di chuyển tự do của một vật từ tọa độ (1-1) đến (5-5) được mô phỏng
trong chương trình có kết quả chạy như sau:
Hình 4-5. Vị trí đầu tiên 1-1
- 73 -
Hình 4-6. Chạy ngẫu nhiên tới vị trí tiếp theo
- 74 -
Hình 4-7. Các bước chạy ngẫu nhiên tiếp theo
- 75 -
Hình 4-9. Dừng khi chạy tới đích
4. 3. 1 Di chuyển tránh chướng ngại vật
Đối tượng di chuyển từ nút (1-1) đến (5-5) mà không đi qua chướng ngại vật
Hình 4-8. Tiếp tục chạy ngẫu nhiên
- 76 -
Hình 4-10. Di chuyển qua chướng ngại vật
- 77 -
4. 3. 2 Di chuyển vòng quanh chướng ngại vật
Đối tượng đi vòng quanh chướng ngại vật mà không đi đến các điểm nằm cách
xa chướng ngại vật hoặc đi xuyên qua chướng ngại vật:
Hình 4-11. Vượt qua chướng ngại vật và về đến đích
- 78 -
Hình 4-12. Di chuyển vòng quanh chướng ngại vật
- 79 -
Hình 4-13. Vòng quanh chướng ngại vật 1 vòng thì dừng
- 80 -
4. 4 Di chuyển cùng nhau kiểu tịnh tiến
Hai đối tượng sẽ cùng nhau di chuyển đến đích:
4. 4. 1 Hiển thị hình ảnh 3D động bằng GnuPlot
Bằng cách sử dụng GnuPlot, chúng ta có thể hiển thị các hình ảnh 3D động trên
nhiều máy, mỗi máy hiển thị một phần hình ảnh động.
Hình ảnh hiển thị 3D trên máy thứ nhất với tọa độ x trong khoảng từ 1 đến 5:
Hình 4-14. Di chuyển tịnh tiến cùng nhau
- 81 -
Hình 4-15. Hình ảnh 3D trên máy thứ nhất sử dụng GnuPlot
Máy thứ hai hiển thị phần hình ảnh 3D tiếp theo với x nằm trong khoảng từ 6 đến
10:
Hình 4-16. Hình ảnh 3D trên máy thứ hai sử dụng GnuPlot
4. 4. 2 Hiển thị hình ảnh 3D của tệp tin VRML
Từ KN của WAVE, WAVE sẽ tạo ra một tệp tin VRML và chương trình sẽ hiển
thị tệp tin VRML này dưới dạng 3D:
- 82 -
Hình 4-17. Tệp tin VRML được hiển thị sau khi được tạo bởi KN
4. 4. 3 Hiển thị hình ảnh 3D với các góc nhìn khác nhau
Các vật thể được mô tả trong tệp tin VRML được mô tả giống như các vật thể
trong tệp tin VRML được hiển thị trong Hình 4-17. Tệp tin VRML được hiển thị sau
khi được tạo bởi KN, sau khi thay đổi Transform, nó lại được hiển thị theo một cách
khác:
- 83 -
Hình 4-18. Các đối tượng hiển thị theo cách khác thi thay đổi Transform
Hình 4-19. Một cách nhìn khác thi thay đổi Transform
4. 4. 4 Hiển thị hình ảnh 3D VRML trên nhiều máy
Các đối tượng khác nhau được đặt trên các máy khác nhau, mỗi một đối tượng
được hiển thị trên một máy. Từ một tệp tin VRML được hiển thị như bây giờ Hình
- 84 -
4-20. Hiển thị đối tượng đầu tiên trên máy 1, mỗi máy sẽ nhận nhiệm vụ đưa ra từng
đối tượng:
Trên máy thứ nhất hiển thị đối tượng đầu tiên là hình trụ:
Hình 4-20. Hiển thị đối tượng đầu tiên trên máy 1
Máy 2 hiển thị hình nón:
- 85 -
Hình 4-21. Hiển thị đối tượng thứ hai trên máy 2
- 86 -
CHƯƠNG 5. PHỤ LỤC
5. 1 JJTree
5. 1. 1 Giới thiệu
JJTree là một bộ tiền xử lý của JavaCC, quá trình biên dịch JJTree kết hợp việc
phân tích từ loại xây dựng nên cây hành động tương ứng với những đoạn mã riêng biệt
trong JavaCC. Việc thực thi JJTree thông qua JavaCC sẽ tạo ra các từ loại. JJTree phát
sinh mã để tạo nên cấu trúc cây cú pháp với nhiều nút, mỗi nút có các lá là các ký hiệu
(biến, từ khóa…) trong ngôn ngữ lập trình.
JJTree định nghĩa một nút giao diện Java mà tất cả việc phân tích các nút của cây
phải được tiến hành. Nút giao diện này cung cấp các phương thức thực thi như việc tạo
dựng các nút cha, thêm vào các nút con và việc gọi lại chính nó.
JJTree thao tác một trong hai nút, simple và multi. Nút simple, mỗi bộ phân tích
cú pháp nút cây cụ thể là một loại SimpleNode. Còn nút mutil, kiểu của bộ phân tích
cú pháp nút cây này được lấy từ tên của nút.
JavaCC phân tích cú pháp từ trên xuống còn JJTree lại xây dựng bộ phân tích cú
pháp từ dưới lên. Để làm điều này JJTree sử dụng một ngăn xếp để đưa vào các nút đã
được tạo ra. Khi mà các nút con này tìm thấy các nút cha thì nó lấy các nút con thêm
vào các nút cha. Nó tiếp tục lấy các nút cha vừa mới được tạo ra để đưa vào ngăn xếp,
nút cha này lại được xem như các nút con và quá trình này lại được tiếp diễn cho tới
khi kết thúc.
5. 1. 2 Các kiểu cấu trúc cây
- Khai báo với số lượng nút cụ thể
void Assignment() #Assignment(2) :
{}
{
Id() "=" Expression()
}
- Khai báo với số lượng nút con chưa xác định rõ
o Có giới hạn
void Assignment() #Assignment(>2) :
o Không có giới hạn
void Assignment()#Assignment
- 87 -
5. 2 Thực thi trên ngôn ngữ simpleLang
Trong phần trên, chúng tôi đã giới thiệu qua về JavaCC, file cú pháp của JavaCC,
các tùy chọn khi biên dịch file cú pháp để tạo ra bộ parser thỏa mãn yêu cầu của người
dùng. Trong phần này, dựa vào những khái niệm và tính chất của file cú pháp, ta sẽ
tìm hiểu cách tạo ra bộ parser cho ngôn ngữ simpleLang – một ngôn ngữ rất đơn giản
định nghĩa toán tử cộng.
Thành phần chính dùng để xây dựng lên bộ parser cho ngôn ngữ simpleLang là
file cú pháp jjt (jjtree) bao gồm các các từ tố và đặc tả cú pháp của ngôn ngữ
simpleLang.
Cú pháp định nghĩa ngôn ngữ simpleLang trong JavaCC có dạng như sau :
void simpleLang() : {} { addExpr() }
void addExpr() : {} { integerLiteral() ( "+"
integerLiteral() )? }
void integerLiteral() : {} { }
SKIP : { " " | "\t" | "\n" | "\r" }
TOKEN : { }
Ý nghĩa :
- Định nghĩa ngôn ngữ simpleLang gồm phép toán duy nhất là toán tử cộng
(addExpr())
- Toán tử cộng được định nghĩa bởi 2 hay nhiều integerLiteral cách nhau bới dấu +
- Một integerLiteral được định nghĩa là một từ tố là một số nguyên
Việc xây dựng bộ parser dựa trên file cú pháp .jj sẽ phải tích hợp rất nhiều đoạn
code java vào trong file .jj và không tận dụng được những lợi ích mà Java IDE mang
lại trong quá trình phát triển chương trình. Như đã nói ở trên, file jjtree được tạo ra làm
bộ tiền xử lý cho quá trình tạo ra file .jj, jjtree sẽ định nghĩa được các hành động, thuộc
tính trên từng node của cây cú pháp, do vậy sẽ có khả năng tùy biến cao hơn cho quá
trình lập trình(thiết lập, thay đổi giá trị, sự kiện… tại mỗi node). Cú pháp của file jjtree
tương ứng với cú pháp của file .jj như sau :
SimpleNode simpleLang() : #Root {} { addExpr() {
return jjtThis; }}
void addExpr() : {} { integerLiteral()
( "+"
integerLiteral() #Add(2) )? }
void integerLiteral() : #IntLiteral {} { }
SKIP : { " " | "\t" | "\n" | "\r" }
TOKEN : { }
- 88 -
Ý nghĩa :
Tương tự như file .jj ở phần trên ta đã xem xét, file jjtree định nghĩa cú pháp và
từ tố theo cùng cách thức, điểm khác biệt đó là jjtree tạo ra các class java (các AST) tại
từng node của cây cú pháp. Trong ngôn ngữ simpleLang có 3 AST được tạo ra (sau
dấu #) là : ASTRoot, ASTAdd và ASTIntLiteral thừa kế từ lớp SimpleNode – mang
những thuộc tính chung nhất của một node trong cây cú pháp.
Ví dụ 1: Cây cú pháp cho 1 biểu thức đơn
Ví dụ 2: Cây cú pháp với phép toán cộng
Làm việc với cây cú pháp
Phần trên ta đã tạo được ra cây cú pháp, việc sử dụng cây cú pháp như thế nào
tùy thuộc vào mục đích của người lập trình. Mỗi node trong cây cú pháp được gọi là
một SimpleNode bao gồm một số thuộc tính và phương thức như : node cha, danh
sách các node con, giá trị tại node, duyệt cây từ node hiện thời đổ xuống…
SimpleParser parser = new SimpleParser(new StringReader(
expression ));
SimpleNode rootNode = parser.simpleLang();
rootNode.dump();
sẽ cho kết quả:
Root
Add
IntLiteral
IntLiteral
Một phương thức quan trọng trong SimpleNode đó là lấy ra giá trị của các node
con của một node, câu lệnh như sau:
SimpleNode lhs = addNode.jjtGetChild( 0 );
- 89 -
SimpleNode rhs = addNode.jjtGetChild( 1 );
sẽ lấy ra 2 node con của node addNode.
Thiết lập và lấy giá trị tại node
Quá trình phân tích cú pháp ở trên chưa cho phép lấy ra giá trị IntLiteral tại các
node mang giá trị là số nguyên. Để làm được điều này ta cần thêm các phương thức set
và get để thiết lập và lấy giá trị tại node:
public class SimpleNode extends Node
{
String m_text;
public void setText( String text ) { m_text =
text; }
public String getText() { return
m_text; }
...
}
Và trong file jjt ta cần sửa mã:
void integerLiteral() : #IntLiteral {} }
thành:
void integerLiteral() : #IntLiteral { Token t; }
{ t= { jjtThis.setText( t.image );} }
Khi đó cây cú pháp của phép toán 42+1 sẽ có dạng như sau:
Root
Add
IntLiteral[42]
IntLiteral[1]
5. 3 Xây dựng bộ parser cho ngôn ngữ Wave
Wave cũng giống như các ngôn ngữ khác, cũng bao gồm các đặc tả về Token,
đặc tả cú pháp của ngôn ngữ. Dựa vào những phần trình bày bên trên, công việc chính
cần làm đó là tạo ra file đặc tả cú pháp (jjt) của Wave.
Một phần file jjt của Wave:
options {
STATIC = false;
- 90 -
MULTI = true;
VISITOR = true;
}
PARSER_BEGIN(WAVEParser)
package ast;
public class WAVEParser {
public static void main(String args[]) {
System.err.println("Reading a \"WAVEProgram\" from standard
input...");
WAVEParser p = new WAVEParser(System.in);
try {
SimpleNode wave = p.Wave();
wave.dump(" ");
System.err.println("Thank you.");
} catch (Exception e) {
System.out.println("Oops.");
System.out.println(e.getMessage());
e.printStackTrace();
}
}
final public String WaveTail() {
return m_waveTail;
}
private String m_waveTail;
}
PARSER_END(WAVEParser)
TOKEN: {
)+>
| <STRING: "`" (~["`","\n","\r"])*"'"
| (["a"-"z","_"] ( | )*)>
|
|
|
}
Ví dụ quá trình parse chuỗi Wave “Fa=1.Fa+1.T=Fa “ sẽ cho kết quả như sau:
- 91 -
WAVEProgram : Fa=1.Fa+1.T=Fa
Zone : Fa=1
Sector : Fa=1
Assign : Fa=1
FrontalVar : Fa
IntConst : 1
Zone : Fa+1
Sector : Fa+1
Sum : Fa+1
FrontalVar : Fa
IntConst : 1
Zone : T=Fa
Sector : T=Fa
TerminalOutput : T=Fa
FrontalVar : Fa
- 92 -
CHƯƠNG 6. TÀI LIỆU THAM KHẢO
TÀI LIỆU TIẾNG ANH
[1] Peter Sapaty, Mobile processing in Distributed and Open environments, 1998
[2] Bell, G., A. Parisi, and M. Pesce, The Virtual Reality Modelling Language:
Version 1.0 Specification, November 9, 1995.
[3] Bruno, J., and S. M. Altman, “A Theory of Asychronous Control Networks,”
IEEE Trans. Comput., Vol. C-20, No. 6, June 1971.
[4] Sapaty, P. S,. “Active Information Field as a Model for the Structural Solution of
Task on Graphs and Networds,” Proc. USSR Academy of Sciences: Technical
Cybernetics, No. 5, 1984 (in Russian).
[5] Varbanov, S., and P. S. Sapaty, “An Information System Based on the Wave
Navigation Techniques,” Abstr. International Conference, AIMSA’86, Varna,
Bulgaria, 1986.
[6] Sapaty, P.S., “The Wave-0 Language as a Framework of Navigational Structures
for Knowledge Bases Using Semantic Networks,” Proc. USSR Academy of
Sciences: Technical Cybernetics, No. 5, 1986 (in Russian).
[7] Borst, P., The First Implementation of the WAVE System for UNIX and TCP/IP
Computer Networks, TR 18/92, Faculty of Informatics, University of Karlsrule.
Karlsruhe, Germany, December 1992.
[8] Borst, P.M., H.-T Goetz, P. S. Sapaty, and W. Xorn, “Paralled Knowledge
Processing in Open Networks,” Proc. International Conference and Exhibition
“Hight- Perfor-mance Computing in Networks” (HPCN Europe ‘94), Munich,
Germany, April 1994.
[9] Corbin M. J., and P. S. Sapaty, “Distributed Object-Based Simulation in Wave,”
J. Simul. Pract. Theory, Vol. 3, No.3, pp. 157-181, 1995.
[10] Merchant F., L. F. Bic, P. M. Borst, M. J. Corbin, M. Dillencourt, M. Fukuda,
and P. S. Sapaty, “Simulating Autonomous Objects in a S patial Database Using
WAVE,” Proc. 9th European Simulation Multiconference, Prague,
Czechoslovakia, June 1995.
[11] Livatharas, C., “Integration of Heterogeneous Databases Using WAVE,” M.Sc.
Project Report, Department of Electrical Engineering, University of Surrey.
Surrey, England, August 1995.
[12] Vuong, S., and I. Ivanov, “Mobile Intelligent Agent Systems: WAVE vs.
JAVA,” Proc, etaCOM’96, Portland, Oreg., May 1996.
[13] Vuong, S., and L. Mathy, “Simulating the Mobile-IP Protocol Using Wave,”
Proc, etaCOM’96, Porland, Oreg., May 1996.
[14] Darling, J. C. C., P.S Sapaty, and M. J. Underhill, “Distributed Virtual Reality: A
Fully Dynamic Approach,” Proc. 15th Workshop on Standards for the
- 93 -
Interoperability of Distributed Simulations, Institute for Simulation and
Training, University of Central Florida, Orlando, Fla., September 1995.
[15] Tan, H. K. V,. “Distributed Dynamic 3D Virtual Reality,” M.Sc Telematics
Diploma Project (base on WAVE), Department of Electrical Engineering,
University of Surrey, Surrey, England, 1997.
WEBSITE THAM KHẢO
[16]
[17] https://javacc.dev.java.net/
[18]
[19]
[20]
[21]
[22]
Các file đính kèm theo tài liệu này:
- Luận văn- Xây dựng trình biên dịch cho ngôn ngữ wave.pdf