游戏2048源代码 - C语言控制台界面版
〇、前言
本文最初是在2014年发表的,当时只是Windows版本的,前段时间有位读者给我发邮件咨询Linux下版本移植问题,于是便花时间支持了Linux下的版本,并修改完善了代码,加入记录最高分的功能,供读者参考学习。
一、游戏介绍
所谓《2048》是最近比较流行的一款数字游戏。原版2048首先在github上发布,原作者是Gabriele Cirulli。它是基于《1024》和《小3传奇》(Threes!)的玩法开发而成的新型数字游戏。
二、游戏规则
游戏的规则很简单,你需要控制所有方块向同一个方向运动,两个相同数字的方块撞在一起之后合并成为他们的和,每次操作之后会在空白的方格处随机生成一个2或者4(生成2的概率要大一些),最终得到一个“2048”的方块就算胜利了。
三、核心算法
1、方块的移动和合并
主要思想:把游戏数字面板抽象成4行4列的二维数组a[4][4],值为0的位置表示空方块,其他表示对应数字方块。把每一行同等对待,只研究一行的移动和合并算法,然后可以通过遍历行来实现所有行的移动合并算法。在一行中,用b[4]表示一行的一维数组,使用两个下标变量来遍历列项,这里使用j和k,其中j总在k的后面,用来寻找k项后面第一个不为0的数字,而k项用于表示当前待比较的项,总是和j项之间隔着若干个数字0,或者干脆紧挨着。不失一般性,考虑往左滑动时,初始情况下j等于1,而k等于0,接着判断j项数字是否大于0,若是,则判断j项和k项数字的关系,分成3种情况处理,分别是(合并)P1: b[k]==b[j],(移动)P2: b[k]==0和(碰撞)P3: b[k]!=0且b[k]!=b[j];若否,则j自加1,然后继续寻找k项后面第一个不为0的数字。
其中P1,P2和P3分别对应如下:
(合并)P1:b[k]==b[j],则b[k] = 2 * b[k](说明两数合并了),且b[j] = 0(合并之后要将残留的j项值清零),接着k自加1,然后进行下一次循环。
(移动)P2:b[k]==0,则表示b[j]之前全是空格子,此时直接移动b[j]到k的位置,也就是b[k] = b[j],然后b[j] = 0(移动后将残留的j项值清零),接着k值不变,然后进行下一次循环。
(碰撞)P3:b[k]!=0且b[k]!=b[j],则表示两数不相等且都不为0,此时将两数靠在一起,也就是b[k+1] = b[j]。接着分两种小情况,若j!=k+1,则b[j] = 0(移动后将残留的j项值清零);若否,则表示两数原先就靠在一起,则不进行特殊处理(相当于未移动)。接着k自加1,然后进行下一次循环。
举一个P1的例子,流程表示如下:
一行内移动合并算法描述如下(此例为左移情况,其他方向与之类似,区别仅仅是遍历二维数组的行项和列项的方式):
int j, k;
for (j = , k = ; j < ; j++) {
if (b[j] > ) { /* 找出k后面第一个不为空的项,下标为j,之后分三种情况 */
if (b[k] == b[j]) { /* P1情况,合并 */
b[k] = * b[k];
b[j] = ;
k = k + ;
} else if (b[k] == ) { /* P2情况,移动 */
b[k] = b[j];
b[j] = ;
} else { /* P3情况,碰撞 */
b[k + ] = b[j];
if (j != k + ) { /* 原先两数不挨着 */
b[j] = ;
}
k = k + ;
}
}
}
2、判断游戏是否结束
核心思想:遍历二维数组,看是否存在横向和纵向两个相邻的元素相等,若存在,则游戏不结束,若不存在,则游戏结束。
算法代码描述如下(board表示真正的游戏源码中使用的二维数组):
/* 检查游戏是否结束 函数定义 */
void check_game_over() {
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
/* 横向和纵向比较挨着的两个元素是否相等,若有相等则游戏不结束 */
if (board[i][j] == board[i][j + ] || board[j][i] == board[j + ][i]) {
if_game_over = ;
return;
}
}
}
if_game_over = ;
}
3、生成随机数
核心思想:根据生成的随机数,对一定的值进行取模,达到生成一定概率的数。在本游戏中,设定4出现的概率为1/10,于是可以利用系统提供的随机数函数生成一个数,然后对10取余,得到的数若大于0则在游戏面板空格处生成一个2,若余数等于0,则生成4。在选择将在哪一个空格出生成数的时候,也是根据系统提供的随机函数生成一个数,然后对空格数取余,然后在第余数个空格出生成数字。
算法代码描述如下(board表示真正的游戏源码中使用的二维数组):
/* 生成随机数 函数定义 */
void add_rand_num() {
srand((unsigned int) time());
int n = rand() % get_null_count(); /* 确定在何处空位置生成随机数 */
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
/* 定位待生成的位置 */
if (board[i][j] == && n-- == ) {
board[i][j] = (rand() % ? : ); /* 生成数字2或4,生成概率为9:1 */
return;
}
}
}
}
4、绘制界面
核心思想:利用系统提供的控制台界面清屏功能,达到刷新界面的效果,利用控制制表符位置,达到绘制游戏数字面板的效果。
由于绘制界面不算是本游戏的本质,且代码段相对较长,所以算法描述在这里省略,读者可以参考完整源代码。
5、计算得分
核心思想:两块带数字的方格合并后的数字为合并的得分,一次上下左右移动后游戏面板上所有合并的得分总和为一次移动的得分,多次移动的得分进行累加作为当前总得分。
如果当前总得分(SCORE)超过最高分(BEST),则最高分被改写为当前总得分,并存储下来,下次启动游戏时会自动载入本机存储的最高分。
四、完整源代码如下,敬请读者批评指正:
/*
* Copyright (C) 2014-2018 Judge Young
* E-mail: yjjtc@126.com
* Version: 2.0
* DateTime: 2018-08-01 23:18
*/ #include <time.h> /* 包含设定随机数种子所需要的time()函数 */
#include <stdio.h> /* 包含C的IO读写功能 */
#include <stdlib.h> /* 包含C标准库的功能 */ #ifdef _WIN32 /* 包含Windows平台相关函数,包括控制台界面清屏及光标设定等功能 */
#include <conio.h>
#include <windows.h>
#include <io.h>
#include <direct.h>
#include <Shlobj.h> #else /* 包含Linux平台相关函数,包括控制台界面清屏及光标设定等功能 */
#include <termio.h>
#include <unistd.h>
#include <bits/signum.h>
#include <signal.h> #define KEY_CODE_UP 0x41
#define KEY_CODE_DOWN 0x42
#define KEY_CODE_LEFT 0x44
#define KEY_CODE_RIGHT 0x43
#define KEY_CODE_QUIT 0x71 struct termios old_config; /* linux下终端属性配置备份 */ #endif static char config_path[] = {}; /* 配置文件路径 */ static void init_game(); /* 初始化游戏 */
static void loop_game(); /* 游戏循环 */
static void reset_game(); /* 重置游戏 */
static void release_game(int signal); /* 释放游戏 */ static int read_keyboard(); static void move_left(); /* 左移 */
static void move_right(); /* 右移 */
static void move_up(); /* 上移 */
static void move_down(); /* 下移 */ static void add_rand_num(); /* 生成随机数,本程序中仅生成2或4,概率之比设为9:1 */
static void check_game_over(); /* 检测是否输掉游戏,设定游戏结束标志 */
static int get_null_count(); /* 获取游戏面板上空位置数量 */
static void clear_screen(); /* 清屏 */
static void refresh_show(); /* 刷新界面显示 */ static int board[][]; /* 游戏数字面板,抽象为二维数组 */
static int score; /* 游戏得分 */
static int best; /* 游戏最高分 */
static int if_need_add_num; /* 是否需要生成随机数标志,1表示需要,0表示不需要 */
static int if_game_over; /* 是否游戏结束标志,1表示游戏结束,0表示正常 */
static int if_prepare_exit; /* 是否准备退出游戏,1表示是,0表示否 */ /* main函数 函数定义 */
int main(int argc, char *argv[]) {
init_game();
loop_game();
release_game();
return ;
} /* 读取键盘 函数定义 */
int read_keyboard() {
#ifdef _WIN32
return _getch();
#else
int key_code;
if (read(, &key_code, ) < ) {
return -;
}
return key_code;
#endif
} /* 开始游戏 函数定义 */
void loop_game() {
while () {
int cmd = read_keyboard(); /* 接收标准输入流字符命令 */ /* 判断是否准备退出游戏 */
if (if_prepare_exit) {
if (cmd == 'y' || cmd == 'Y') {
/* 退出游戏,清屏后退出 */
clear_screen();
return;
} else if (cmd == 'n' || cmd == 'N') {
/* 取消退出 */
if_prepare_exit = ;
refresh_show();
continue;
} else {
continue;
}
} /* 判断是否已经输掉游戏 */
if (if_game_over) {
if (cmd == 'y' || cmd == 'Y') {
/* 重玩游戏 */
reset_game();
continue;
} else if (cmd == 'n' || cmd == 'N') {
/* 退出游戏,清屏后退出 */
clear_screen();
return;
} else {
continue;
}
} if_need_add_num = ; /* 先设定不默认需要生成随机数,需要时再设定为1 */ #ifdef _WIN32
/* 命令解析,w,s,a,d字符代表上下左右命令,q代表退出 */
switch (cmd) {
case 'a':
case :move_left();
break;
case 's':
case :move_down();
break;
case 'w':
case :move_up();
break;
case 'd':
case :move_right();
break;
case 'q':
case :if_prepare_exit = ;
break;
default:continue;
}
#else
/* 命令解析,上下左右箭头代表上下左右命令,q代表退出 */
switch (cmd) {
case 'a':
case KEY_CODE_LEFT:move_left();
break;
case 's':
case KEY_CODE_DOWN:move_down();
break;
case 'w':
case KEY_CODE_UP:move_up();
break;
case 'd':
case KEY_CODE_RIGHT:move_right();
break;
case KEY_CODE_QUIT:if_prepare_exit = ;
break;
default:continue;
}
#endif
/* 打破得分纪录 */
if (score > best) {
best = score;
FILE *fp = fopen(config_path, "w");
if (fp) {
fwrite(&best, sizeof(best), , fp);
fclose(fp);
}
} /* 默认为需要生成随机数时也同时需要刷新显示,反之亦然 */
if (if_need_add_num) {
add_rand_num();
refresh_show();
} else if (if_prepare_exit) {
refresh_show();
}
}
} /* 重置游戏 函数定义 */
void reset_game() {
score = ;
if_need_add_num = ;
if_game_over = ;
if_prepare_exit = ; /* 了解到游戏初始化时出现的两个数一定会有个2,所以先随机生成一个2,其他均为0 */
int n = rand() % ;
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
board[i][j] = (n-- == ? : );
}
} /* 前面已经生成了一个2,这里再生成一个随机的2或者4,概率之比9:1 */
add_rand_num(); /* 在这里刷新界面并显示的时候,界面上已经默认出现了两个数字,其他的都为空(值为0) */
refresh_show();
} /* 生成随机数 函数定义 */
void add_rand_num() {
srand((unsigned int) time());
int n = rand() % get_null_count(); /* 确定在何处空位置生成随机数 */
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
/* 定位待生成的位置 */
if (board[i][j] == && n-- == ) {
board[i][j] = (rand() % ? : ); /* 生成数字2或4,生成概率为9:1 */
return;
}
}
}
} /* 获取空位置数量 函数定义 */
int get_null_count() {
int n = ;
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
board[i][j] == ? ++n : ;
}
}
return n;
} /* 检查游戏是否结束 函数定义 */
void check_game_over() {
int i;
for (i = ; i < ; ++i) {
int j;
for (j = ; j < ; ++j) {
/* 横向和纵向比较挨着的两个元素是否相等,若有相等则游戏不结束 */
if (board[i][j] == board[i][j + ] || board[j][i] == board[j + ][i]) {
if_game_over = ;
return;
}
}
}
if_game_over = ;
} /*
* 如下四个函数,实现上下左右移动时数字面板的变化算法
* 左和右移动的本质一样,区别仅仅是列项的遍历方向相反
* 上和下移动的本质一样,区别仅仅是行项的遍历方向相反
* 左和上移动的本质也一样,区别仅仅是遍历时行和列互换
*/ /* 左移 函数定义 */
void move_left() {
/* 变量i用来遍历行项的下标,并且在移动时所有行相互独立,互不影响 */
int i;
for (i = ; i < ; ++i) {
/* 变量j为列下标,变量k为待比较(合并)项的下标,循环进入时k<j */
int j, k;
for (j = , k = ; j < ; ++j) {
if (board[i][j] > ) /* 找出k后面第一个不为空的项,下标为j,之后分三种情况 */
{
if (board[i][k] == board[i][j]) {
/* 情况1:k项和j项相等,此时合并方块并计分 */
score += board[i][k++] *= ;
board[i][j] = ;
if_need_add_num = ; /* 需要生成随机数和刷新界面 */
} else if (board[i][k] == ) {
/* 情况2:k项为空,则把j项赋值给k项,相当于j方块移动到k方块 */
board[i][k] = board[i][j];
board[i][j] = ;
if_need_add_num = ;
} else {
/* 情况3:k项不为空,且和j项不相等,此时把j项赋值给k+1项,相当于移动到k+1的位置 */
board[i][++k] = board[i][j];
if (j != k) {
/* 判断j项和k项是否原先就挨在一起,若不是则把j项赋值为空(值为0) */
board[i][j] = ;
if_need_add_num = ;
}
}
}
}
}
} /* 右移 函数定义 */
void move_right() {
/* 仿照左移操作,区别仅仅是j和k都反向遍历 */
int i;
for (i = ; i < ; ++i) {
int j, k;
for (j = , k = ; j >= ; --j) {
if (board[i][j] > ) {
if (board[i][k] == board[i][j]) {
score += board[i][k--] *= 2;
board[i][j] = ;
if_need_add_num = ;
} else if (board[i][k] == ) {
board[i][k] = board[i][j];
board[i][j] = ;
if_need_add_num = ;
} else {
board[i][--k] = board[i][j];
if (j != k) {
board[i][j] = ;
if_need_add_num = ;
}
}
}
}
}
} /* 上移 函数定义 */
void move_up() {
/* 仿照左移操作,区别仅仅是行列互换后遍历 */
int i;
for (i = ; i < ; ++i) {
int j, k;
for (j = , k = ; j < ; ++j) {
if (board[j][i] > ) {
if (board[k][i] == board[j][i]) {
score += board[k++][i] *= 2;
board[j][i] = ;
if_need_add_num = ;
} else if (board[k][i] == ) {
board[k][i] = board[j][i];
board[j][i] = ;
if_need_add_num = ;
} else {
board[++k][i] = board[j][i];
if (j != k) {
board[j][i] = ;
if_need_add_num = ;
}
}
}
}
}
} /* 下移 函数定义 */
void move_down() {
/* 仿照左移操作,区别仅仅是行列互换后遍历,且j和k都反向遍历 */
int i;
for (i = ; i < ; ++i) {
int j, k;
for (j = , k = ; j >= ; --j) {
if (board[j][i] > ) {
if (board[k][i] == board[j][i]) {
score += board[k--][i] *= 2;
board[j][i] = ;
if_need_add_num = ;
} else if (board[k][i] == ) {
board[k][i] = board[j][i];
board[j][i] = ;
if_need_add_num = ;
} else {
board[--k][i] = board[j][i];
if (j != k) {
board[j][i] = ;
if_need_add_num = ;
}
}
}
}
}
} /* 清屏 */
void clear_screen() {
#ifdef _WIN32
/* 重设光标输出位置清屏可以减少闪烁,system("cls")为备用清屏命令,均为Windows平台相关*/
COORD pos = {, };
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), pos);
CONSOLE_CURSOR_INFO info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &info);
#else
printf("\033c"); /* linux下的清屏命令 */
printf("\033[?25l"); /* linux下的隐藏输入光标 */
#endif
} /* 刷新界面 函数定义 */
void refresh_show() {
clear_screen(); printf("\n\n\n\n");
printf(" GAME: 2048 SCORE: %05d BEST: %06d\n", score, best);
printf(" --------------------------------------------------"); /* 绘制方格和数字 */
printf("\n\n ┌────┬────┬────┬────┐\n");
int i;
for (i = ; i < ; ++i) {
printf(" │");
int j;
for (j = ; j < ; ++j) {
if (board[i][j] != ) {
if (board[i][j] < ) {
printf(" %d │", board[i][j]);
} else if (board[i][j] < ) {
printf(" %d │", board[i][j]);
} else if (board[i][j] < ) {
printf(" %d│", board[i][j]);
} else if (board[i][j] < ) {
printf("%4d│", board[i][j]);
} else {
int n = board[i][j];
int k;
for (k = ; k < ; ++k) {
n = n >> ;
if (n == ) {
printf("2^%02d│", k); /* 超过四位的数字用2的幂形式表示,如2^13形式 */
break;
}
}
}
} else printf(" │");
} if (i < ) {
printf("\n ├────┼────┼────┼────┤\n");
} else {
printf("\n └────┴────┴────┴────┘\n");
}
}
printf("\n");
printf(" --------------------------------------------------\n");
printf(" [W]:UP [S]:DOWN [A]:LEFT [D]:RIGHT [Q]:EXIT"); if (get_null_count() == ) {
check_game_over(); /* 判断是否输掉游戏 */
if (if_game_over) {
printf("\r GAME OVER! TRY THE GAME AGAIN? [Y/N]: \b\b\b\b");
#ifdef _WIN32
CONSOLE_CURSOR_INFO info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &info);
#else
printf("\033[?25h"); /* linux下的显示输入光标 */
#endif
}
} /* 判断是否准备退出游戏 */
if (if_prepare_exit) {
printf("\r DO YOU REALLY WANT TO QUIT THE GAME? [Y/N]: \b\b");
#ifdef _WIN32
CONSOLE_CURSOR_INFO info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &info);
#else
printf("\033[?25h"); /* linux下的显示输入光标 */
#endif
} fflush(); /* 刷新输出缓冲区 */
} /* 初始化游戏 */
void init_game() {
#ifdef _WIN32
system("cls"); /* 获取游戏存档路径,Windows下放在C:\Users\UserName\AppData\Local\2048目录下 */
char m_lpszDefaultDir[MAX_PATH];
char szDocument[MAX_PATH] = {};
memset(m_lpszDefaultDir, , _MAX_PATH);
LPITEMIDLIST pidl = NULL;
SHGetSpecialFolderLocation(NULL, CSIDL_LOCAL_APPDATA, &pidl);
if (pidl && SHGetPathFromIDList(pidl, szDocument)) {
GetShortPathName(szDocument, m_lpszDefaultDir, _MAX_PATH);
}
sprintf(config_path, "%s\\2048", m_lpszDefaultDir);
if (_access(config_path, ) == -) {
_mkdir(config_path);
}
sprintf(config_path, "%s\\2048\\2048.dat", m_lpszDefaultDir);
#else
/* 获取游戏存档路径,Linux下放在当前用户主目录下 */
sprintf(config_path, "%s/.2048", getenv("HOME")); tcgetattr(, &old_config); /* 获取终端属性 */
struct termios new_config = old_config; /* 创建新的终端属性 */
new_config.c_lflag &= ~ICANON; /* 设置非正规模式 */
new_config.c_lflag &= ~ECHO; /* 关闭输入回显 */
new_config.c_cc[VMIN] = ; /* 设置非正规模式下的最小字符数 */
new_config.c_cc[VTIME] = ; /* 设置非正规模式下的读延时 */
tcsetattr(, TCSANOW, &new_config); /* 设置新的终端属性 */ printf("\033[?25l"); signal(SIGINT, release_game);
#endif /* 读取游戏最高分数 */
FILE *fp = fopen(config_path, "r");
if (fp) {
fread(&best, sizeof(best), , fp);
fclose(fp);
} else {
best = ;
fp = fopen(config_path, "w");
if (fp) {
fwrite(&best, sizeof(best), , fp);
fclose(fp);
}
} reset_game();
} /* 释放游戏 */
void release_game(int signal) {
#ifdef _WIN32
system("cls");
CONSOLE_CURSOR_INFO info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &info);
#else
if (signal == SIGINT) {
printf("\n");
}
tcsetattr(, TCSANOW, &old_config); /* 还原回旧的终端属性 */
printf("\033[?25h");
#endif
exit();
}
五、运行界面如下,仅供读者参考玩乐:
其中,按方向键,或者w、s、a、d键为上、下、左、右移动,按q键为退出游戏。
六、版本移植问题
在本文中的源代码是Windows系统的版本,但游戏的核心算法无论在那个系统上都是一样的,区别仅仅是界面绘制刷新的实现部分可能存在差异。比如在Linux上的getch()函数有回显,所以可能会需要更好的命令输入逻辑,而且conio.h并不属于C标准库中,所以在Linux下引用不到此头文件,而Linux下getch()函数存在于curses.h头文件中,所以需要更改头文件。还有,在本文源代码中关于清屏的代码在Linux下失效,所以若想移植需要修改清屏逻辑,达到刷新界面的逻辑,比如调用Linux下的清屏命令system("clear"),效果如何,读者可以试试。
七、版本移植
当前最新版本已经支持Windows和Linux双系统下编译运行了,读者可以下载源码参考学习,给出意见建议,然后编译运行,顺便挑战一下最高分~
游戏2048源代码 - C语言控制台界面版的更多相关文章
- C语言控制台窗体图形界面编程(总结)
本系列文章是笔者通过学习<C语言控制台窗体界面编程(修正版)>而写(关于此文档的很多其它信息请看本系列文章第一篇),旨在让大家更加清晰简洁easy地学习C语言控制台窗体界面的编程. 通过本 ...
- 开源游戏“2048”IOS移植版
简介: 这个游戏是我在今年(2014/05)课余时闲着无聊做的一个益智类小游戏,总共花了4个工作日才完成,为了游戏的效率,做了很多优化,目前在IE5以上浏览器能够流畅运行,运行时如果屏幕分辨率不兼容, ...
- 丢沙包游戏(或杀人游戏)的C语言实现
丢沙包游戏(或杀人游戏)用C语言实现: 游戏简述: 杀人游戏(或者丢沙包游戏),设定一些人(人数为:num)一起玩游戏,从某个指定的人(设定为:start)开始轮流扔沙包,扔沙包人的下一个人为1,每隔 ...
- Android 带你玩转实现游戏2048 其实2048只是个普通的控件(转)
1.概述 博主本想踏入游戏开放行业,无奈水太深,不会游泳:于是乎,只能继续开发应用,但是原生Android也能开发游戏么,2048.像素鸟.别踩什么来着:今天给大家带来一篇2048的开发篇,别怕不分上 ...
- Microsoft源代码注释语言(SAL)提供设置批注
Microsoft源代码注释语言(SAL)提供设置批注可以使用描述的功能如何使用其参数,它对其假设并确保它使其在完成. 批注可标头文件 <sal.h>定义. Visual Studio C ...
- Android 带你玩转实现游戏2048 其实2048只是个普通的控件
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40020137,本文出自:[张鸿洋的博客] 1.概述 博主本想踏入游戏开放行业,无 ...
- 双人对战的球类游戏ios源代码项目
双人对战的球类游戏源代码,这个是一款双人对战的ios球类游戏源代码.游戏的源代码也比較具体的,我们在屏幕上下看到各有一个球门.内有一球,两边通过控制轮盘使球进入对方的球门的.事实上玩法也非常easy的 ...
- cocos2d-x《农场模拟经营养成》游戏完整源代码
cocos2d-x农场模拟经营养成游戏完整源代码,cocos2d-x引擎开发,使用JSON交互,支持IOS与 Android,解压后1016MB. 非常强大的游戏源代码 完整游戏源代码 ...
- Go 语言控制台输入&生成随机数
Go 语言控制台输入&生成随机数 1. 不同基础类型之间的转化对于不同的基础类型之间的转化,Go 提供了 strconv包.它实现了字符串与其他基本数据类型之间的转化.其中最常用的数值转化函数 ...
随机推荐
- 番外篇--Moddule Zero介绍
1.1 ABPZero - 概述 介绍 微软ASP.NET身份框架 权限 会话 角色管理 默认角色 用户管理 多租户 设置管理 审计日志 1.1.1 介绍 Modulde Zero实现了ASP.NET ...
- nxlog4go 简介 - 基于log4go的下一代go语言日志系统
nxlog4go的项目网址: https://github.com/ccpaging/nxlog4go 项目历史 ccpaging's log4go forked from https://githu ...
- ASIHTTPRequest
ASIHTTPRequest,是一个直接在CFNetwork上做的开源项目,提供了一个比官方更方便更强大的HTTP网络传输的封装.
- RMQ算法 (ST算法)
概述: RMQ(Range Minimum/Maximum Query),即区间最值查询,是指这样一个问题:对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中 ...
- 【jsp】MyEclipse10.7.1最新版+破解下载
MyEclipse企业级工作平台[1](MyEclipse Enterprise Workbench ,简称MyEclipse)是对EclipseIDE的扩展,利用它我们可以在数据库和JavaEE的开 ...
- linux的nvme驱动参数调优
nvme的设备,可以调优的参数比较少,相关的代码如下: blk_sysfs.c static struct queue_sysfs_entry queue_requests_entry = { .at ...
- Python装饰器小代码
# coding=utf-8import timedef outer(fun): def inner(): start = time.time() fun() runtime = time.time( ...
- SpringMvc4.x---快捷的ViewController
@RequestMapping("/index") public String hello(){ return "index"; } 此处无任何的业务处理,只是 ...
- CSS样式中常用的字体名称
css中引入字体: @font-face { font-family: "AncientWar"; src: url('style/css/fonts/AncientWar.ttf ...
- matlab判断文件或文件夹是否存在
当前目录中包含以下文件及文件夹: startup.m win64/ … 判断当前目录中是否存在startup.m文件 if ~exist('startup.m','file')==0 error ...