@for(ii(I):@sum(ii(I):x(I))=1)
应改为:
@sum(ii(I):x(I))=1
就行了。
您正在看的汇编语言是:hello,world!win32汇编小程序。首先我们看一个“复杂”的Win32汇编程序
程序用来显示一个消息框
--------------------------------------------------
文件名:3.asm
.386
.model flat ,stdcall
NULLequ 0
MB_OK equ 0
ExitProcess PROTO :Dword
MessageBoxA PROTO :DWORD,:DWORD,:DWORD,:Dword
includelib kernel32.lib
includelib user32.lib
.data
szTextdb "Hello, world!",0
szCaption db "Win32Asm",0
.code
start:
push MB_OK
lea eax,szCaption
push eax
lea eax,szText
push eax
push NULL
call messageboxa
xor eax,eax
push eax
call exitprocess
end start
--------------------------------------------------
编译链接:
分下面两步进行:
ml /c /coff 3.asm
link /subsystem:Windows /libpath:d:\masm7\lib 3.obj
第一步编译生成3.obj文件
/c 表示只编译,不链接
/coff 表示生成COFF格式的目标文件
第二步链接生成3.exe文件
/subsystem:windows 表示闭姿贺生成Windows文件
/libpath:d:\masm7\lib 表示引入库的路径为:d:\masm7\lib。
在安装Masm32后,引入库位于Masm32\Lib目录下。
也可设置环境变量Lib的值:在dos提示符下键入Set Lib=d:\masm7\lib,这样“链接”就可简单写成:
link /subsystem:Windows 3.obj,试想一下,在程序调试过程中,修改源程序是常用的事啦,每次编译链接都要带/libpath:...那该有多烦人呢。当然,我们也可在源程序中直接给出引入库的位置,这样,链接时就方便啦,如下:
includelib d:\masm7\lib\kernel32.lib
includelib d:\masm7\lib\user32.lib
--------------------------------------------------
执行:在dos提示符下键入3,回车,出现一个消息框,哈哈,真正的Win32程序!
--------------------------------------------------
深入分析:
看一下源程序,轿派有这么两行:call messageboxa\call exitprocess。大家一看都知道,这是子程序调用,但是我们并没写这样的子程序,事实上,这些是API函数。作为函数,我们在调用时可能需要传送给函数一些参数,程册中序怎么知道传送的参数有哪些,类型是什么呢?就是通过函数原型定义,如下所示:
ExitProcess PROTO :Dword
MessageBoxA PROTO :DWORD,:DWORD,:DWORD,:Dword
可以看出,ExitProcess有一个参数,MessageBoxA有四个参数,这些参数都是Dword类型。
在Win32中,参数的传递都是通过堆栈来完成的。象MessageBoxA这个函数有四个参数,究竟是左边的先压入堆栈还是右边的先入栈呢?.model flat,stdcall给出了答案。stdcall 指定参数是从右到左压入堆栈的,且调整堆栈是在子程序返回时完成的。在源程序中不需要用“add sp,值”来保持堆栈平衡。对MessageBox,在API手册中是这样定义的:
int MessageBox(
HWND hWnd, // handle of owner window
LPCTSTR lpText, // address of text in message box
LPCTSTR lpCaption, // address of title of message box
UINT uType&n
您正在看的汇编语言是:hello,world!win32汇编小程序。
bsp// style of message box
)
所以会有我们的程序段:
push MB_OK
lea eax,szCaption
push eax
lea eax,szText
push eax
push NULL
call messageboxa
看看上面的程序,不难想到,假如在写程序时,少往堆栈里压入一个数据,那将是一个致命的错误。能不能将这种检查参数个数是否匹配的工作交给计算机来完成呢?这是可以的,INVOKE指令可以帮助我们完成这样的工作。假如你的参数个数不正确,连接器将给出错误提示。所以,极力建议你使用invoke代替call来调用子程序,当然,这不是绝对的。使用invoke上面的指令就可简写成下面的样子,看起来简炼多啦,查错也方便啦!
invoke messageboxa, NULL,addr szText,addr szCaption,MB_OK
另外,像NULL,MB_OK都是一些常量,这样的常量有很多,还有很多的结构,如果在我们的程序中一开始都写这么多的东西,可能一下子就把你吓怕啦,也容易出错,更不便于看程序的主要部分。hutch整理的Windows.inc包含了WIN32编程所需要的常量和结构体的定义,我们可简单的用一个include指令将这些常量和结构的定义插入到我们的文件中:
include d:\masm32\include\Windows.inc
但是Windows.inc中并不包含函数原型的声明,还要从其他的头文件中得到函数原型的声明,比如:messageboxa的原型声明在user32.inc文件中,exitprocess在kernel32.inc文件中。这些头文件都放在 \masm32\include文件夹下。
还有,要用Windows.inc,必须使用option casemap:none,它的意思是告诉 MASM 要区分符号的大小写,譬如:start和START是不一样的。否则,一个小小的程序,可能会出成百上千的错误呀!
其他的,就不再细说啦,到此,上面的程序可重新修改如下:
-----------------------------------------------------------------
最终的结果
.386表示要用到386指令
.model flat,stdcall 32位程序,要用flat啦!stadcall,标准调用
option casemap:none 区别大小写
includeWindows.inc 包括常量及结构定义
includekernel32.inc 函数原型声明
includeuser32.inc
includelib kernel32.lib 用到的引入库
includelib user32.lib
.data数据区,定义2个字符串
szTextdb "Hello, world!",0
szCaption db "Win32Asm",0
.code 代码开始执行处
start:
invoke MessageBox,NULL,addr szText,addr szCaption,MB_OK
调用MessageBoxAPI函数
invoke ExitProcess,NULL 程序退出
end start结束
------------------------------------
编译链接:
ml /c /coff /I d:\masm7\include 3.asm 注意开关符识别大小写
link /subsystem:Windows /libpath:d:\masm7\lib 3.obj
/I d:\masm7\include 表示*.inc文件的位置,也可设置环境变量Set include=d:\masm7\include来简化 *** 作,也可在程序中明确指出*.inc的位置。
前面讲的都是用两条指令来完成编译链接,实际上用一条指令也可完成,如下:
ml /coff /I d:\masm7\include 3.asm /link /subsystem:Windows /libpath:lib
若*.inc及引入库在源程序中都明确指出其位置,则可简化为:
ml /coff 3.asm /link /subsystem:
用MVC方式实现的贪吃蛇游戏,共有4个类。运行GreedSnake运行即可。主要是观察者模式的使用,我已经添加了很多注释了。1、
/*
* 程序名称:贪食蛇
* 原作者:BigF
* 修改者:algo
* 说明:我以前也用C写过这个程序,现在看到BigF用Java写的这个,发现虽然作者自称是Java的初学者,
* 但是明显编写程序的素养不错,程序结构写得很清晰,有仔衫销些细微得地方也写得很简洁,一时兴起之
* 下,我认真解读了这个程序,发现数据和表现分开得很好,而我近日正在学习MVC设计模式,
* 因此尝试把程序得结构改了一下,用MVC模式来实现,对源程序得改动不多。
* 我同时也为程序增加了一些自己理解得注释,希望对大家阅读有帮助。
*/
package mvcTest
/**
* @author WangYu
* @version 1.0
* Description:
* </pre>
* Create on :Date :2005-6-13 Time:15:57:16
* LastModified:
* History:
*/
public class GreedSnake {
public static void main(String[] args) {
SnakeModel model = new SnakeModel(20,30)
SnakeControl control = new SnakeControl(model)
SnakeView view = new SnakeView(model,control)
//添加一个观察者,让view成为model的观塌枯察者
model.addObserver(view)
(new Thread(model)).start()
}
}
-------------------------------------------------------------
2、
package mvcTest
//SnakeControl.java
import java.awt.event.KeyEvent
import java.awt.event.KeyListener
/**
* MVC中的Controler,负责接收念游用户的 *** 作,并把用户 *** 作通知Model
*/
public class SnakeControl implements KeyListener{
SnakeModel model
public SnakeControl(SnakeModel model){
this.model = model
}
public void keyPressed(KeyEvent e) {
int keyCode = e.getKeyCode()
if (model.running){ // 运行状态下,处理的按键
switch (keyCode) {
case KeyEvent.VK_UP:
model.changeDirection(SnakeModel.UP)
break
case KeyEvent.VK_DOWN:
model.changeDirection(SnakeModel.DOWN)
break
case KeyEvent.VK_LEFT:
model.changeDirection(SnakeModel.LEFT)
break
case KeyEvent.VK_RIGHT:
model.changeDirection(SnakeModel.RIGHT)
break
case KeyEvent.VK_ADD:
case KeyEvent.VK_PAGE_UP:
model.speedUp()
break
case KeyEvent.VK_SUBTRACT:
case KeyEvent.VK_PAGE_DOWN:
model.speedDown()
break
case KeyEvent.VK_SPACE:
case KeyEvent.VK_P:
model.changePauseState()
break
default:
}
}
// 任何情况下处理的按键,按键导致重新启动游戏
if (keyCode == KeyEvent.VK_R ||
keyCode == KeyEvent.VK_S ||
keyCode == KeyEvent.VK_ENTER) {
model.reset()
}
}
public void keyReleased(KeyEvent e) {
}
public void keyTyped(KeyEvent e) {
}
}
-------------------------------------------------------------
3、
/*
*
*/
package mvcTest
/**
* 游戏的Model类,负责所有游戏相关数据及运行
* @author WangYu
* @version 1.0
* Description:
* </pre>
* Create on :Date :2005-6-13 Time:15:58:33
* LastModified:
* History:
*/
//SnakeModel.java
import javax.swing.*
import java.util.Arrays
import java.util.LinkedList
import java.util.Observable
import java.util.Random
/**
* 游戏的Model类,负责所有游戏相关数据及运行
*/
class SnakeModel extends Observable implements Runnable {
boolean[][] matrix// 指示位置上有没蛇体或食物
LinkedList nodeArray = new LinkedList()// 蛇体
Node food
int maxX
int maxY
int direction = 2// 蛇运行的方向
boolean running = false// 运行状态
int timeInterval = 200// 时间间隔,毫秒
double speedChangeRate = 0.75// 每次得速度变化率
boolean paused = false// 暂停标志
int score = 0// 得分
int countMove = 0// 吃到食物前移动的次数
// UP and DOWN should be even
// RIGHT and LEFT should be odd
public static final int UP = 2
public static final int DOWN = 4
public static final int LEFT = 1
public static final int RIGHT = 3
public SnakeModel( int maxX, int maxY) {
this.maxX = maxX
this.maxY = maxY
reset()
}
public void reset(){
direction = SnakeModel.UP// 蛇运行的方向
timeInterval = 200// 时间间隔,毫秒
paused = false// 暂停标志
score = 0// 得分
countMove = 0// 吃到食物前移动的次数
// initial matirx, 全部清0
matrix = new boolean[maxX][]
for (int i = 0i <maxX++i) {
matrix[i] = new boolean[maxY]
Arrays.fill(matrix[i], false)
}
// initial the snake
// 初始化蛇体,如果横向位置超过20个,长度为10,否则为横向位置的一半
int initArrayLength = maxX >20 ? 10 : maxX / 2
nodeArray.clear()
for (int i = 0i <initArrayLength++i) {
int x = maxX / 2 + i//maxX被初始化为20
int y = maxY / 2//maxY被初始化为30
//nodeArray[x,y]: [10,15]-[11,15]-[12,15]~~[20,15]
//默认的运行方向向上,所以游戏一开始nodeArray就变为:
// [10,14]-[10,15]-[11,15]-[12,15]~~[19,15]
nodeArray.addLast(new Node(x, y))
matrix[x][y] = true
}
// 创建食物
food = createFood()
matrix[food.x][food.y] = true
}
public void changeDirection(int newDirection) {
// 改变的方向不能与原来方向同向或反向
if (direction % 2 != newDirection % 2) {
direction = newDirection
}
}
/**
* 运行一次
* @return
*/
public boolean moveOn() {
Node n = (Node) nodeArray.getFirst()
int x = n.x
int y = n.y
// 根据方向增减坐标值
switch (direction) {
case UP:
y--
break
case DOWN:
y++
break
case LEFT:
x--
break
case RIGHT:
x++
break
}
// 如果新坐标落在有效范围内,则进行处理
if ((0 <= x &&x <maxX) &&(0 <= y &&y <maxY)) {
if (matrix[x][y]) { // 如果新坐标的点上有东西(蛇体或者食物)
if (x == food.x &&y == food.y) { // 吃到食物,成功
nodeArray.addFirst(food)// 从蛇头赠长
// 分数规则,与移动改变方向的次数和速度两个元素有关
int scoreGet = (10000 - 200 * countMove) / timeInterval
score += scoreGet >0 ? scoreGet : 10
countMove = 0
food = createFood()// 创建新的食物
matrix[food.x][food.y] = true// 设置食物所在位置
return true
} else // 吃到蛇体自身,失败
return false
} else { // 如果新坐标的点上没有东西(蛇体),移动蛇体
nodeArray.addFirst(new Node(x, y))
matrix[x][y] = true
n = (Node) nodeArray.removeLast()
matrix[n.x][n.y] = false
countMove++
return true
}
}
return false// 触到边线,失败
}
public void run() {
running = true
while (running) {
try {
Thread.sleep(timeInterval)
} catch (Exception e) {
break
}
if (!paused) {
if (moveOn()) {
setChanged()// Model通知View数据已经更新
notifyObservers()
} else {
JOptionPane.showMessageDialog(null,
"you failed",
"Game Over",
JOptionPane.INFORMATION_MESSAGE)
break
}
}
}
running = false
}
private Node createFood() {
int x = 0
int y = 0
// 随机获取一个有效区域内的与蛇体和食物不重叠的位置
do {
Random r = new Random()
x = r.nextInt(maxX)
y = r.nextInt(maxY)
} while (matrix[x][y])
return new Node(x, y)
}
public void speedUp() {
timeInterval *= speedChangeRate
}
public void speedDown() {
timeInterval /= speedChangeRate
}
public void changePauseState() {
paused = !paused
}
public String toString() {
String result = ""
for (int i = 0i <nodeArray.size()++i) {
Node n = (Node) nodeArray.get(i)
result += "[" + n.x + "," + n.y + "]"
}
return result
}
}
class Node {
int x
int y
Node(int x, int y) {
this.x = x
this.y = y
}
}
------------------------------------------------------------
4、
package mvcTest
//SnakeView.java
import javax.swing.*
import java.awt.*
import java.util.Iterator
import java.util.LinkedList
import java.util.Observable
import java.util.Observer
/**
* MVC模式中得Viewer,只负责对数据的显示,而不用理会游戏的控制逻辑
*/
public class SnakeView implements Observer {
SnakeControl control = null
SnakeModel model = null
JFrame mainFrame
Canvas paintCanvas
JLabel labelScore
public static final int canvasWidth = 200
public static final int canvasHeight = 300
public static final int nodeWidth = 10
public static final int nodeHeight = 10
public SnakeView(SnakeModel model, SnakeControl control) {
this.model = model
this.control = control
mainFrame = new JFrame("GreedSnake")
Container cp = mainFrame.getContentPane()
// 创建顶部的分数显示
labelScore = new JLabel("Score:")
cp.add(labelScore, BorderLayout.NORTH)
// 创建中间的游戏显示区域
paintCanvas = new Canvas()
paintCanvas.setSize(canvasWidth + 1, canvasHeight + 1)
paintCanvas.addKeyListener(control)
cp.add(paintCanvas, BorderLayout.CENTER)
// 创建底下的帮助栏
JPanel panelButtom = new JPanel()
panelButtom.setLayout(new BorderLayout())
JLabel labelHelp
labelHelp = new JLabel("PageUp, PageDown for speed", JLabel.CENTER)
panelButtom.add(labelHelp, BorderLayout.NORTH)
labelHelp = new JLabel("ENTER or R or S for start", JLabel.CENTER)
panelButtom.add(labelHelp, BorderLayout.CENTER)
labelHelp = new JLabel("SPACE or P for pause", JLabel.CENTER)
panelButtom.add(labelHelp, BorderLayout.SOUTH)
cp.add(panelButtom, BorderLayout.SOUTH)
mainFrame.addKeyListener(control)
mainFrame.pack()
mainFrame.setResizable(false)
mainFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE)
mainFrame.setVisible(true)
}
void repaint() {
Graphics g = paintCanvas.getGraphics()
//draw background
g.setColor(Color.WHITE)
g.fillRect(0, 0, canvasWidth, canvasHeight)
// draw the snake
g.setColor(Color.BLACK)
LinkedList na = model.nodeArray
Iterator it = na.iterator()
while (it.hasNext()) {
Node n = (Node) it.next()
drawNode(g, n)
}
// draw the food
g.setColor(Color.RED)
Node n = model.food
drawNode(g, n)
updateScore()
}
private void drawNode(Graphics g, Node n) {
g.fillRect(n.x * nodeWidth,
n.y * nodeHeight,
nodeWidth - 1,
nodeHeight - 1)
}
public void updateScore() {
String s = "Score: " + model.score
labelScore.setText(s)
}
public void update(Observable o, Object arg) {
repaint()
}
}
希望采纳
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)