因为使用了数据库,要安装数据库。

头文件:

ifndef __DLIST_H

#define __DLIST_H

/* This file is from Linux Kernel (include/linux/list.h)

* and modified by simply removing hardware prefetching of list items.

* Here by copyright, credits attributed to wherever they belong.

* Kulesh Shanmugasundaram (kulesh [squiggly] isis.poly.edu)

*/

/*

* Simple doubly linked list implementation.

*

* Some of the internal functions (“__xxx”) are useful when

* manipulating whole lists rather than single entries, as

* sometimes we already know the next/prev entries and we can

* generate better code by using them directly rather than

* using the generic single-entry routines.

*/

/**

* container_of - cast a member of a structure out to the containing structure

*

* @ptr:the pointer to the member.

* @type:the type of the container struct this is embedded in.

* @member:the name of the member within the struct.

*

*/

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

#define container_of(ptr, type, member) ({\

const typeof( ((type *)0)->member ) *__mptr = (ptr);\

(type *)( (char *)__mptr - offsetof(type,member) );})

/*

* These are non-NULL pointers that will result in page faults

* under normal circumstances, used to verify that nobody uses

* non-initialized list entries.

*/

#define LIST_POISON1  ((void *) 0x00100100)

#define LIST_POISON2  ((void *) 0x00200200)

struct list_head {

struct list_head *next, *prev;

};

#define LIST_HEAD_INIT(name) { &(name), &(name) }

#define LIST_HEAD(name) \

struct list_head name = LIST_HEAD_INIT(name)

#define INIT_LIST_HEAD(ptr) do { \

(ptr)->next = (ptr); (ptr)->prev = (ptr); \

} while (0)

/*

* Insert a new entry between two known consecutive entries.

*

* This is only for internal list manipulation where we know

* the prev/next entries already!

*/

static inline void __list_add(struct list_head *new,

struct list_head *prev,

struct list_head *next)

{

next->prev = new;

new->next = next;

new->prev = prev;

prev->next = new;

}

/**

* list_add – add a new entry

* @new: new entry to be added

* @head: list head to add it after

*

* Insert a new entry after the specified head.

* This is good for implementing stacks.

*/

static inline void list_add(struct list_head *new, struct list_head *head)

{

__list_add(new, head, head->next);

}

/**

* list_add_tail – add a new entry

* @new: new entry to be added

* @head: list head to add it before

*

* Insert a new entry before the specified head.

* This is useful for implementing queues.

*/

static inline void list_add_tail(struct list_head *new, struct list_head *head)

{

__list_add(new, head->prev, head);

}

/*

* Delete a list entry by making the prev/next entries

* point to each other.

*

* This is only for internal list manipulation where we know

* the prev/next entries already!

*/

static inline void __list_del(struct list_head *prev, struct list_head *next)

{

next->prev = prev;

prev->next = next;

}

/**

* list_del – deletes entry from list.

* @entry: the element to delete from the list.

* Note: list_empty on entry does not return true after this, the entry is in an undefined state.

*/

static inline void list_del(struct list_head *entry)

{

__list_del(entry->prev, entry->next);

entry->next = (void *) 0;

entry->prev = (void *) 0;

}

/**

* list_del_init – deletes entry from list and reinitialize it.

* @entry: the element to delete from the list.

*/

static inline void list_del_init(struct list_head *entry)

{

__list_del(entry->prev, entry->next);

INIT_LIST_HEAD(entry);

}

/**

* list_move – delete from one list and add as another’s head

* @list: the entry to move

* @head: the head that will precede our entry

*/

static inline void list_move(struct list_head *list,

struct list_head *head)

{

__list_del(list->prev, list->next);

list_add(list, head);

}

/**

* list_move_tail – delete from one list and add as another’s tail

* @list: the entry to move

* @head: the head that will follow our entry

*/

static inline void list_move_tail(struct list_head *list,

struct list_head *head)

{

__list_del(list->prev, list->next);

list_add_tail(list, head);

}

/**

* list_empty – tests whether a list is empty

* @head: the list to test.

*/

static inline int list_empty(struct list_head *head)

{

return head->next == head;

}

static inline void __list_splice(struct list_head *list,

struct list_head *head)

{

struct list_head *first = list->next;

struct list_head *last = list->prev;

struct list_head *at = head->next;

first->prev = head;

head->next = first;

last->next = at;

at->prev = last;

}

/**

* list_splice – join two lists

* @list: the new list to add.

* @head: the place to add it in the first list.

*/

static inline void list_splice(struct list_head *list, struct list_head *head)

{

if (!list_empty(list))

__list_splice(list, head);

}

/**

* list_splice_init – join two lists and reinitialise the emptied list.

* @list: the new list to add.

* @head: the place to add it in the first list.

*

* The list at @list is reinitialised

*/

static inline void list_splice_init(struct list_head *list,

struct list_head *head)

{

if (!list_empty(list)) {

__list_splice(list, head);

INIT_LIST_HEAD(list);

}

}

/**

* list_entry – get the struct for this entry

* @ptr:    the &struct list_head pointer.

* @type:    the type of the struct this is embedded in.

* @member:    the name of the list_struct within the struct.

*/

#define list_entry(ptr, type, member) \

((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))

/**

* list_for_each    -    iterate over a list

* @pos:    the &struct list_head to use as a loop counter.

* @head:    the head for your list.

*/

#define list_for_each(pos, head) \

for (pos = (head)->next; pos != (head); \

pos = pos->next)

/**

* list_for_each_prev    -    iterate over a list backwards

* @pos:    the &struct list_head to use as a loop counter.

* @head:    the head for your list.

*/

#define list_for_each_prev(pos, head) \

for (pos = (head)->prev; pos != (head); \

pos = pos->prev)

/**

* list_for_each_safe    -    iterate over a list safe against removal of list entry

* @pos:    the &struct list_head to use as a loop counter.

* @n:        another &struct list_head to use as temporary storage

* @head:    the head for your list.

*/

#define list_for_each_safe(pos, n, head) \

for (pos = (head)->next, n = pos->next; pos != (head); \

pos = n, n = pos->next)

/**

* list_for_each_entry    -    iterate over list of given type

* @pos:    the type * to use as a loop counter.

* @head:    the head for your list.

* @member:    the name of the list_struct within the struct.

*/

#define list_for_each_entry(pos, head, member)                \

for (pos = list_entry((head)->next, typeof(*pos), member);    \

&pos->member != (head);                     \

pos = list_entry(pos->member.next, typeof(*pos), member))

/**

* list_for_each_entry_safe – iterate over list of given type safe against removal of list entry

* @pos:    the type * to use as a loop counter.

* @n:        another type * to use as temporary storage

* @head:    the head for your list.

* @member:    the name of the list_struct within the struct.

*/

#define list_for_each_entry_safe(pos, n, head, member)            \

for (pos = list_entry((head)->next, typeof(*pos), member),    \

n = list_entry(pos->member.next, typeof(*pos), member);    \

&pos->member != (head);                     \

pos = n, n = list_entry(n->member.next, typeof(*n), member))

#endif

服务端代码:

#include #include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include "dlist.h"

struct node_data{

int site;

int sign;

int sock_fd;

int id[100];

int total;

};

typedef struct node{

struct node_data data;

struct list_head list;

}server, *pserver;

#define SQL_SIZE 128

char sql[SQL_SIZE] = {'\0'};

sqlite3 *db_menu;//菜单数据库

sqlite3 *db_bd;//运营数据库

char *errmsg;

void menu_add(void);

void store_mod(void);

void menu_query(void);

void menu_del(void);

void quer_bd(void);

void *sock_write(void *arg);

void *sock_write(void *arg)

{

int i;

while(1){

printf("******************************\n");

printf("\t1:—>\t\t添加新菜品\n");

printf("\t2:—>\t\t删除旧菜品\n");

printf("\t3:—>\t\t查询菜品\n");

printf("\t4:—>\t\t修改菜品信息\n");

printf("\t5:—>\t\t统计运营额\n");

printf("******************************\n");

scanf("%d",&i);

switch(i){

case 1:

menu_add();

break;

case 2:

menu_del();

break;

case 3:

menu_query();

break;

case 4:

store_mod();

break;

case 5:

quer_bd();

break;

}

}

return;

}

void quer_bd(void)

{

int ret;

int row;

int i;

int colum;

char **result;

sprintf(sql, "select * from bdo");

ret = sqlite3_get_table(db_menu, sql, &result, &row, &colum, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

for(i=0 ; i

if(0 == i%3){

printf("\n");

}

printf("%s\t",result[i]);

}

printf("\n");

}

void store_mod(void)

{

int i;

int ret;

int p;

int s;

char n[20];

printf("请输入你要更新菜名的id\n");

scanf("%d", &i);

printf("请输入你要更改的信息price, stock\n");

scanf("%d%d", &p, &s);

while(getchar() != '\n');

printf("请输入要改的名字\n");

gets(n);

sprintf(sql, "update menu set name = '%s', price = %d, stock = %d where id = %d", n, p, s, i);

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

printf("更改成功\n");

}

void menu_query(void)

{

int ret;

int row;

int colum;

int i;

char **result;

sprintf(sql, "select * from menu");

ret = sqlite3_get_table(db_menu, sql, &result, &row, &colum, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

for(i=0 ; i

if(0 == i%4){

printf("\n");

}

printf("%s\t",result[i]);

}

printf("\n");

}

void menu_add(void)

{

int ret;

int i;

char n[20];

int p;

int s = 99;

printf("请给菜的id,price输入\n");

scanf("%d%d", &i, &p);

while(getchar() != '\n');

printf("请给菜取名\n");

gets(n);

sprintf(sql, "insert into menu(id, name, price, stock) values(%d, '%s', %d, %d)", i, n, p, s);

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

printf("保存成功\n");

}

void menu_del(void)

{

int id;

int ret;

int i;

printf("请输入要删除菜名的id\n");

scanf("%d",&i);

sprintf(sql, "delete from menu where id = %d", i);

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

printf("删除成功\n");

}

int main(void)

{

int tem;

int ret;

int fd;

int i;

int j;

int k;

int w = 0;

int d = 0;

int h = 0;

int obj = 0;

fd_set read_fd;

char buf[BUFSIZ];

int row;

int colum;

char **result = NULL;

pserver head;

pserver p;

pserver new;

pserver q;

int bind_fd;

struct sockaddr_in sock_add;

int maxfd;

tem = sizeof(sock_add);

pthread_t tid_write;

pthread_attr_t attr;

/*建立2个数据库,分别为menu与bdo*/

sqlite3_open("./menu.db", &db_menu);

sprintf(sql, "create table menu(id int, name varchar[20], price int, stock int)");

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

}

sprintf(sql, "create table bdo(id varchar[4],name varchar[20], price varchar[4])");

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

}

/*创建一个分离线程*/

pthread_attr_init(&attr);

pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED);

ret = pthread_create(&tid_write, &attr, (void *)sock_write, NULL);

if(ret < 0){

perror("pthread_create");

exit(1);

}

/*链表的建立与给对应的位赋值*/

head = (pserver )malloc(sizeof(server));

if(NULL == head){

list_for_each_entry(p, &head->list, list);

perror("malloc");

exit(1);

}

INIT_LIST_HEAD(&head->list);

for(i=0; i<50; i++){

//链表建立

new = (pserver )malloc(sizeof(server));

if(NULL == new){

perror("malloc");

exit(1);

}

new->data.site = i+1;

new->data.sign = 0;

new->data.sock_fd = 0;

list_add(&new->list, &head->list);

}

#if 0

struct list_head *pos, *n;

pserver tmp;

list_for_each_safe(pos, n, &(head->list)){

tmp = list_entry(pos, server, list);

printf("data.site:%d\t", tmp->data.sign);

}

printf("\n");

#endif

/*网络初始化与绑定ip、端口*/

bzero(&sock_add, tem);

sock_add.sin_family = AF_INET;

sock_add.sin_port = htons(10000);

sock_add.sin_addr.s_addr = inet_addr("192.168.0.245");

bind_fd = socket(PF_INET, SOCK_STREAM, 0);

if(bind_fd < 0){

perror("socket");

exit(1);

}

int on = 1;

setsockopt(bind_fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));

ret = bind(bind_fd, (struct sockaddr *)&sock_add, tem);

if(ret < 0){

perror("bind");

exit(1);

}

maxfd = bind_fd;

ret = listen(bind_fd, 5);

if(ret < 0){

perror("listen");

exit(1);

}

/*多路复用*/

while(1){

FD_ZERO(&read_fd);

FD_SET(bind_fd, &read_fd);

bzero(buf, BUFSIZ);

p = head;

list_for_each_entry(p, &head->list, list){

//设置select

if(p->data.sign != 0){

FD_SET(p->data.sock_fd, &read_fd);

if(maxfd < p->data.sock_fd){

maxfd = p->data.sock_fd;

}

}

}

ret = select(maxfd+1, &read_fd, NULL, NULL, NULL);

if(ret < 0){

perror("select");

exit(1);

}

else if(FD_ISSET(bind_fd, &read_fd)){

fd = accept(bind_fd, NULL, NULL);

if(p->data.sock_fd < 0){

perror("accept");

exit(1);

}

p = head;

list_for_each_entry(p, &head->list, list){

if(0 == p->data.sign){

p->data.sock_fd = fd;

p->data.sign = 1;

break;

}

}

}

p = head;

list_for_each_entry(p, &head->list, list){

//找对应的fd

if(FD_ISSET(p->data.sock_fd, &read_fd)){

bzero(buf, BUFSIZ);

ret = read(p->data.sock_fd, buf, BUFSIZ);

if(ret < 0){

perror("read");

exit(1);

}else if(0 == ret){

//处理客户端关闭

i = p->data.site;

close(p->data.sock_fd);

bzero(&p->data, sizeof(p->data));

p->data.site = i;

}else{

if(0 == strncmp(buf, "order", 5)){//点菜

bzero(buf, BUFSIZ);

obj = 0;

sprintf(sql, "select * from menu where stock > 0");

ret = sqlite3_get_table(db_menu, sql, &result, &row, &colum, &errmsg);

if(SQLITE_OK != ret){

printf("insert:%s\n", errmsg);

exit(1);

}

for(i=0 ; i

if((i%4 == 0)&& (i != 0)){

sprintf((buf+obj), "%s", "\n");

obj += 1;

sprintf((buf + obj), "%s", result[i]);

obj += strlen(result[i]);

}else{

if(0 == i){

sprintf((buf + obj), "%s", result[i]);

obj += strlen(result[i]);

}else{

sprintf((buf+obj), "%s","\t");

obj += 1;

sprintf((buf + obj), "%s", result[i]);

obj += strlen(result[i]);

}

}

}

write(p->data.sock_fd, buf, strlen(buf));

}else if(0 == strncmp(buf, "shout", 5)){//呼叫服务员

printf("\t第%d桌:呼叫服务员\n", p->data.site);

strcpy(buf, "请稍等");

write(p->data.sock_fd, buf, strlen(buf));

}else if(0 == strncmp(buf, "check", 5)){//买单

bzero(buf, BUFSIZ);

strcpy(buf, (char *)&p->data.total);

write(p->data.sock_fd, buf, strlen(buf));

p->data.sign = 0;

p->data.total = 0;

}else if(0 == strncmp(buf, "place", 5)){//下单

bzero(buf, BUFSIZ);

read(p->data.sock_fd, buf, BUFSIZ);

for(j=0; j

//一次打印一个菜的信息

p->data.id[j] = (int )*(buf + j);

if(p->data.id[j] != 0){

sprintf(sql, "select * from menu where id = %d and stock > 0", p->data.id[j]);

result = NULL;

ret = sqlite3_get_table(db_menu, sql, &result, &row, &colum, &errmsg);

if(SQLITE_OK != ret){

printf("insert1:%s\n", errmsg);

exit(1);

}

if((0 == row)){

//没有该id的菜

bzero(buf, BUFSIZ);

strcpy(buf, "没有您要点的菜的id=");

sprintf(buf+strlen(buf), "%d\n", p->data.id[j]);

write(p->data.sock_fd, buf, strlen(buf));

break;

}

for(k=0 ; k

if(k%colum == 0){

printf("\n");

}

printf("%s\t", result[k]);

}

printf("\n");

k = atoi(result[7]);//转化菜的库存

sprintf(sql, "update menu set stock = %d where id = %d", --k, p->data.id[j]);

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);//把所点的菜的库存减1

if(SQLITE_OK != ret){

printf("insert2:%s\n", errmsg);

exit(1);

}

sprintf(sql, "insert into bdo(id, name, price) values('%s', '%s', '%s')", result[4], result[5], result[6]);

ret = sqlite3_exec(db_menu, sql, NULL, NULL, &errmsg);

//把所点的菜的信息放到营运数据库中

if(SQLITE_OK != ret){

printf("insert3:%s\n", errmsg);

exit(1);

}

h = atoi(result[6]);//转化菜的价格

w += h;//把菜的价格相加

}else{

break;//当菜的id为0时,返回

}

}//for()

p->data.total += w;//把菜的价格放入该桌信息链表total中

printf("第%d号桌\t合计:%d\n", p->data.site, p->data.total);

w = 0;

if(0 == strncmp(buf, "没有", 2)){//没有改菜的id时返回

break;

}else{//找到菜的信息时,表示下单成功

bzero(buf, BUFSIZ);

strcat(buf, "订单成功!");

write(p->data.sock_fd, buf, strlen(buf));

}

}//elseif

}//if

}//else

}//list_for

}//while

sqlite3_close(db_menu);

return 0;

}

客户端代码:

#include #include #include #include #include #include #include #include #include #include int sock_fd; char buf[BUFSIZ]; void call_waiter(void); void check_out(void); void place_orader(void); void order_dishes(void); int main(void) { int i; int ret; struct sockaddr_in sock_add; bzero(&sock_add, sizeof(sock_add)); sock_add.sin_family = AF_INET; sock_add.sin_port = htons(10000); sock_add.sin_addr.s_addr = inet_addr("192.168.0.245"); sock_fd = socket(PF_INET, SOCK_STREAM, 0); if(sock_fd < 0){ perror("socket"); exit(1); } ret = connect(sock_fd, (struct sockaddr *)&sock_add, sizeof(sock_add)); if(ret < 0){ perror("connect"); exit(1); } while(1){ printf("***********************************\n"); printf("\t1:->\t点菜\n"); printf("\t2:->\t下单\n"); printf("\t3:->\t结账\n"); printf("\t4:->\t呼叫服务员\n"); printf("***********************************\n"); do{ ret = scanf("%d", &i); while(getchar() != '\n'); }while(ret != 1); switch(i){ case 1: order_dishes();//点菜 break; case 2: place_orader();//下单 break; case 3: check_out();//结账 exit(0); case 4: call_waiter();//呼叫服务员 break; } } return;  } void order_dishes(void)//点菜 { int ret=0; bzero(buf, BUFSIZ); strcpy(buf, "order"); write(sock_fd, buf, strlen(buf)); bzero(buf, BUFSIZ); ret = read(sock_fd, buf, BUFSIZ); printf("%s\n", buf); } void place_orader(void)//下单 { char bufl[BUFSIZ]; int id[100]; int i = 0; int j; printf("请在点完菜输入1001\n"); do{ scanf("%d",&id[i]); i++; }while(id[i-1] != 1001); bzero(bufl, BUFSIZ); strcpy(bufl, "place"); write(sock_fd, bufl, strlen(bufl)); sleep(1); bzero(buf, BUFSIZ); for(j=0; j

酒店服务员程序变量c语言,酒店点餐系统c语言版相关推荐

  1. c语言作业系统输出超限,C语言网Online Judge系统支持语言和编译说明

    Online Judge系统支持语言和编译情况: 语言 编译器 语言版本 编译参数 C gcc 4.6.3 C99 gcc Main.c -o Main -Wall -lm –static -std= ...

  2. c语言p1-melepeo,学生选课系统c语言程序代码

    学生选课系统c语言程序代码 (25页) 本资源提供全文预览,点击全文预览即可全文预览,如果喜欢文档就下载吧,查找使用更方便哦! 14.90 积分 #include #includestruct cou ...

  3. 麦当劳点餐系统C语言程序,麦当劳点餐系统:要求和实现.doc

    麦当劳点餐系统:要求和实现 SEMII-Java作业:麦当劳点餐系统要求和实现 第一部分:需求概述 麦当劳是世界上领先的食品服务零售商,它为全世界提供最受欢迎的食品,如汉堡和披萨等.本项目创建一个用于 ...

  4. c语言static. volatile,嵌入式系统C语言重点语法const、volatile、static、堆栈等的意义及用法...

    原标题:嵌入式系统C语言重点语法const.volatile.static.堆栈等的意义及用法 在单片机应用中,堆栈是个特殊的存储区,主要功能是暂时存放数据和地址,通常用来保护断点和现场. 要点:堆, ...

  5. 嵌入式C语言编程课件,嵌入式系统C语言编程基础PPT课件

    <嵌入式系统C语言编程基础PPT课件>由会员分享,可在线阅读,更多相关<嵌入式系统C语言编程基础PPT课件(81页珍藏版)>请在人人文库网上搜索. 1.嵌入式系统C语言编程基础 ...

  6. c语言程序设计工资纳税系统,C语言程序设计纳税工资系统

    C语言程序设计纳税工资系统 第一章 课程设计目的和要求1.1 课程设计的目的C 语言一直是程序设计语言的主流之一数据类型丰富.齐全.C 语言供了整数.实数.字符.字符串等基本数据类型,还提供数组.指针 ...

  7. linux用c语言模拟抢票系统,C语言可以编写抢票软件吗

    看见网上抢票软件都没有用C语言编写的,看不懂不知道怎么修改,是不是C语言不能编译啊? 解决方案 2 可以!骑自行车也可以到罗马的! 3 不好写. 还是用脚本之类的语言写吧. 例如python. 3 C ...

  8. 万维c语言作业,万维考试系统-C语言题库.doc

    万维考试系统-C语言题库.doc 第 1 题 (10.0 分) 题号390 功能编写函数求 1100 中奇数的平方和. 结果为 166650.000000. 答案 float s0; int i; f ...

  9. c语言程序设计航空查询系统,c语言航班信息查询系统实验报告.doc

    c语言航班信息查询系统实验报告 PAGE 13 - 软件学院大作业任务书 题 目:航班信息查询系统 专 业: 班 级: 姓 名: 学 号: 完成人数: 起讫日期: 任课教师: 职称: 讲师 部分管主任 ...

最新文章

  1. 待解决的问题--用DOS命令删除远程主机系统文件的方法
  2. BIOS中断相关资料和应用
  3. 由歌词引发的模式思考之下篇(模拟Spring的BeanFactory)
  4. tomcat学习笔记1
  5. html1怎样插入视频,HTML视频教程,第1章 HTML初识
  6. 宁波计算机软件再好的大学是,浙江这些实力较强的大学,分数会不会虚高?
  7. 《无线通信基础》笔记
  8. 将class文件反编译成java文件-(纯实际操作)
  9. MATLAB中uigetfile函数使用方法
  10. c语言作业朱鸣华,C语言程序设计习题解析与上机指导 第3版
  11. win10计算机出现乱码,win10系统出现汉字乱码如何解决
  12. 优思学院|六西格玛:如何最有效地制定目标?
  13. 系统激活成功仍显示水印,取消激活方法
  14. mac虚拟摄像头开发
  15. 解决使用mp4v2封装的mp4文件在Wowza的hls上无法播放问题
  16. html 全屏撒花的效果,微信里怎么实现撒花效果?
  17. 直方图处理(规定化)
  18. AMS1117的输入电压范围
  19. 使用HttpClient模拟POST请求
  20. 程序员如何进入人工智能和大数据领域

热门文章

  1. 线上水果店如何运营?生鲜小程序制作 水果店小程序制作 生鲜行业引爆客源
  2. 2023最新SSM计算机毕业设计选题大全(附源码+LW)之java机场网上订票系统00rk3
  3. js 删除数组中指定元素
  4. 导致服务器“中毒”的几种行为
  5. python基础--循环--模拟骰子
  6. 北大教授饶毅500余字毕业典礼致全文
  7. 《南方周末》真厉害!
  8. 李飞飞计算机视觉课CS231n第一天
  9. [系统安全] 二十七.WannaCry勒索病毒分析 (3)蠕虫传播机制解析及IDA和OD逆向
  10. the foundry mari4.6中文版