深入理解計(jì)算機(jī)操作系統(tǒng)第二章家庭作業(yè)

2.55 編譯運(yùn)行下列代碼茫蛹,并且確定機(jī)器字節(jié)順序拴袭。

#include <stdio.h>

typedef unsigned char *byte_pointer;

void show_bytes(byte_pointer start, size_t len) {  // size_t == unsigned 8bit integer
    size_t i;
    for (i = 0; i < len; ++i) {
        printf("%.2x ", start[i]);
    }
    printf("\n");
}

void show_int(int x) {
    show_bytes((byte_pointer) &x, sizeof(int));
}

void show_float(float x) {
    show_bytes((byte_pointer) &x, sizeof(float));
}

void show_pointer(void *x) {
    printf("x's address value is [%p] \n", x);
    show_bytes((byte_pointer) &x, sizeof(void *));
}

void test_show_bytes(int val) {
    int ival = val;
    float fval = (float) ival;
    int *pval = &ival;
    show_int(ival);
    show_float(fval);
    show_pointer(pval);
}

int main(void) {
    int test_number = 1234;
    test_show_bytes(test_number); // 1234 == 100 1101 0010 == 00 00 04 d2(大端) == d2 04 00 00 (小端)
    return 0;
}

我的機(jī)器是小端順序,intel 基本上都是小端胶哲。

為什么 byte_pointer 使用 unsigned char * (而不是 char *)?

#include <stdio.h>

int main()
{
    char c = 129;// 1000 0001      11111111 1000 0001
    char unsigned uc = 129;
    printf("%d\n", c);
    printf("%x\n", c);
    printf("%d\n", uc);
    printf("%x\n", uc);
    return 1;
}

/* 輸出
-127
ffffff81
129
81
 * /

2.56 試著用不同示例值運(yùn)行上面的代碼。

同上潭辈。

2.57 同上面一樣去編寫程序 show_short, show_long, show_double 并運(yùn)行鸯屿。

#include <stdio.h>

typedef unsigned char *byte_pointer;

void show_bytes(byte_pointer start, size_t len) {
    size_t i;
    for (i = 0; i < len; ++i) {
        printf("%.2x ", start[i]);
    }
    printf("\n\n");
}

void show_short(short x) {
    show_bytes((byte_pointer) &x, sizeof(short));
}

void show_long(long x) {
    show_bytes((byte_pointer) &x, sizeof(long));
}

void show_double(double x) {
    show_bytes((byte_pointer) &x, sizeof(double));
}

void test_show_bytes(int ival) {
    short sval = (short) ival;
    long lval = (long) ival;
    double dval = (double) ival;
    show_short(sval);
    show_long(lval);
    show_double(dval);
}

int main(void) {
    test_show_bytes(0x12345678);
    return 0;
}
78 56

78 56 34 12

00 00 00 78 56 34 b2 41

2.58 編寫 is_little_endian 函數(shù)判斷大端還是小端,返回 0 和 1把敢。

#include <stdio.h>

int main() {
    int i = 65538; // 00000000 00000001 00000000 00000010
    unsigned char *j = (unsigned char *) &i;

    if (*j == 2) {
        printf("1");
        return 1;   // little endian
    }
    printf("0");
    return 0;
}

2.59 編寫程序返回 x 的最低有效字節(jié)寄摆,和 y 中剩下的字節(jié)。

#include <stdio.h>

int main() {
    int i = 65538; // 0x10002
    int j = 12345; // 0x3039
    printf("%x", (i & 0xff) | (j & ~0xff));
    return 1;
}

2.60 將字節(jié)從 0 開始編號(hào)修赞,寫出函數(shù)實(shí)現(xiàn)將參數(shù) x 的字節(jié) i 被替換成字節(jié) b婶恼。

#include <stdio.h>
#include <assert.h>

int replace_byte(unsigned x, unsigned char i, unsigned char b) {
    return ((-1 - (0xff << (i  << 3))) & x) | (b<<(i<<3));
}

int main() {
    unsigned rep_0 = replace_byte(0x12345678, 0, 0xAB);
    unsigned rep_3 = replace_byte(0x12345678, 3, 0xAB);

    assert(rep_0 == 0x123456AB);
    assert(rep_3 == 0xAB345678);
    return 1;
}

2.61

#include <stdio.h>

int get_msb(int x) {
    return ((!x)
            | (!(~x))
            | (!((~(x & 0xff)) & 0xff)) 
            | (!(x >> ((sizeof(x) - 1)<<3 ))));  
}

int main() {
    printf("%d", get_msb(0xff));
    printf("%d", get_msb(0));
    printf("%d", get_msb(0x11ff));
    printf("%d", get_msb(0xff00));
    printf("%d", get_msb(0xffff00));
    return 1;
}

2.62

#include <stdio.h>

int main() {
    int a = -1;
    int res = !(~(a >> 1));
    printf("%d", res);
    return 1;
}

2.63

#include <stdio.h>

unsigned srl(unsigned x, int k) {
    unsigned xsra = (int) x >> k;
    return (~(-1 << ((sizeof(int) << 3) - k))) & xsra;
}

int main() {
    unsigned a = 0xffffffff;
    unsigned v = srl(a, 3);
    printf("%x", v);
    return 1;
}
#include <stdio.h>

unsigned sra(int x, int k) {
    unsigned xsrl = (unsigned) x >> k;
    int bits = sizeof(int) << 3;
    return (-!!(1 << (bits - 1) & x)) << (bits - k) | xsrl;
}

int main() {
    unsigned a = 0xffffffff;
    unsigned v = sra(a, 15);
    printf("%x", v);
    return 1;
}

2.64

#include <stdio.h>

int any_odd_one(unsigned x) {
    return !!(x & 0xaaaaaaaa);
}

int main() {
    unsigned a = 0x5;
    int b = any_odd_one(a);
    printf("%d", b);
    return 1;
}

2.55

#include <stdio.h>
#include <assert.h>

int odd_ones(unsigned x) {
    x ^= x >> 16;
    x ^= x >> 8;
    x ^= x >> 4;
    x ^= x >> 2;
    x ^= x >> 1;
    x &= 0x1;
    return x;
}

int main(int argc, char *argv[]) {
    assert(odd_ones(0x10101011));
    assert(!odd_ones(0x01010101));
    return 0;
}

2.66

#include <stdio.h>
#include <assert.h>

unsigned left_most_one(unsigned x) {
    x |= x >> 1;
    x |= x >> 2;
    x |= x >> 4;
    x |= x >> 8;
    x |= x >> 16;
    return (x + 1) >> 1;
}

int main() {
    assert(left_most_one(0xff00) == 0x8000);
    assert(left_most_one(0x6600) == 0x4000);
    return 1;
}

2.67

  1. c11 標(biāo)準(zhǔn)指出:如果右移的位數(shù)為負(fù)數(shù),或者左移的位數(shù)大于等于數(shù)值寬度柏副,則結(jié)果在標(biāo)準(zhǔn)中未定義勾邦。
  2. 和 3.
#include <stdio.h>
#include <assert.h>

int int_size_is_32() {
    int set_msb = 1 << 31;
    int beyond_msb = set_msb << 1;

    return set_msb && !beyond_msb;
}

int int_size_is_32_for_16bit() {
    int set_msb = 1 << 15 << 15 << 1;
    int beyond_msb = set_msb << 1;

    return set_msb && !beyond_msb;
}

int main(int argc, char *argv[]) {
    assert(int_size_is_32());
    assert(int_size_is_32_for_16bit());
    return 0;
}

2.68

#include <stdio.h>
#include <assert.h>

int lower_one_mask(int n) {
    return ~(-1 << n);
}

int main() {
    assert(lower_one_mask(6) == 0x3f);
    assert(lower_one_mask(17) == 0x1ffff);
    return 1;
}

2.69

#include <stdio.h>
#include <assert.h>

unsigned rotate_left(unsigned x, int n) {
    return (x << n) | (x >> ((sizeof(x) << 3) - n));
// 這么寫不對(duì),考慮 n = w 的情況割择,左移位數(shù)等于數(shù)值寬度眷篇,在 c 標(biāo)準(zhǔn)中未規(guī)范
// return (x << (n-1)<<1) | (x >> ((sizeof(x) << 3) - n));
}

int main() {
    assert(rotate_left(0x12345678, 4) == 0x23456781);
    assert(rotate_left(0x12345678, 20) == 0x67812345);
    assert(rotate_left(0x12345678, 0) == 0x12345678);
    return 1;
}

2.70

#include <stdio.h>
#include <assert.h>

int fits_bits(int x, int n) {
    int spare = (sizeof(int) << 3) - n;
    return x == (x << spare >> spare);
}

int main(int argc, char *argv[]) {
    assert(!fits_bits(0xFF, 8));
    assert(!fits_bits(~0xFF, 8));

    assert(fits_bits(0b0010, 3));
    assert(!fits_bits(0b1010, 3));
    assert(!fits_bits(0b0110, 3));

    assert(fits_bits(~0b11, 3));
    assert(!fits_bits(~0b01000011, 3));
    assert(!fits_bits(~0b111, 3));
    return 0;
}

2.71

書本答案未曾考慮到算術(shù)右移。

#include <stdio.h>
#include <assert.h>

typedef unsigned packed_t;

int xbyte(packed_t word, int byteNum) {
    packed_t a = (word >> (bytenum << 3)) & 0xff;
    return -(a >> 7) << 8 | a;
}

int main(int argc, char *argv[]) {
    assert(0x0000000f == xbyte(0x00000f00, 1));
    assert(0xffffffff == xbyte(0x0000ff00, 1));
    assert(xbyte(0xAABBCCDD, 1) == 0xFFFFFFCC);
    assert(xbyte(0x00112233, 2) == 0x11);
    return 0;
}

2.72

#include <stdio.h>
#include <assert.h>
#include <string.h>
#include <stdlib.h>

void copy_int(int val, void *buf, int maxbytes) {
    if (maxbytes >= (int) sizeof(val)) {
        memcpy(buf, (void *) &val, sizeof(val));
    }
}

int main() {
    int a = 0x12345678;
    unsigned maxbytes = sizeof(int);
    void *buf = malloc(maxbytes);

    copy_int(a, buf, maxbytes);
    assert(0x12345678 == *(int *) buf);
    int b = 0xffffffff;
    copy_int(b, buf, 0);
    assert(0x12345678 == *(int *) buf);
    assert(0xffffffff != *(int *) buf);

    return 1;
}

2.73

#include <stdio.h>
#include <assert.h>
#include <limits.h>

int saturating_add(int x, int y) {
    int sum = x + y;
    int min = INT_MIN;
    int neg_over = (min & x) && (min & y) && !(min & sum);
    int poa_over = !(min & x) && !(min & y) && (min & sum);
    (negOver && (sum = INT_MIN)) || (posOver && (sum = INT_MAX));
    return sum;
}

int main(int argc, char *argv[]) {
    assert(INT_MAX == saturating_add(INT_MAX, 0x1234));
    assert(INT_MIN == saturating_add(INT_MIN, -0x1234));
    assert(0x11 + 0x22 == saturating_add(0x11, 0x22));
    return 0;
}

2.74

#include <stdio.h>
#include <assert.h>
#include <limits.h>

int tsub_ok(int x, int y) {
    int a = (!!(INT_MIN & x)) && (y == -y);
    int b = !((!!(INT_MIN & x)) ^ (!!(INT_MIN & y)));
    return a || b;
}

int main(int argc, char *argv[]) {
    assert(!tsub_ok(1, INT_MIN));
    assert(tsub_ok(1, 3));
    assert(tsub_ok(-1, INT_MIN));
    assert(!tsub_ok(0x00, INT_MIN));
    assert(tsub_ok(0x00, 0x00));
    return 0;
}

2.75

#include <stdio.h>
#include <assert.h>
#include <inttypes.h>
#include <limits.h>

int signed_high_prod(int x, int y) {
    int64_t mul = (int64_t) x * y;
    return mul >> 32;
}

unsigned unsigned_high_prod(unsigned x, unsigned y) {
    int sig_x = x >> 31;
    int sig_y = y >> 31;
    int signed_prod = signed_high_prod(x, y);
    return signed_prod + x * sig_y + y * sig_x;
}

unsigned another_unsigned_high_prod(unsigned x, unsigned y) {
    uint64_t mul = (uint64_t) x * y;
    return mul >> 32;
}

int main(int argc, char *argv[]) {
    unsigned x = 0x12345678;
    unsigned y = 0xffffffff;
    assert(another_unsigned_high_prod(x, y) == unsigned_high_prod(x, y));
    assert(signed_high_prod(x, y) != unsigned_high_prod(x, y));
    return 0;
}

2.76

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>

void *another_calloc(size_t nmemb, size_t size) {
    if (nmemb == 0 || size == 0) {
        return NULL;
    }

    size_t buf_size = nmemb * size;
    if (nmemb == buf_size / size) {
        void *ptr = malloc(buf_size);
        memset(ptr, 0, buf_size);
        return ptr;
    }
    return NULL;
}

int main(int argc, char *argv[]) {
    void *p;
    p = another_calloc(0x1234, 2);
    assert(p != NULL);
    free(p);

    p = another_calloc(SIZE_MAX, 2);
    assert(p == NULL);
    free(p);
    return 0;
}

2.77

#include <stdio.h>
#include <assert.h>

int A(int x) {
    return (x << 4) + x;
}

int B(int x) {
    return x - (x << 3);
}

int C(int x) {
    return (x << 6) - (x << 2);
}

int D(int x) {
    return (x << 4) - (x << 7);
}

int main(int argc, char *argv[]) {
    int x = 0x87654321;
    assert(A(x) == 17 * x);
    assert(B(x) == -7 * x);
    assert(C(x) == 60 * x);
    assert(D(x) == -112 * x);
    return 0;
}

2.78

#include <stdio.h>
#include <assert.h>
#include <limits.h>

int divide_power2(int x, int k) {
    int is_neg = x & INT_MIN;
    (is_neg && (x = x + (1 << k) - 1));
    return x >> k;
}

int main(int argc, char* argv[]) {
    int x = 0x80000007;
    assert(divide_power2(x, 1) == x / 2);
    assert(divide_power2(x, 2) == x / 4);
    return 0;
}

2.79

#include <stdio.h>
#include <limits.h>
#include <assert.h>

int div(int x, int k) {
    int is_neg = x & INT_MIN;
    is_neg && (x = x + (1 << k) - 1);
    return x >> k;
}

int mul3div4(int x) {
    return div((x << 2) - x, 2);
}

int main(int argc, char *argv[]) {
    int x = 0x87654321;
    assert(mul3div4(x) == x * 3 / 4);
    return 0;
}

2.80

#include <stdio.h>
#include <limits.h>
#include <assert.h>

int threeforths(int x) {
    int is_neg = x & INT_MIN;
    int f = x & ~0x3;
    int l = x & 0x3;

    int fd4 = f >> 2;
    int fd4m3 = (fd4 << 1) + fd4;

    int lm3 = (l << 1) + l;
    int bias = (1 << 2) - 1;
    (is_neg && (lm3 += bias));
    int lm3d4 = lm3 >> 2;

    return fd4m3 + lm3d4;
}

int main(int argc, char *argv[]) {
    assert(threeforths(0) == 0);
    assert(threeforths(1) == 0);
    assert(threeforths(2) == 1);
    assert(threeforths(3) == 2);
    assert(threeforths(8) == 6);
    assert(threeforths(9) == 6);
    assert(threeforths(10) == 7);
    assert(threeforths(11) == 8);
    assert(threeforths(12) == 9);

    assert(threeforths(-8) == -6);
    assert(threeforths(-9) == -6);
    assert(threeforths(-10) == -7);
    assert(threeforths(-11) == -8);
    assert(threeforths(-12) == -9);
    return 0;
}

2.81

-1 << k

~(-1 << k) << j

2.82

錯(cuò)荔泳。 x 是 INT_MIN
對(duì)蕉饼。
對(duì)。
對(duì)玛歌。
對(duì)椎椰。

2.83

#include <stdio.h>
#include <limits.h>
#include <assert.h>

int threeforths(int x) {
    int is_neg = x & INT_MIN;
    int f = x & ~0x3;
    int l = x & 0x3;

    int fd4 = f >> 2;
    int fd4m3 = (fd4 << 1) + fd4;

    int lm3 = (l << 1) + l;
    int bias = (1 << 2) - 1;
    (is_neg && (lm3 += bias));
    int lm3d4 = lm3 >> 2;

    return fd4m3 + lm3d4;
}

int main(int argc, char *argv[]) {
    assert(threeforths(0) == 0);
    assert(threeforths(1) == 0);
    assert(threeforths(2) == 1);
    assert(threeforths(3) == 2);
    assert(threeforths(8) == 6);
    assert(threeforths(9) == 6);
    assert(threeforths(10) == 7);
    assert(threeforths(11) == 8);
    assert(threeforths(12) == 9);

    assert(threeforths(-8) == -6);
    assert(threeforths(-9) == -6);
    assert(threeforths(-10) == -7);
    assert(threeforths(-11) == -8);
    assert(threeforths(-12) == -9);
    return 0;
}

2.83

A.
n = 0.yyyyy...
n << k = y.yyyyy... = Y + n
n << k - n = Y
n = Y/(2^k - 1)

B.

(a).
y = 101, Y = 5, k = 3
n = 5/7

(b).
y = 0110, Y = 6, k = 4
n = 2/5

(c).
y = 010011, Y = 19, k = 6
n = 19/63

2.84

#include <stdio.h>
#include <assert.h>

unsigned f2u(float x) {
    return *(unsigned *) &x;
}

int float_le(float x, float y) {
    unsigned ux = f2u(x);
    unsigned uy = f2u(y);

    unsigned sx = ux >> 31;
    unsigned sy = uy >> 31;

    return (ux << 1 == 0 && uy << 1 == 0) || /* both zeros */
           (sx && !sy) ||                         /* x < 0, y >= 0 or x <= 0, y > 0 */
           (!sx && ux <= uy) ||            /* x > 0, y >= 0 or x >= 0, y > 0 */
           (sx && ux >= uy);                /* x < 0, y <= 0 or x <= 0, y < 0 */
}

int main(int argc, char *argv[]) {
    assert(float_le(-0, +0));
    assert(float_le(+0, -0));
    assert(float_le(0, 3));
    assert(float_le(-4, -0));
    assert(float_le(-4, 4));
    return 0;
}

2.85

// 注意題目中并沒有說(shuō)有符號(hào)位
bias = 2^(k - 1) - 1
V = 2^E * M

a. 
7.0 = 0b111.000...
M = 0b1.11
f = 0b0.11
E = 2
e = bias + E
V = 7.0
結(jié)果是 10...1 110...

b.
bias+n 1...

c.
1...101 0...

2.86

0 0...(15) 0 0...(62)1  2^(1-bias-63)
0 0...(14)1 1 0....(63) 2^(1-bias)
0 1...(14)0 1 1...(63)  2^bias * (2-2^-63)

2.87

描述 Hex M E V D
-0 0x8000 0 -14 -0 -0.0
>2 least 0x4001 1025/1024 1 1025/512 2.00195312
512 0x6000 1 9 512 512.0
bigest denormalized 0x03FF 1023/1024 -14 1023/(2^24) 6.09755516e-5
-∞ 0xFC00 - - -∞ -∞
ox3BB0 0x3BB0 123/64 -1 123/128 0.9609375

2.88

A bit A value B bit B value
1 01110 001 -9/16 1 0110 0010 -9/16
0 10110 101 13*2^4 0 1110 1010 13*2^4
1 00111 110 -7/2^10 1 0000 0111 -7/2^10
0 00000 101 5/2^17 0 0000 0001 1/2^10
1 11011 000 -2^12 1 1110 1111 -31*2^3
0 11000 100 3*2^8 0 1111 0000 +oo

2.89

#include <stdio.h>
#include <assert.h>
#include <limits.h>

/*
 * most important thing is that all double number come from ints
 */

/* right */
int A(int x, double dx) {
    return (float) x == (float) dx;
}

/* wrong when y is INT_MIN */
int B(int x, double dx, int y, double dy) {
    return dx - dy == (double) (x - y);
}

/* right */
int C(double dx, double dy, double dz) {
    return (dx + dy) + dz == dx + (dy + dz);
}

/*
 * wrong
 *
 * FIXME I don't know what conditions cause false
 */
int D(double dx, double dy, double dz) {
    return (dx * dy) * dz == dx * (dy * dz);
}

/* wrong when dx != 0 and dz == 0 */
int E(double dx, double dz) {
    return dx / dx == dz / dz;
}

int main(int argc, char *argv[]) {

    int x = 111;
    int y = 222;
    int z = 333;
    double dx = (double) x;
    double dy = (double) y;
    double dz = (double) z;

    printf("%x %x %x\n", x, y, z);

    assert(A(x, dx));   // double 可以表示所有32位整數(shù)
    assert(!B(0, (double) (int) 0, INT_MIN, (double) (int) INT_MIN));    // 溢出問(wèn)題
    assert(C(dx, dy, dz));    // double 可以表示所有32位整數(shù)
    assert(!D((double) (int) 0x64e73387, (double) (int) 0xd31cb264, (double) (int) 0xd22f1fcd)); // double 并不能表示所有的 64位數(shù),導(dǎo)致舍入問(wèn)題
    assert(!E(dx, (double) (int) 0));    // 除數(shù)為零
    return 0;
}

2.90

#include <stdio.h>
#include <assert.h>
#include <math.h>

float u2f(unsigned x) {
  return *(float*) &x;
}

/* 2^x */
float fpwr2(int x) {
  /* Result exponent and fraction */
  unsigned exp, frac;
  unsigned u;

  if (x < 2-pow(2,7)-23) {
    /* too small. return 0.0 */
    exp = 0;
    frac = 0;
  } else if (x < 2-pow(2,7)) {
    /* Denormalized result */
    exp = 0;
    frac = 1 << (unsigned)(x - (2-pow(2,7)-23));
  } else if (x < pow(2,7)-1+1) {
    /* Normalized result */
    exp = pow(2,7)-1+x;
    frac = 0;
  } else {
    /* Too big, return +oo */
    exp = 0xFF;
    frac = 0;
  }

  /* pack exp and frac into 32 bits */
  u = exp << 23 | frac;
  /* Result as float */
  return u2f(u);
}

int main(int argc, char* argv[]) {
  assert(fpwr2(0) == powf(2,0));
  assert(fpwr2(100) == powf(2,100));
  assert(fpwr2(-100) == powf(2,-100));
  assert(fpwr2(10000) == powf(2,10000));
  assert(fpwr2(-10000) == powf(2,-10000));
  return 0;
}

2.91

A.
0b11.0010010000111111011011

B.
0b11.001001(001)...

C.
第九個(gè)

2.92

float_bits float_negate(float_bits f) {
    unsigned sign = f >> 31;
    unsigned exp = f >> 23 & 0xff;
    unsigned frac = f & 0x7fffff;
    if (exp == 0xff && frac) {
        return f;
    }
    return (~sign << 31) | (exp << 23) | frac;
}

2.93

float_bits float_absval(float_bits f) {
    unsigned sig = f >> 31;
    unsigned exp = f >> 23 & 0xFF;
    unsigned frac = f & 0x7FFFFF;

    int is_NAN = (exp == 0xFF) && (frac != 0);
    if (is_NAN) {
        return f;
    }

    return 0 << 31 | exp << 23 | frac;
}

2.94

float_bits float_twice(float_bits f) {
    unsigned sig = f >> 31;
    unsigned exp = f >> 23 & 0xFF;
    unsigned frac = f & 0x7FFFFF;

    int is_NAN_or_oo = (exp == 0xFF);
    if (is_NAN_or_oo) {
        return f;
    }

    if (exp == 0) {
        /* Denormalized */
        frac <<= 1;
    } else if (exp == 0xFF - 1) {
        /* twice to oo */
        exp = 0xFF;
        frac = 0;
    } else {
        /* Normalized */
        exp += 1;
    }

    return sig << 31 | exp << 23 | frac;
}

2.95

float_bits float_half(float_bits f) {
    unsigned sig = f >> 31;
    unsigned rest = f & 0x7FFFFFFF;
    unsigned exp = f >> 23 & 0xFF;
    unsigned frac = f & 0x7FFFFF;

    int is_NAN_or_oo = (exp == 0xFF);
    if (is_NAN_or_oo) {
        return f;
    }

    /*
     * round to even, we care about last 2 bits of frac
     *
     * 00 => 0 just >>1
     * 01 => 0 (round to even) just >>1
     * 10 => 1 just >>1
     * 11 => 1 + 1 (round to even) just >>1 and plus 1
     */
    int addition = (frac & 0x3) == 0x3;

    if (exp == 0) {
        /* Denormalized */
        frac >>= 1;
        frac += addition;
    } else if (exp == 1) {
        /* Normalized to denormalized */
        rest >>= 1;
        rest += addition;
        exp = rest >> 23 & 0xFF;
        frac = rest & 0x7FFFFF;
    } else {
        /* Normalized */
        exp -= 1;
    }

    return sig << 31 | exp << 23 | frac;
}

2.96

int float_f2i(float_bits f) {
    unsigned sig = f >> 31;
    unsigned exp = f >> 23 & 0xFF;
    unsigned frac = f & 0x7FFFFF;
    unsigned bias = 0x7F;

    int num;
    unsigned E;
    unsigned M;

    /*
     * consider positive numbers
     *
     * 0 00000000 00000000000000000000000
     *   ===>
     * 0 01111111 00000000000000000000000
     *   0 <= f < 1
     * get integer 0
     *
     * 0 01111111 00000000000000000000000
     *   ===>
     * 0 (01111111+31) 00000000000000000000000
     *   1 <= f < 2^31
     * integer round to 0
     *
     * 0 (01111111+31) 00000000000000000000000
     *   ===>
     * greater
     *   2^31 <= f < oo
     * return 0x80000000
     */
    if (exp >= 0 && exp < 0 + bias) {
        /* number less than 1 */
        num = 0;
    } else if (exp >= 31 + bias) {
        /* number overflow */
        /* or f < 0 and (int)f == INT_MIN */
        num = 0x80000000;
    } else {
        E = exp - bias;
        M = frac | 0x800000;
        if (E > 23) {
            num = M << (E - 23);
        } else {
            /* whether sig is 1 or 0, round to zero */
            num = M >> (23 - E);
        }
    }

    return sig ? -num : num;
}

2.97

#include <stdio.h>
#include <assert.h>
#include <limits.h>

typedef unsigned float_bits;

float_bits float_i2f(int i) {
    unsigned sign = i & 0x80000000;
    if (i == 0) {
        return sign;
    }
    if (i == 0x80000000) {
        return 0xcf000000;
    }

    unsigned transI = i;
    if (sign) {
        transI = ~i + 1;
    }
    int noSignI = transI & 0x7fffffff;

    int isOne = noSignI & 0x80000000;
    int count = 1;

    while (!isOne) {
        noSignI <<= 1;
        isOne = noSignI & 0x80000000;
        count++;
    }

    int exp = 127 + (32 - count);
    int frac = (transI << count >> 1 + 8) & 0x7fffff;

    int ret = sign | exp << 23 | frac;

    if (32 - count > 23) {
        int mid = (transI << count >> 1 + 8 << 1 + 8) + 0b100000000;
        if (transI << count > mid) {
            ret++;
        } else {
            if (transI << count == mid && frac & 0x1) {
                ret++;
            }
        }
    }
    return ret;
}

int main() {
    unsigned a;
    int b;
    int arr[] = {1, 2, 4, INT_MAX, INT_MIN, 0, -1, -2, 123456789, 2147483582, 2147483584, 2147483585};
    int count = sizeof(arr) / sizeof(int);
    for (int i = 0; i < count; i++) {
        b = arr[i];
        a = float_i2f(b);
        printf("\n %d ==>\t", b);
        printf(" %f ==>\t", (float) b);
        printf(" %f \n", *(float *) (&a));
        assert((float) b == *(float *) (&a));
    }
    return 0;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末沾鳄,一起剝皮案震驚了整個(gè)濱河市慨飘,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖瓤的,帶你破解...
    沈念sama閱讀 212,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件休弃,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡圈膏,警方通過(guò)查閱死者的電腦和手機(jī)塔猾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,755評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)稽坤,“玉大人丈甸,你說(shuō)我怎么就攤上這事∧蛲剩” “怎么了睦擂?”我有些...
    開封第一講書人閱讀 158,369評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)杖玲。 經(jīng)常有香客問(wèn)我顿仇,道長(zhǎng),這世上最難降的妖魔是什么摆马? 我笑而不...
    開封第一講書人閱讀 56,799評(píng)論 1 285
  • 正文 為了忘掉前任臼闻,我火速辦了婚禮,結(jié)果婚禮上囤采,老公的妹妹穿的比我還像新娘述呐。我一直安慰自己,他們只是感情好蕉毯,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,910評(píng)論 6 386
  • 文/花漫 我一把揭開白布市埋。 她就那樣靜靜地躺著,像睡著了一般恕刘。 火紅的嫁衣襯著肌膚如雪缤谎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,096評(píng)論 1 291
  • 那天褐着,我揣著相機(jī)與錄音坷澡,去河邊找鬼。 笑死含蓉,一個(gè)胖子當(dāng)著我的面吹牛频敛,可吹牛的內(nèi)容都是我干的挂捅。 我是一名探鬼主播秀撇,決...
    沈念sama閱讀 39,159評(píng)論 3 411
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼咖刃!你這毒婦竟也來(lái)了差油?” 一聲冷哼從身側(cè)響起拗军,我...
    開封第一講書人閱讀 37,917評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤任洞,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后发侵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體交掏,經(jīng)...
    沈念sama閱讀 44,360評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,673評(píng)論 2 327
  • 正文 我和宋清朗相戀三年刃鳄,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了盅弛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,814評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡叔锐,死狀恐怖挪鹏,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情愉烙,我是刑警寧澤讨盒,帶...
    沈念sama閱讀 34,509評(píng)論 4 334
  • 正文 年R本政府宣布,位于F島的核電站齿梁,受9級(jí)特大地震影響催植,放射性物質(zhì)發(fā)生泄漏肮蛹。R本人自食惡果不足惜勺择,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,156評(píng)論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望伦忠。 院中可真熱鬧省核,春花似錦、人聲如沸昆码。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)赋咽。三九已至旧噪,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間脓匿,已是汗流浹背淘钟。 一陣腳步聲響...
    開封第一講書人閱讀 32,123評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留陪毡,地道東北人米母。 一個(gè)月前我還...
    沈念sama閱讀 46,641評(píng)論 2 362
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像毡琉,于是被迫代替她去往敵國(guó)和親铁瞒。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,728評(píng)論 2 351

推薦閱讀更多精彩內(nèi)容

  • __block和__weak修飾符的區(qū)別其實(shí)是挺明顯的:1.__block不管是ARC還是MRC模式下都可以使用桅滋,...
    LZM輪回閱讀 3,297評(píng)論 0 6
  • 史上最全的iOS面試題及答案 iOS面試小貼士———————————————回答好下面的足夠了----------...
    Style_偉閱讀 2,346評(píng)論 0 35
  • 多線程慧耍、特別是NSOperation 和 GCD 的內(nèi)部原理。運(yùn)行時(shí)機(jī)制的原理和運(yùn)用場(chǎng)景。SDWebImage的原...
    LZM輪回閱讀 2,004評(píng)論 0 12
  • ———————————————回答好下面的足夠了---------------------------------...
    恒愛DE問(wèn)候閱讀 1,713評(píng)論 0 4