用java编写五子棋小游戏,Java五子棋小游戏完整源代码-简书
目录
项目结构核心代码ArrComparator.java类ChessMap.java类ChessPanel.java类效果图展示
项目结构
这个是在网上找的资源,出处记不得了,记录一下。程序的总体结构,很简单的:
核心代码
代码如下:
ArrComparator.java类
导入Java。util。比较器;/** * 排序比较器*/arr类比较器实现比较器对象{ int column=2;int排序顺序=-1;//递减public arr comparator(){ } public int compare(Object a,Object b){ if(int[]){ return sort order *(int[])a)[column]-((int[])b)[column]);}抛出new IllegalArgumentException( param a,b必须int[]. );}}
ChessMap.java类
导入javax。挥棒。*;导入Java。awt。*;导入Java。awt。事件。*;导入Java。网。网址;@SuppressWarnings(serial )公共类象棋地图扩展JFrame { private ImageIcon map//棋盘背景位图私人图标黑棋;//黑子位图私人图像图标白棋;//白子位图私人chespanel CP//棋盘私人面板东;列兵JPanel西私有静态最终int FINAL _ WIDTH=450私有静态final int FINAL _ HEIGHT=500//以下为下拉菜单私有菜单菜单栏;private JMenu[] menu={new JMenu(开始),新的JMenu(设置),新的JMenu(帮助)};private JMenuItem[]menuitem 1={ new JMenuItem(重新开始),新的JMenuItem(悔棋),新的JMenuItem(退出)};private JMenuItem[]menuitem 2={ new JMenuItem(禁手选择),新的JMenuItem(人机博弈),新的JMenuItem(人人对弈)};private JMenuItem[]menuitem 3={ new JMenuItem(规则),新的JMenuItem(关于)};私有布尔haveai=true//人与人下还是人与电脑下,真的与电脑下鼠标点击鼠标点击=新鼠标点击();mouse moved mouse moved=新鼠标moved();menuitem clicked menu clicked=new menuitem clicked();//构造函数public ChessMap(){//改变系统默认字体字体font=新字体(对话框,字体。平原,12);Java。util。枚举键=ui管理器。获取默认值().keys();而(键。hasmoreelements()){ Object key=keys。next element();对象值=ui管理器。get(键);if(javax的值实例。挥棒。普拉夫。font uire source){ ui管理器。put(键、字体);}}setTitle(五子棋);setSize(FINAL_WIDTH,FINAL _ HEIGHT);setResizable(false);init();设置位置(工具包。getdefaulttoolkit()
.getScreenSize().width / 2- FINAL_WIDTH / 2, Toolkit.getDefaultToolkit().getScreenSize().height/ 2 - FINAL_HEIGHT / 2);setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);cp.reset();setVisible(true);}//初始化与默认值 public void init() { map=new ImageIcon(getClass().getResource("bg.jpg"));blackchess=new ImageIcon(getClass().getResource("blackchess.gif"));whitechess=new ImageIcon(getClass().getResource("whitechess.gif"));cp=new ChessPanel(map,blackchess,whitechess);menubar=new JMenuBar();menuitem1[0].setActionCommand("Restart");menuitem1[1].setActionCommand("Rollback");menuitem1[2].setActionCommand("Exit");menuitem2[0].setActionCommand("Forbid");menuitem2[1].setActionCommand("Robot");menuitem2[2].setActionCommand("Human");menuitem3[0].setActionCommand("Rule");menuitem3[1].setActionCommand("About");for(int i=0;i<3;i++)menu[0].add(menuitem1[i]);for(int i=0;i<3;i++)menu[1].add(menuitem2[i]);for(int i=0;i<2;i++)menu[2].add(menuitem3[i]);for(int i=0;i<3;i++)menubar.add(menu[i]);Container p = getContentPane();setJMenuBar(menubar);east = new JPanel();west = new JPanel();p.add(east, "East");p.add(west, "West");p.add(cp, "Center");cp.addMouseListener(mouseclicked);cp.addMouseMotionListener(mousemoved);menuitem1[0].addActionListener(menuclicked);menuitem1[1].addActionListener(menuclicked);menuitem1[2].addActionListener(menuclicked);menuitem2[0].addActionListener(menuclicked);menuitem2[1].addActionListener(menuclicked);menuitem2[2].addActionListener(menuclicked);menuitem3[0].addActionListener(menuclicked);menuitem3[1].addActionListener(menuclicked);}class Mouseclicked extends MouseAdapter//判断鼠标左击并通知棋盘和电脑{public void mouseClicked(MouseEvent e){ if(cp.win==false){ if(haveai){ //和电脑博弈 Point p1=new Point(); p1=cp.getPoint(e.getX(),e.getY()); int x=p1.x; int y=p1.y; // 如果该位置已经放置棋子 System.out.println("x="+x+",y="+y); if (cp.isChessOn[x][y] != 2) return; // 玩家为黑棋,考虑禁手 if( cp.able_flag && cp.bw == 0) { int type = cp.getType(x,y,cp.bw); String str = null; switch(type){ case 20: str = "黑长连禁手!请选择其它位置下棋!"; break; case 21: str = "黑四四禁手!请选择其它位置下棋!"; break; case 22: str = "黑三三禁手!请选择其它位置下棋!"; break; default : break; } if(str != null) { JOptionPane.showMessageDialog(null,str); return; } } boolean flag=cp.haveWin(x, y, cp.bw); cp.update( x, y ); cp.putVoice(); //落子声音 // 第一步棋,需初始化设置边界值 if( cp.chess_num == 1){ if(x-1>=0) cp.x_min = x-1; if(x-1<=15) cp.x_max = x+1; if(y-1>=0) cp.y_min = y-1; if(y-1<=15) cp.y_max = y+1; } else cp.resetMaxMin(x,y); if (flag) { cp.wined(1 - cp.bw); return; } cp.putOne(cp.bw);}else{ //和人博弈Point p1=new Point(); p1=cp.getPoint(e.getX(),e.getY()); int x=p1.x; int y=p1.y; // 如果该位置已经放置棋子 System.out.println("x="+x+",y="+y); if (cp.isChessOn[x][y] != 2) return; // 玩家为黑棋,考虑禁手 if( cp.able_flag && cp.bw == 0) { int type = cp.getType(x,y,cp.bw); String str = null; switch(type){ case 20: str = "黑长连禁手!请选择其它位置下棋!"; break; case 21: str = "黑四四禁手!请选择其它位置下棋!"; break; case 22: str = "黑三三禁手!请选择其它位置下棋!"; break; default : break; } if(str != null) { JOptionPane.showMessageDialog(null,str); return; } } boolean flag=cp.haveWin(x, y, cp.bw); cp.update( x, y ); cp.putVoice(); //落子声音 cp.repaint(); // 第一步棋,需初始化设置边界值 if( cp.chess_num == 1){ if(x-1>=0) cp.x_min = x-1; if(x-1<=15) cp.x_max = x+1; if(y-1>=0) cp.y_min = y-1; if(y-1<=15) cp.y_max = y+1; } else cp.resetMaxMin(x,y); if (flag) { cp.wined(1 - cp.bw); return; }} } }}class MouseMoved implements MouseMotionListener//调试用,获得鼠标位置{public void mouseMoved(MouseEvent e) { cp.showMousePos(e.getPoint()); } public void mouseDragged(MouseEvent e) {}}class Menuitemclicked implements ActionListener//菜单消息处理{public void actionPerformed(ActionEvent e) { JMenuItem target = (JMenuItem)e.getSource(); String actionCommand = target.getActionCommand(); if(actionCommand.equals("Restart")){ //重开一局 cp.reset(); if(cp.sbw==cp.WHITE_ONE) cp.update(7, 7); //player=cp.BLACK_ONE; } if(actionCommand.equals("Rollback")){ //悔棋 if(cp.win) { JOptionPane.showMessageDialog(null,"棋局已经结束,不能悔棋!请重新开始新的棋局!"); return; } // 当前轮到玩家下棋,取消两步 否则,取消一步 if(cp.chess_num >= 2 && cp.bw == cp.sbw){ cp.isChessOn[cp.pre[cp.chess_num-1][0]][cp.pre[cp.chess_num-1][1]] = 2; cp.isChessOn[cp.pre[cp.chess_num-2][0]][cp.pre[cp.chess_num-2][1]] = 2; cp.chess_num -= 2; cp.repaint(); } else if(cp.chess_num >= 1 && cp.bw == 1-cp.sbw){ cp.isChessOn[cp.pre[cp.chess_num-1][0]][cp.pre[cp.chess_num-1][1]] = 2; cp.chess_num --; cp.repaint(); } } else if(actionCommand.equals("Exit")){ //退出 System.exit(1); } else if(actionCommand.equals("Forbid")){ //禁手选择 Object[] options = { "无禁手", "有禁手" }; int sel = JOptionPane.showOptionDialog( null, "你的选择:", "禁手选择", JOptionPane.DEFAULT_OPTION, JOptionPane.QUESTION_MESSAGE, null, options, options[0]); if(sel==1){ cp.able_flag=true; System.out.println("有禁手"); }else{ cp.able_flag=false; System.out.println("无禁手"); } } else if(actionCommand.equals("Robot")){ //人机博弈 haveai=true; Object[] options = { "人类先手", "机器先手" }; int sel = JOptionPane.showOptionDialog( null, "你的选择:", "先手选择", JOptionPane.DEFAULT_OPTION, JOptionPane.QUESTION_MESSAGE, null, options, options[0]); if(sel==1){ //机器先手 cp.sbw=cp.WHITE_ONE; cp.update(7, 7); System.out.println("机器先手"); }else{ //人先手 //player=cp.BLACK_ONE; cp.sbw=cp.BLACK_ONE; System.out.println("人先手"); } } else if(actionCommand.equals("Human")){ //人人博弈 haveai=false; cp.setHumanhuman(true); }else if(actionCommand.equals("Rule")){ //规则 JOptionPane.showConfirmDialog(null, "1、无禁手:" +"n"+" 黑白双方依次落子,任一方先在棋盘上形成连续的五个(含五个以上)棋子的一方为胜。" +"n"+"2、有禁手:(走禁手就输,禁手不能落子)" +"n"+" 鉴于无禁手规则黑棋必胜,人们不断采用一些方法限制黑棋先行的优势,以平衡黑白双方的形式。" +"n"+" 于是针对黑棋的各种禁手逐渐形成。" +"n"+" 禁手主要分为以下几类:" +"n"+" (1)黑长连禁手:连成六个以上连续相同的棋子。" +"n"+" (2)黑三三禁手:两个以上的活三。" + "n"+" (3)黑四四禁手:两个以上的四。" + "n"+" 禁手是针对黑棋而言的,白棋没有任何禁手。" ,"规则",JOptionPane.CLOSED_OPTION,JOptionPane.INFORMATION_MESSAGE); } else if(actionCommand.equals("About")){ //版权与帮助 JOptionPane.showConfirmDialog(null,"团队成员:n" +"自行添加","关于",JOptionPane.CLOSED_OPTION,JOptionPane.INFORMATION_MESSAGE); } }} public static void main(String[] args) { new ChessMap(); }}
ChessPanel.java类
import javax.sound.sampled.AudioInputStream;import javax.swing.*;import java.applet.AudioClip;import java.awt.*;import java.net.URL;import java.util.Arrays;import java.util.LinkedList;import java.util.Random;@SuppressWarnings("serial")public class ChessPanel extends JPanel{private ImageIcon map;//棋盘背景位图 private ImageIcon blackchess;//黑子位图 private ImageIcon whitechess;//白子位图 public int isChessOn [][];//棋局 protected boolean win = false; // 是否已经分出胜负 protected int win_bw; // 胜利棋色 protected int deep = 3, weight = 7; // 搜索的深度以及广度 public int drawn_num = 110; // 和棋步数 int chess_num = 0; // 总落子数目 public int[][] pre = new int[drawn_num + 1][2]; // 记录下棋点的x,y坐标 最多 (drawn_num + 1) 个 public int sbw = 0; //玩家棋色黑色0,白色1 public int bw = 0; // 当前应该下的棋色 0:黑色(默认), 1:白色 // 边界值,用于速度优化 protected int x_max = 15, x_min = 0; protected int y_max = 15, y_min = 0; protected boolean able_flag = true; // 是否选择禁手标志 0:无禁手 1:有禁手(默认 private int h;//棋子长 private int w;//棋子宽 private int insx;//插入棋子的位置 private int insy; private Point mousePoint;//鼠标当前位置 private int winer;//获胜方 private boolean humanhuman=false; //是否是人人对弈 private int plast=0;//走了几步了, public int BLACK_ONE;//0表黑子 public int WHITE_ONE;//1表白子 public int NONE_ONE;//2表无子 public int N;//棋盘边长 //-------声音 String[] choics = { "put.wav", "win.wav","lost.wav" }; //声音文件名数组 URL file1 = getClass().getResource(choics[0]); //落子声音文件 URL file2 = getClass().getResource(choics[1]); //获胜声音文件 URL file3 = getClass().getResource(choics[2]); //失败声音文件 AudioClip soundPut = java.applet.Applet.newAudioClip(file1); //落子声音剪辑对象 AudioClip soundWin = java.applet.Applet.newAudioClip(file2); //获胜声音剪辑对象 AudioClip soundLost = java.applet.Applet.newAudioClip(file3); //失败声音剪辑对象 public ChessPanel(){} public ChessPanel(ImageIcon r_map,ImageIcon r_blackchess,ImageIcon r_whitechess) { N=15; map=new ImageIcon(); blackchess=new ImageIcon(); whitechess=new ImageIcon(); map=r_map; blackchess=r_blackchess; whitechess=r_whitechess; NONE_ONE=2; BLACK_ONE=0; WHITE_ONE=1; winer=NONE_ONE; isChessOn=new int[N][N]; h=blackchess.getIconHeight()*(N-1); w=blackchess.getIconWidth()*(N-1); insx=0; insy=0; mousePoint=new Point(); } public void reset(){//重开一局 winer=NONE_ONE; for(int i=0;i<N;i++) for(int j=0;j<N;j++){ isChessOn[i][j]=NONE_ONE; } chess_num = 0; win = false; win_bw=2; bw = 0; x_max = 15; x_min = 0; y_max = 15;y_min = 0; repaint(); } public void showMousePos(Point p){//调试用,显示鼠标位置 int cw; cw=h/N; mousePoint.x=p.x/cw; mousePoint.y=p.y/cw; repaint(); } public Point getPoint(int x,int y){ int cw; insx=x; insy=y; cw=h/N; Point r=new Point(x/cw,y/cw); return r; } public void gameOver(int r_winer){//游戏胜负已分 winer=r_winer; } public void paint(Graphics g){//整体布局 super.paint(g); paintChessMap(g); paintChess(g); if(winer==BLACK_ONE){ g.drawString(new String("游戏结束!黑棋获胜!"),500,200); } else if(winer==WHITE_ONE){ g.drawString(new String("游戏结束!白棋获胜!"),500,200); } } private void paintChessMap(Graphics g){//画棋盘 map.paintIcon(this,g,10,10); int j; g.setColor(Color.BLACK); for(j=0;j<N;j++){//画线 g.drawLine(h/N/2,h/N*j+h/N/2,w-w/N+(N%2)*(h/N/2),h/N*j+h/N/2); g.drawLine(w/N*j+h/N/2,h/N/2,w/N*j+h/N/2,h-h/N+(N%2)*(h/N/2)); } g.fillRect(w/N*7+h/N/2-3,h/N*7+h/N/2-3,6,6);//画5个黑方块 g.fillRect(w/N*3+h/N/2-3,h/N*3+h/N/2-3,6,6); g.fillRect(w/N*11+h/N/2-3,h/N*3+h/N/2-3,6,6); g.fillRect(w/N*3+h/N/2-3,h/N*11+h/N/2-3,6,6); g.fillRect(w/N*11+h/N/2-3,h/N*11+h/N/2-3,6,6); } private void paintChess(Graphics g){//画棋子 int i,j; for(i=0;i<N;i++) for(j=0;j<N;j++){ if(isChessOn[i][j]==BLACK_ONE){ blackchess.paintIcon(this,g,w/N*i,h/N*j); } else if(isChessOn[i][j]==WHITE_ONE){ whitechess.paintIcon(this,g,w/N*i,h/N*j); } } } //-------------------------------下棋声音设置------------------------------------------------- //落子声音 public void putVoice(){soundPut.play(); } //获胜声音 public void winVoice(){ soundWin.play(); } //失败声音 public void lostVoice(){ soundLost.play(); } //----------------------电脑下棋-------------------------------// public void putOne(int bwf ) { //bwf 棋色 0:黑色 1:白色 int x, y, mx = -100000000; x = y = -1; // 搜索最优下棋点 int[][] bests = getBests( bwf ); for (int k = 0; k < bests.length; k++) { int i = bests[k][0]; int j = bests[k][1]; // 有成5,则直接下子,并退出循环..没有,则思考对方情况 if (getType(i, j, bwf) == 1) { x = i; y = j; break; } if (getType(i, j,1 - bwf) == 1) { x = i; y = j; break; } // 预存当前边界值 int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max; // 预设己方下棋,并更新边界值 isChessOn[i][j] = bwf; resetMaxMin(i,j); // 预测未来 int t = findMin(-100000000, 100000000, deep); // 还原预设下棋位置以及边界值 isChessOn[i][j] = 2; x_min=temp1; x_max=temp2; y_min=temp3; y_max=temp4; // 差距小于1000,50%概率随机选取 //System.out.println("外 :" + i + "," + j + " mx:" + mx + " t:" + t); if (t - mx > 1000 Math.abs(t - mx)<1000 && randomTest(3)) { x = i; y = j; mx = t; //System.out.println(i + "," + j + " mx:" + mx + " t:" + t); } } System.out.println("x="+x+",y="+y); // addChess(x,y,(bwf+1)%2,true); // repaint(); int step=0;step++;System.out.println("step "+step+":-----------------------------------------------");for(int i=0;i<15;i++,System.out.print("n"))for(int j=0;j<15;j++){if(isChessOn[j][i]!=2)System.out.print(isChessOn[j][i]);elseSystem.out.print(isChessOn[j][i]);} // 判断是否已分胜负 boolean flag = haveWin(x, y, bwf); //记录 update( x, y ); repaint(); // 重设边界值 resetMaxMin(x,y); // 胜负已分 if (flag) wined(bwf); if (!flag && chess_num >= drawn_num) { win = true; String str = drawn_num + "步没分胜负,判和棋!"; JOptionPane.showMessageDialog(null,str); return; } } //---------搜索当前搜索状态极大值--------------------------------// //alpha 祖先节点得到的当前最小最大值,用于alpha 剪枝 //beta 祖先节点得到的当前最大最小值,用于beta 剪枝。 //step 还要搜索的步数 //return 当前搜索子树极大值 protected int findMax(int alpha, int beta, int step) { int max = alpha; if (step == 0) { return evaluate(); } int[][] rt = getBests(1 - sbw); for (int i = 0; i < rt.length; i++) { int x = rt[i][0]; int y = rt[i][1]; if (getType(x, y, 1 - sbw) == 1) //电脑可取胜 return 100 * ( getMark(1) + step*1000 ); isChessOn[x][y] = 1 - sbw; // 预存当前边界值 int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max; resetMaxMin(x,y); int t = findMin(max, beta, step - 1); isChessOn[x][y] = 2; // 还原预设边界值 x_min=temp1; x_max=temp2; y_min=temp3; y_max=temp4; if (t > max) max = t; //beta 剪枝 if (max >= beta) return max; } return max; } //-----------------------搜索当前搜索状态极小值---------------------------------// //alpha 祖先节点得到的当前最小最大值,用于alpha 剪枝 //beta 祖先节点得到的当前最大最小值,用于beta 剪枝 //step 还要搜索的步数 //return 当前搜索子树极小值。 protected int findMin(int alpha, int beta, int step) { int min = beta; if (step == 0) { return evaluate(); } int[][] rt = getBests(sbw); for (int i = 0; i < rt.length; i++) { int x = rt[i][0]; int y = rt[i][1]; int type = getType(x, y, sbw); if (type == 1) //玩家成5 return -100 * ( getMark(1) + step*1000 ); // 预存当前边界值 int temp1=x_min,temp2=x_max,temp3=y_min,temp4=y_max; isChessOn[x][y] = sbw; resetMaxMin(x,y); int t = findMax( alpha, min, step - 1 ); isChessOn[x][y] = 2; // 还原预设边界值 x_min=temp1; x_max=temp2; y_min=temp3; y_max=temp4; if (t < min) min = t; //alpha 剪枝 if (min <= alpha) { return min; } } return min; } //-----------------选取局部最优的几个落子点作为下一次扩展的节点---------// //bwf 棋色 0:黑棋 1:白棋 //return 选出来的节点坐标 private int[][] getBests(int bwf) { int i_min=(x_min==0 ? x_min:x_min-1); int j_min=(y_min==0 ? y_min:y_min-1); int i_max=(x_max==15 ? x_max:x_max+1); int j_max=(y_max==15 ? y_max:y_max+1); int n = 0; int type_1,type_2; int[][] rt = new int[(i_max-i_min) * (j_max-j_min)][3]; for ( int i = i_min; i < i_max; i++) for (int j = j_min; j < j_max; j++) if (isChessOn[i][j] == 2) { type_1 = getType(i, j, bwf); type_2 = getType(i, j, 1 - bwf); if(able_flag && bwf==0 && (type_1 == 20 type_1 == 21 type_1 == 22)) // 禁手棋位置,不记录 continue; rt[n][0] = i; rt[n][1] = j; rt[n][2] = getMark(type_1) + getMark(type_2); n++; } // 对二维数组排序 Arrays.sort(rt, new ArrComparator()); int size = weight > n? n:weight; int[][] bests = new int[size][3]; System.arraycopy(rt, 0, bests, 0, size); return bests; } //----------------------------计算指定方位上的棋型-------------------// // x,y 方向线基准一点。 //ex,ey 指定方向步进向量。 // k 棋子颜色,0:黑色,1:白色 // 该方向上的棋子数目 以及 活度 private int[] count(int x, int y, int ex, int ey, int bwf) { // 该方向没意义,返回0 if( !makesense(x, y, ex, ey, bwf)) return new int[] {0, 1}; // 正方向 以及 反方向棋子个数 int rt_1 = 1,rt_2 = 1; // 总棋子个数 int rt = 1; // 正方向 以及 反方向连子的活度 int ok_1 = 0,ok_2 =0; // 总活度 int ok = 0; // 连子中间有无空格 boolean flag_mid1 =false,flag_mid2 = false; // 连子中间空格的位置 int flag_i1 = 1,flag_i2 = 1; if (isChessOn[x][y] != 2) { throw new IllegalArgumentException("position x,y must be empty!.."); } int i; // 往正方向搜索 for (i = 1; x + i * ex < 15 && x + i * ex >= 0 && y + i * ey < 15 && y + i * ey >= 0; i++) { if (isChessOn[x + i * ex][y + i * ey] == bwf) rt_1++; // 位置为空,若中空标志为false,则记为中空并继续搜索 否则,break else if(isChessOn[x + i * ex][y + i * ey] == 2) { if(!flag_mid1) { flag_mid1 = true; flag_i1 = i; } else break; } // 位置为对方棋子 else break; } // 计算正方向活度,, // 最后一个位置不超过边界 if (x + i * ex < 15 && x + i * ex >= 0 && y + i * ey < 15 && y + i * ey >= 0) { // 最后一个位置为空位 +1活 if( isChessOn[x + i * ex][y + i * ey] == 2) { ok_1++; // 若是在尾部检测到连续的空格而退出搜索,则不算有中空 if(rt_1 == flag_i1) flag_mid1 = false; // 若中空的位置在4以下 且 棋子数>=4,则这一边的4非活 if(flag_mid1 && rt_1 > 3 && flag_i1 < 4) { ok_1--; } } // 最后一个位置不是空格,且搜索了2步以上,若前一个是空格, 则不算中空,且为活的边 else if( isChessOn[x + i * ex][y + i * ey] != bwf && i >= 2) if(isChessOn[x + (i-1) * ex][y + (i-1) * ey] == 2) { ok_1++; flag_mid1 = false; } } // 最后一个位置是边界 搜索了2步以上,且前一个是空格, 则不算中空,且为活的边 else if(i >= 2 && isChessOn[x + (i-1) * ex][y + (i-1) * ey] == 2) { ok_1++; flag_mid1 = false; } // 往反方向搜索 for (i = 1; x - i * ex >= 0 && x - i * ex < 15 && y - i * ey >= 0 && y - i * ey < 15; i++) { if (isChessOn[x - i * ex][y - i * ey] == bwf) rt_2++; else if(isChessOn[x - i * ex][y - i * ey] == 2) { if(!flag_mid2) { flag_mid2 = true; flag_i2 = i; } else break; } else break; } // 计算反方向活度 if (x - i * ex < 15 && x - i * ex >= 0 && y - i * ey < 15 &&
郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。