十几行的lingo小程序,根本display不出来model,初学小菜鸟求大神附体

十几行的lingo小程序,根本display不出来model,初学小菜鸟求大神附体,第1张

如下的语句有误(因循环团烂派的集合是不塌贺历唯存在的):

@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()

}

}

希望采纳


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12446054.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-25
下一篇 2023-05-25

发表评论

登录后才能评论

评论列表(0条)

保存