最小c编译器

最小c编译器(来源 (最好在linux下操作))代码有好几个版本,我选择otccelfn.c 。

 

/*Obfuscated Tiny C Compiler with ELF outputCopyright (C) 2001-2003 Fabrice BellardThis software is provided 'as-is', without any express or impliedwarranty.  In no event will the authors be held liable for any damagesarising from the use of this software.Permission is granted to anyone to use this software for any purpose,including commercial applications, and to alter it and redistribute itfreely, subject to the following restrictions:1. The origin of this software must not be misrepresented; you must notclaim that you wrote the original software. If you use this softwarein a product, an acknowledgment in the product and its documentation *is* required.2. Altered source versions must be plainly marked as such, and must not bemisrepresented as being the original software.3. This notice may not be removed or altered from any source distribution.
*/
#ifndef TINY
#include <stdarg.h>
#endif
#include <stdio.h>/* vars: value of variables loc : local variable indexglo : global variable ptrdata: base of data segmentind : output code ptrprog: output codersym: return symbolsym_stk: symbol stackdstk: symbol stack pointerdptr, dch: macro state* 'vars' format: For each character TAG_TOK at offset 'i' before asymbol in sym_stk, we have:v = (int *)(vars + 8 * i + TOK_IDENT)[0] p = (int *)(vars + 8 * i + TOK_IDENT)[0] v = 0    : undefined symbol, p = list of use points.v = 1    : define symbol, p = pointer to define text.v < LOCAL: offset on stack, p = 0.otherwise: symbol with value 'v', p = list of use points.* 'sym_stk' format:TAG_TOK sym1 TAG_TOK sym2 .... symN '\0''dstk' points to the last '\0'.
*/
int tok, tokc, tokl, ch, vars, rsym, prog, ind, loc, glo, file, sym_stk, dstk, dptr, dch, last_id, data, text, data_offset;#define ALLOC_SIZE 99999#define ELFOUT/* depends on the init string */
#define TOK_STR_SIZE 48
#define TOK_IDENT    0x100
#define TOK_INT      0x100
#define TOK_IF       0x120
#define TOK_ELSE     0x138
#define TOK_WHILE    0x160
#define TOK_BREAK    0x190
#define TOK_RETURN   0x1c0
#define TOK_FOR      0x1f8
#define TOK_DEFINE   0x218
#define TOK_MAIN     0x250#define TOK_DUMMY   1
#define TOK_NUM     2#define LOCAL   0x200#define SYM_FORWARD 0
#define SYM_DEFINE  1/* tokens in string heap */
#define TAG_TOK    ' '
#define TAG_MACRO  2/* additionnal elf output defines */
#ifdef ELFOUT#define ELF_BASE      0x08048000
#define PHDR_OFFSET   0x30#define INTERP_OFFSET 0x90
#define INTERP_SIZE   0x13#ifndef TINY
#define DYNAMIC_OFFSET (INTERP_OFFSET + INTERP_SIZE + 1)
#define DYNAMIC_SIZE   (11*8)#define ELFSTART_SIZE  (DYNAMIC_OFFSET + DYNAMIC_SIZE)
#else
#define DYNAMIC_OFFSET 0xa4
#define DYNAMIC_SIZE   0x58#define ELFSTART_SIZE  0xfc
#endif/* size of startup code */
#define STARTUP_SIZE   17/* size of library names at the start of the .dynstr section */
#define DYNSTR_BASE      22#endifpdef(t)
{*(char *)dstk++ = t;
}inp()
{if (dptr) {ch = *(char *)dptr++;if (ch == TAG_MACRO) {dptr = 0;ch = dch;}} elsech = fgetc(file);/*    printf("ch=%c 0x%x\n", ch, ch); */
}isid()
{return isalnum(ch) | ch == '_';
}/* read a character constant */
getq()
{if (ch == '\\') {inp();if (ch == 'n')ch = '\n';}
}next()
{int t, l, a;while (isspace(ch) | ch == '#') {if (ch == '#') {inp();next();if (tok == TOK_DEFINE) {next();pdef(TAG_TOK); /* fill last ident tag */*(int *)tok = SYM_DEFINE;*(int *)(tok + 4) = dstk; /* define stack */}/* well we always save the values ! */while (ch != '\n') {pdef(ch);inp();}pdef(ch);pdef(TAG_MACRO);}inp();}tokl = 0;tok = ch;/* encode identifiers & numbers */if (isid()) {pdef(TAG_TOK);last_id = dstk;while (isid()) {pdef(ch);inp();}if (isdigit(tok)) {tokc = strtol(last_id, 0, 0);tok = TOK_NUM;} else {*(char *)dstk = TAG_TOK; /* no need to mark end of string (wesuppose data is initied to zero */tok = strstr(sym_stk, last_id - 1) - sym_stk;*(char *)dstk = 0;   /* mark real end of ident for dlsym() */tok = tok * 8 + TOK_IDENT;if (tok > TOK_DEFINE) {tok = vars + tok;/*        printf("tok=%s %x\n", last_id, tok); *//* define handling */if (*(int *)tok == SYM_DEFINE) {dptr = *(int *)(tok + 4);dch = ch;inp();next();}}}} else {inp();if (tok == '\'') {tok = TOK_NUM;getq();tokc = ch;inp();inp();} else if (tok == '/' & ch == '*') {inp();while (ch) {while (ch != '*')inp();inp();if (ch == '/')ch = 0;}inp();next();} else{t = "++#m--%am*@R<^1c/@%[_[H3c%@%[_[H3c+@.B#d-@%:_^BKd<<Z/03e>>`/03e<=0f>=/f<@.f>@1f==&g!=\'g&&k||#l&@.BCh^@.BSi|@.B+j~@/%Yd!@&d*@b";while (l = *(char *)t++) {a = *(char *)t++;tokc = 0;while ((tokl = *(char *)t++ - 'b') < 0)tokc = tokc * 64 + tokl + 64;if (l == tok & (a == ch | a == '@')) {
#if 0printf("%c%c -> tokl=%d tokc=0x%x\n", l, a, tokl, tokc);
#endifif (a == ch) {inp();tok = TOK_DUMMY; /* dummy token for double tokens */}break;}}}}
#if 0{int p;printf("tok=0x%x ", tok);if (tok >= TOK_IDENT) {printf("'");if (tok > TOK_DEFINE) p = sym_stk + 1 + (tok - vars - TOK_IDENT) / 8;elsep = sym_stk + 1 + (tok - TOK_IDENT) / 8;while (*(char *)p != TAG_TOK && *(char *)p)printf("%c", *(char *)p++);printf("'\n");} else if (tok == TOK_NUM) {printf("%d\n", tokc);} else {printf("'%c'\n", tok);}}
#endif
}#ifdef TINY
#define skip(c) next()
#elsevoid error(char *fmt,...)
{va_list ap;va_start(ap, fmt);fprintf(stderr, "%d: ", ftell((FILE *)file));vfprintf(stderr, fmt, ap);fprintf(stderr, "\n");exit(1);va_end(ap);
}void skip(c)
{if (tok != c) {error("'%c' expected", c);}next();
}#endif/* from 0 to 4 bytes */
o(n)
{/* cannot use unsigned, so we must do a hack */while (n && n != -1) {*(char *)ind++ = n;n = n >> 8;}
}#ifdef ELFOUT/* put a 32 bit little endian word 'n' at unaligned address 't' */
put32(t, n)
{*(char *)t++ = n;*(char *)t++ = n >> 8;*(char *)t++ = n >> 16;*(char *)t++ = n >> 24;
}/* get a 32 bit little endian word at unaligned address 't' */
get32(t)
{int n;return  (*(char *)t & 0xff) |(*(char *)(t + 1) & 0xff) << 8 |(*(char *)(t + 2) & 0xff) << 16 |(*(char *)(t + 3) & 0xff) << 24;
}#else#define put32(t, n) *(int *)t = n
#define get32(t) *(int *)t#endif/* output a symbol and patch all references to it */
gsym1(t, b)
{int n;while (t) {n = get32(t); /* next value *//* patch absolute reference (always mov/lea before) */if (*(char *)(t - 1) == 0x05) {/* XXX: incorrect if data < 0 */if (b >= data && b < glo)put32(t, b + data_offset);elseput32(t, b - prog + text + data_offset);} else {put32(t, b - t - 4);}t = n;}
}gsym(t)
{gsym1(t, ind);
}/* psym is used to put an instruction with a data field which is areference to a symbol. It is in fact the same as oad ! */
#define psym oad/* instruction + address */
oad(n, t)
{o(n);put32(ind, t);t = ind;ind = ind + 4;return t;
}/* load immediate value */
li(t)
{oad(0xb8, t); /* mov $xx, %eax */
}gjmp(t)
{return psym(0xe9, t);
}/* l = 0: je, l == 1: jne */
gtst(l, t)
{o(0x0fc085); /* test %eax, %eax, je/jne xxx */return psym(0x84 + l, t);
}gcmp(t)
{o(0xc139); /* cmp %eax,%ecx */li(0);o(0x0f); /* setxx %al */o(t + 0x90);o(0xc0);
}gmov(l, t)
{int n;o(l + 0x83);n = *(int *)t;if (n && n < LOCAL)oad(0x85, n);else {t = t + 4;*(int *)t = psym(0x05, *(int *)t);}
}/* l is one if '=' parsing wanted (quick hack) */
unary(l)
{int n, t, a, c;n = 1; /* type of expression 0 = forward, 1 = value, other =lvalue */if (tok == '\"') {li(glo + data_offset);while (ch != '\"') {getq();*(char *)glo++ = ch;inp();}*(char *)glo = 0;glo = glo + 4 & -4; /* align heap */inp();next();} else {c = tokl;a = tokc;t = tok;next();if (t == TOK_NUM) {li(a);} else if (c == 2) {/* -, +, !, ~ */unary(0);oad(0xb9, 0); /* movl $0, %ecx */if (t == '!')gcmp(a);elseo(a);} else if (t == '(') {expr();skip(')');} else if (t == '*') {/* parse cast */skip('(');t = tok; /* get type */next(); /* skip int/char/void */next(); /* skip '*' or '(' */if (tok == '*') {/* function type */skip('*');skip(')');skip('(');skip(')');t = 0;}skip(')');unary(0);if (tok == '=') {next();o(0x50); /* push %eax */expr();o(0x59); /* pop %ecx */o(0x0188 + (t == TOK_INT)); /* movl %eax/%al, (%ecx) */} else if (t) {if (t == TOK_INT)o(0x8b); /* mov (%eax), %eax */else o(0xbe0f); /* movsbl (%eax), %eax */ind++; /* add zero in code */}} else if (t == '&') {gmov(10, tok); /* leal EA, %eax */next();} else {n = 0;if (tok == '=' & l) {/* assignment */next();expr();gmov(6, t); /* mov %eax, EA */} else if (tok != '(') {/* variable */gmov(8, t); /* mov EA, %eax */if (tokl == 11) {gmov(0, t);o(tokc);next();}}}}/* function call */if (tok == '(') {if (n)o(0x50); /* push %eax *//* push args and invert order */a = oad(0xec81, 0); /* sub $xxx, %esp */next();l = 0;while(tok != ')') {expr();oad(0x248489, l); /* movl %eax, xxx(%esp) */if (tok == ',')next();l = l + 4;}put32(a, l);next();if (n) {oad(0x2494ff, l); /* call *xxx(%esp) */l = l + 4;} else {/* forward reference */t = t + 4;*(int *)t = psym(0xe8, *(int *)t);}if (l)oad(0xc481, l); /* add $xxx, %esp */}
}sum(l)
{int t, n, a;if (l-- == 1)unary(1);else {sum(l);a = 0;while (l == tokl) {n = tok;t = tokc;next();if (l > 8) {a = gtst(t, a); /* && and || output code generation */sum(l);} else {o(0x50); /* push %eax */sum(l);o(0x59); /* pop %ecx */if (l == 4 | l == 5) {gcmp(t);} else {o(t);if (n == '%')o(0x92); /* xchg %edx, %eax */}}}/* && and || output code generation */if (a && l > 8) {a = gtst(t, a);li(t ^ 1);gjmp(5); /* jmp $ + 5 */gsym(a);li(t);}}
}expr()
{sum(11);
}test_expr()
{expr();return gtst(0, 0);
}block(l)
{int a, n, t;if (tok == TOK_IF) {next();skip('(');a = test_expr();skip(')');block(l);if (tok == TOK_ELSE) {next();n = gjmp(0); /* jmp */gsym(a);block(l);gsym(n); /* patch else jmp */} else {gsym(a); /* patch if test */}} else if (tok == TOK_WHILE | tok == TOK_FOR) {t = tok;next();skip('(');if (t == TOK_WHILE) {n = ind;a = test_expr();} else {if (tok != ';')expr();skip(';');n = ind;a = 0;if (tok != ';')a = test_expr();skip(';');if (tok != ')') {t = gjmp(0);expr();gjmp(n - ind - 5);gsym(t);n = t + 4;}}skip(')');block(&a);gjmp(n - ind - 5); /* jmp */gsym(a);} else if (tok == '{') {next();/* declarations */decl(1);while(tok != '}')block(l);next();} else {if (tok == TOK_RETURN) {next();if (tok != ';')expr();rsym = gjmp(rsym); /* jmp */} else if (tok == TOK_BREAK) {next();*(int *)l = gjmp(*(int *)l);} else if (tok != ';')expr();skip(';');}
}/* 'l' is true if local declarations */
decl(l)
{int a;while (tok == TOK_INT | tok != -1 & !l) {if (tok == TOK_INT) {next();while (tok != ';') {if (l) {loc = loc + 4;*(int *)tok = -loc;} else {*(int *)tok = glo;glo = glo + 4;}next();if (tok == ',') next();}skip(';');} else {/* put function address */*(int *)tok = ind;next();skip('(');a = 8;while (tok != ')') {/* read param name and compute offset */*(int *)tok = a;a = a + 4;next();if (tok == ',')next();}next(); /* skip ')' */rsym = loc = 0;o(0xe58955); /* push   %ebp, mov %esp, %ebp */a = oad(0xec81, 0); /* sub $xxx, %esp */block(0);gsym(rsym);o(0xc3c9); /* leave, ret */put32(a, loc); /* save local variables */}}
}#ifdef ELFOUTgle32(n)
{put32(glo, n);glo = glo + 4;
}/* used to generate a program header at offset 't' of size 's' */
gphdr1(n, t)
{gle32(n);n = n + ELF_BASE;gle32(n);gle32(n);gle32(t);gle32(t);
}elf_reloc(l)
{int t, a, n, p, b, c;p = 0;t = sym_stk;while (1) {/* extract symbol name */t++;a = t;while (*(char *)t != TAG_TOK && t < dstk)t++;if (t == dstk)break;/* now see if it is forward defined */tok = vars + (a - sym_stk) * 8 + TOK_IDENT - 8;b = *(int *)tok;n = *(int *)(tok + 4);if (n && b != 1) {
#if 0{char buf[100];memcpy(buf, a, t - a);buf[t - a] = '\0';printf("extern ref='%s' val=%x\n", buf, b);}
#endifif (!b) {if (!l) {/* symbol string */memcpy(glo, a, t - a);glo = glo + t - a + 1; /* add a zero */} else if (l == 1) {/* symbol table */gle32(p + DYNSTR_BASE);gle32(0);gle32(0);gle32(0x10); /* STB_GLOBAL, STT_NOTYPE */p = p + t - a + 1; /* add a zero */} else {p++;/* generate relocation patches */while (n) {a = get32(n);/* c = 0: R_386_32, c = 1: R_386_PC32 */c = *(char *)(n - 1) != 0x05;put32(n, -c * 4);gle32(n - prog + text + data_offset);gle32(p * 256 + c + 1);n = a;}}} else if (!l) {/* generate standard relocation */gsym1(n, b);}}}
}elf_out(c)
{int glo_saved, dynstr, dynstr_size, dynsym, hash, rel, n, t, text_size;/*****************************//* add text segment (but copy it later to handle relocations) */text = glo;text_size = ind - prog;/* add the startup code */ind = prog;o(0x505458); /* pop %eax, push %esp, push %eax */t = *(int *)(vars + TOK_MAIN);oad(0xe8, t - ind - 5);o(0xc389);  /* movl %eax, %ebx */li(1);      /* mov $1, %eax */o(0x80cd);  /* int $0x80 */glo = glo + text_size;/*****************************//* add symbol strings */dynstr = glo;/* libc name for dynamic table */glo++;glo = strcpy(glo, "libc.so.6") + 10;glo = strcpy(glo, "libdl.so.2") + 11;/* export all forward referenced functions */elf_reloc(0);dynstr_size = glo - dynstr;/*****************************//* add symbol table */glo = (glo + 3) & -4;dynsym = glo;gle32(0);gle32(0);gle32(0);gle32(0);elf_reloc(1);/*****************************//* add symbol hash table */hash = glo;n = (glo - dynsym) / 16;gle32(1); /* one bucket (simpler!) */gle32(n);gle32(1);gle32(0); /* dummy first symbol */t = 2;while (t < n)gle32(t++);gle32(0);/*****************************//* relocation table */rel = glo;elf_reloc(2);/* copy code AFTER relocation is done */memcpy(text, prog, text_size);glo_saved = glo;glo = data;/* elf header */gle32(0x464c457f);gle32(0x00010101);gle32(0);gle32(0);gle32(0x00030002);gle32(1);gle32(text + data_offset); /* address of _start */gle32(PHDR_OFFSET); /* offset of phdr */gle32(0);gle32(0);gle32(0x00200034);gle32(3); /* phdr entry count *//* program headers */gle32(3); /* PT_INTERP */gphdr1(INTERP_OFFSET, INTERP_SIZE);gle32(4); /* PF_R */gle32(1); /* align */gle32(1); /* PT_LOAD */gphdr1(0, glo_saved - data);gle32(7); /* PF_R | PF_X | PF_W */gle32(0x1000); /* align */gle32(2); /* PT_DYNAMIC */gphdr1(DYNAMIC_OFFSET, DYNAMIC_SIZE);gle32(6); /* PF_R | PF_W */gle32(0x4); /* align *//* now the interpreter name */glo = strcpy(glo, "/lib/ld-linux.so.2") + 0x14;/* now the dynamic section */gle32(1); /* DT_NEEDED */gle32(1); /* libc name */gle32(1); /* DT_NEEDED */gle32(11); /* libdl name */gle32(4); /* DT_HASH */gle32(hash + data_offset);gle32(6); /* DT_SYMTAB */gle32(dynsym + data_offset);gle32(5); /* DT_STRTAB */gle32(dynstr + data_offset);gle32(10); /* DT_STRSZ */gle32(dynstr_size);gle32(11); /* DT_SYMENT */gle32(16);gle32(17); /* DT_REL */gle32(rel + data_offset);gle32(18); /* DT_RELSZ */gle32(glo_saved - rel);gle32(19); /* DT_RELENT */gle32(8);gle32(0);  /* DT_NULL */gle32(0);t = fopen(c, "w");fwrite(data, 1, glo_saved - data, t);fclose(t);
}
#endifmain(n, t)
{if (n < 3) {printf("usage: otccelf file.c outfile\n");return 0;}dstk = strcpy(sym_stk = calloc(1, ALLOC_SIZE), " int if else while break return for define main ") + TOK_STR_SIZE;glo = data = calloc(1, ALLOC_SIZE);ind = prog = calloc(1, ALLOC_SIZE);vars = calloc(1, ALLOC_SIZE);t = t + 4;file = fopen(*(int *)t, "r");data_offset = ELF_BASE - data; glo = glo + ELFSTART_SIZE;ind = ind + STARTUP_SIZE;inp();next();decl(0);t = t + 4;elf_out(*(int *)t);return 0;
}

 这段代码实现的编译器是c语言的一个子集,有一个小程序(otccex.c )来说明这个子集(英文说明 ):

 

/* #!/usr/local/bin/otcc */
/** Sample OTCC C example. You can uncomment the first line and install* otcc in /usr/local/bin to make otcc scripts !  *//* Any preprocessor directive except #define are ignored. We put thisinclude so that a standard C compiler can compile this code too. */
#include <stdio.h>/* defines are handled, but macro arguments cannot be given. Norecursive defines are tolerated */
#define DEFAULT_BASE 10/* global variables can be used */
int base;/** Only old style K&R prototypes are parsed. Only int arguments are* allowed (implicit types).* * By benchmarking the execution time of this function (for example* for fib(35)), you'll notice that OTCC is quite fast because it* generates native i386 machine code.  */
fib(n)
{if (n <= 2)return 1;elsereturn fib(n-1) + fib(n-2);
}/* Identifiers are parsed the same way as C: begins with letter or'_', and then letters, '_' or digits */
fact(n)
{/* local variables can be declared. Only 'int' type is supported */int i, r;r = 1;/* 'while' and 'for' loops are supported */for(i=2;i<=n;i++)r = r * i;return r;
}/* Well, we could use printf, but it would be too easy */
print_num(n, b)
{int tab, p, c;/* Numbers can be entered in decimal, hexadecimal ('0x' prefix) andoctal ('0' prefix) *//* more complex programs use malloc */tab = malloc(0x100); p = tab;while (1) {c = n % b;/* Character constants can be used */if (c >= 10)c = c + 'a' - 10;elsec = c + '0';*(char *)p = c;p++;n = n / b;/* 'break' is supported */if (n == 0)break;}while (p != tab) {p--;printf("%c", *(char *)p);}free(tab);
}/* 'main' takes standard 'argc' and 'argv' parameters */
main(argc, argv)
{/* no local name space is supported, but local variables AREsupported. As long as you do not use a globally definedvariable name as local variable (which is a bad habbit), youwon't have any problem */int s, n, f;/* && and || operator have the same semantics as C (left to rightevaluation and early exit) */if (argc != 2 && argc != 3) {/* '*' operator is supported with explicit casting to 'int *','char *' or 'int (*)()' (function pointer). Of course, 'int'are supposed to be used as pointers too. */s = *(int *)argv;help(s);return 1;}/* Any libc function can be used because OTCC uses dynamic linking */n = atoi(*(int *)(argv + 4));base = DEFAULT_BASE;if (argc >= 3) {base = atoi(*(int *)(argv + 8));if (base < 2 || base > 36) {/* external variables can be used too (here: 'stderr') */fprintf(stderr, "Invalid base\n");return 1;}}printf("fib(%d) = ", n);print_num(fib(n), base);printf("\n");printf("fact(%d) = ", n);if (n > 12) {printf("Overflow");} else {/* why not using a function pointer ? */f = &fact;print_num((*(int (*)())f)(n), base);}printf("\n");return 0;
}/* functions can be used before being defined */
help(name)
{printf("usage: %s n [base]\n", name);printf("Compute fib(n) and fact(n) and output the result in base 'base'\n");
}

 可以使用gcc编译otccelfn.c,使用命令:

gcc -O2 otccelfn.c -o otccelfn 

  然后使用命令"otccelfn otccex.c otccex"来编译例子程序。然后修改文件为用户可执行(chmod u+x otccex).

  再使用命令"./otccex 5"可以得到下面输出:

 

fib(5) = 5
fact(5) = 120
 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/545975.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在Java中使用Collat​​or和String类进行字符串比较

Given two strings and we have to compare them using Collator and String classed in Java. 给定两个字符串&#xff0c;我们必须使用Java中分类的Collat​​or和String进行比较。 Using Collator class – to compare two strings, we use compare() method – it returns…

Oracle数据库中表格的级联删除问题

数据库表中没有设置级联删除.怎样用SQL语句实现:如:EMP表中有字段DEPT_NO是外键POS表中有字段DEPT_NO是外键DEPT表中有字段DEPT_NO,如何实现删除DEPT表中数据时将EMP表,POS表中的相关数据也删除;这里有两种方法&#xff1a;方法一&#xff1a;触发器解决create or replace trig…

IDEA 2020.2 重磅发布,动画级新功能预览!

Guide 关注了 IDEA 的官推&#xff0c;平时没事就会去看看有没有啥比较好的更新。今天下午看到IntelliJ IDEA 2020.2 都已经发布并且还支持了 Java15。然后&#xff0c;我就去官网简单看了一下新特性。单看新特性&#xff0c;这个新版本还是有一点香的。虽然我还木有升级到这个…

访问控制模型ACL和RBAC

2019独角兽企业重金招聘Python工程师标准>>> 1.ACL ACL是最早也是最基本的一种访问控制机制&#xff0c;它的原理非常简单&#xff1a;每一项资源&#xff0c;都配有一个列表&#xff0c;这个列表记录的就是哪些用户可以对这项资源执行CRUD中的那些操作。当系统试图…

最常见并发面试题整理!(速度收藏)

前言并发编程是面试中必问的知识点之一&#xff0c;所以本文整理了一些最为常见的并发面试题&#xff0c;一起来看吧~1. synchronized的实现原理以及锁优化&#xff1f;synchronized的实现原理synchronized作用于「方法」或者「代码块」&#xff0c;保证被修饰的代码在同一时间…

JavaScript中的嵌套事件处理(在鼠标移动事件上)

Multiple event handling is the secret ingredient of dynamic WebPages seen now-a-days. 多重事件处理是当今动态网页的秘密组成部分。 Now, let’s get started... 现在&#xff0c;让我们开始吧... Example Code 范例程式码 <html lang"en"><head&…

在Oracle中CHAR,NCHAR,VARCHAR,VARCHAR2,NVARCHAR2这五种类型的区别

【在Oracle中CHAR,NCHAR,VARCHAR,VARCHAR2,NVARCHAR2这五种类型的区别 】1.CHAR(size)和VARCHAR(size)的区别 CHAR为定长的字段&#xff0c;最大长度为2K字节&#xff1b; VARCHAR为可变长的字段&#xff0c;最大长度为4K字节&#xff1b; 2.CHAR(size)和NCHAR(size)的区别 CHA…

Android 照相功能

使用内置的Camera 应用程序捕获图像 探索Android 所提供的内置功能&#xff0c;内置的图像捕获与存储功能为Android 上全部媒体功能提供了一个非常好的切入点&#xff0c;为我们在以后的章节中处理音频和视频奠定了基础。如今介绍怎样利用内置的Camera&#xff08;摄像头&#…

皮尔逊相关性_皮尔逊的相关性及其在机器学习中的意义

皮尔逊相关性Today we would be using a statistical concept i.e. Pearsons correlation to help us understand the relationships between the feature values (independent values) and the target value (dependent value or the value to be predicted ) which will furt…

磊哥最近面试了好多人,聊聊我的感受!(附面试知识点)

这是我的第 84 篇原创文章作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;一些读者可能知道&#xff0c;磊哥前段时间又回来上班了&#xff0c;详见《磊哥又滚回职场了...》…

M4 宏处理器

2019独角兽企业重金招聘Python工程师标准>>> M4 宏处理器 Brian W. KernighanDennis M. Ritchie Bell LaboratoriesMurray Hill, New Jersey 07974 翻译&#xff1a;寒蝉退士 译者声明&#xff1a;译者对译文不做任何担保&#xff0c;译者对译文不拥有任何权利并且不…

oracle的nvl和nvl2

NVL( string1, replace_with) 功能&#xff1a;如果string1为NULL&#xff0c;则NVL函数返回replace_with的值&#xff0c;否则返回string1的值&#xff0c;如果两个参数都为NULL &#xff0c;则返回NULL。NVL2(expr1,expr2,expr3)功能&#xff1a;oracle中常用函数&#xff0c…

Java SecurityManager checkAwtEventQueueAccess()方法与示例

SecurityManager类的checkAwtEventQueueAccess()方法 (SecurityManager Class checkAwtEventQueueAccess() method) checkAwtEventQueueAccess() method is available in java.lang package. checkAwtEventQueueAccess()方法在java.lang包中可用。 checkAwtEventQueueAccess() …

绝了,几款主流的 JSON 库性能对比!

本篇通过JMH&#xff08;Oracle官方测试框架&#xff09;来测试一下Java中几种常见的JSON解析库的性能。每次都在网上看到别人说什么某某库性能是如何如何的好&#xff0c;碾压其他的库。但是百闻不如一见&#xff0c;只有自己亲手测试过的才是最值得相信的。JSON不管是在Web开…

DWZ使用笔记

DWZ使用笔记 一、前言 在近期的一个项目中&#xff0c;引入了DWZ这个富client框架&#xff0c;算是一次尝试吧。期间也遇到不少问题&#xff0c;总算一一攻克了。特以此文记之。本人用的是dwz-ria-1.4.5Asp.net webform&#xff0c;写这篇笔记时最新版本号已经是1.4.6了。DWZ官…

Dynamic_Performance_Tables_not_accessible_问题_解决不能动态统计

Dynamic Performance Tables not accessible, Automatic Statistics Disabled for this session You can disable statistics in the preference menu,or obtanin select priviliges on the v$session,v$sesstat and v$statname tables第一种处理方法&#xff08;不推荐&#x…

ruby三元操作符_在Ruby中使用操作符将元素添加到数组实例中

ruby三元操作符In the previous articles, we have gone through ways through which we can create Array instances. Some of them were Public instance methods and some were Public class methods. We should also know how they both differ from each other. Now we kn…

阿里的简历多久可以投递一次?次数多了有没有影响?可以同时进行吗?

最近&#xff0c;无论是读者群&#xff0c;还是公众号后台&#xff0c;很多人都比较关注以下几个问题&#xff1a;阿里的简历是半年只能投递一次吗&#xff1f;阿里的面试可以多个部门同时进行吗&#xff1f;面试没过&#xff0c;又被系统捞起来了&#xff0c;我该怎么办&#…

c语言给定一个非空整数数组_C程序检查给定整数的所有位是否为一(1)

c语言给定一个非空整数数组Problem statement: Write a C Program to check if all the bits of a given integer is one (1). 问题陈述&#xff1a;编写一个C程序来检查给定整数的所有位是否都是一(1) 。 Solution: We can use bitwise operator here to solve the problem. …

记一次蚂蚁金服面试被虐经历

本文来自作者投稿&#xff0c;原作者&#xff1a;yes面试前的小姐姐来说说前不久蚂蚁金服一面的情况。说来也是巧合&#xff0c;当时在群里有位蚂蚁金服的小姐姐发了个内推&#xff0c;看了下JD感觉可以试试于是就私聊了小姐姐发简历内推了。我16年也就是大三上就开始实习了&am…