Android 应用跳帧,一段时间后崩溃

Android App Skips Frames, crashes after a while

我之前发布过关于我的应用程序跳帧的帖子,因为我可能没有进行后台活动和正确更新 UI,这里有一个 link 问题: Doing background stuff on a different UI

我使用了 AsyncTask、Handlers、Timers、Threads...我仍然遇到应用跳帧的相同错误。

这是我的完整代码:

public class MainActivity extends Activity {
    public static final String DEBUG_TAG = "MainActvity";
    public int mInterval = 5000; // 10 seconds by default, can be changed here, it should be aligned
                                 // with how many secs in takes to get latency packets
    private NetworkInformation dsNetwork;

    private TextView wifiTextView;
    private TextView ipTextView;
    private TextView macTextView;
    private TextView ssidTextView;
    private TextView linkSpeedTextView;
    private TextView frequencyTextView;
    private TextView signalLevelTextView;
    private TextView dlBandwidthTextView;
    private TextView upBandwidthTextView;
    private TextView latencyTextView;

    private Handler handler;
    //private Runnable runnableCode;
    //private Timer timerAsync;
    //private TimerTask timerTaskAsync;


    @Override
    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        wifiTextView = (TextView)findViewById(R.id.wifi_textView);
        ipTextView = (TextView)findViewById(R.id.ip_val_textView);
        macTextView = (TextView)findViewById(R.id.mac_val_textView);
        ssidTextView = (TextView)findViewById(R.id.ssid_val_textView);
        linkSpeedTextView = (TextView)findViewById(R.id.linkSpeed_val_textView);
        frequencyTextView = (TextView)findViewById(R.id.frequency_val_textView);
        signalLevelTextView = (TextView)findViewById(R.id.signalLevel_val_textView);
        dlBandwidthTextView = (TextView)findViewById(R.id.dlBandwidth_val_textView);
        upBandwidthTextView = (TextView)findViewById(R.id.upBandwidth_val_textView);
        latencyTextView = (TextView)findViewById(R.id.latency_val_textView);

        handler = new Handler();
        startRepeatingTask();
    }


    Runnable mStatusChecker = new Runnable() {
        @Override
        public void run() {
            updateUI(dsNetwork); //this function can change value of mInterval.
            handler.postDelayed(mStatusChecker, mInterval);
        }
    };

    void startRepeatingTask() {
        dsNetwork = new NetworkInformation(getApplicationContext());
        mStatusChecker.run();
    }

    void stopRepeatingTask() {
        handler.removeCallbacks(mStatusChecker);
    }



    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }



    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }



    private void updateUI(NetworkInformation dsNetwork){
        // Handles updating the textviews in the UI
        //Log.d(DEBUG_TAG, "updateUI(NetworkInformation)");

        if (dsNetwork.isConnected()){
            wifiTextView.setText(R.string.wifi_is_on);
            wifiTextView.setTextColor(Color.GREEN);

            ipTextView.setText(dsNetwork.getIpAddress());
            macTextView.setText(dsNetwork.getMacAddress());
            ssidTextView.setText(dsNetwork.getSsid());
            linkSpeedTextView.setText(String.valueOf(dsNetwork.getLinkSpeed()) + " Mbps");
            frequencyTextView.setText(String.valueOf(dsNetwork.getFrequency()) + " MHz");
            signalLevelTextView.setText(String.valueOf(dsNetwork.getSignalLevel()) + "%");
            dlBandwidthTextView.setText(String.valueOf(dsNetwork.getDlBandwidth()/1000) + " Mbps");
            upBandwidthTextView.setText(String.valueOf(dsNetwork.getDlBandwidth()/1000) + " Mbps");
            latencyTextView.setText(String.valueOf(dsNetwork.getLatency("10.10.0.62")) + " ms");
        }
        else {
            wifiTextView.setText(R.string.wifi_is_off);
            wifiTextView.setTextColor(Color.RED);

            ipTextView.setText("N/A");
            macTextView.setText("N/A");
            ssidTextView.setText("N/A");
            linkSpeedTextView.setText("N/A");
            frequencyTextView.setText("N/A");
            signalLevelTextView.setText("N/A");
            dlBandwidthTextView.setText("N/A");
            upBandwidthTextView.setText("N/A");
            latencyTextView.setText("N/A");
        }

    }

}

这是处理获取信息的网络 class:

public class NetworkInformation {


    public static int SIGNAL_LEVEL_LIMIT = 101;
    public Context mContext;
    public static String DEBUG_TAG = "NetworkInformation";


    private boolean isConnected;
    private String networkType; //Wifi, Mobile Data, Ethernet...
    private String ssid;
    private String ipAddress;
    private String macAddress;
    private int frequency;      // in MHz
    private int linkSpeed;      // in Mbps
    private int signalLevel;    // Measured from 1 to SIGNAL_LEVEL_LIMIT
    private int dlBandwidth;    // downstream bandwidth in Kbps
    private int upBandwidth;    // upstream bandwidth in Kbps


    public boolean isConnected() {
        return isConnected;
    }

    public String getNetworkType() {
        return networkType;
    }

    public String getSsid() {
        return ssid;
    }

    public int getFrequency() {
        return frequency;
    }

    public String getIpAddress() {
        return ipAddress;
    }

    public String getMacAddress() {
        return macAddress;
    }

    public int getLinkSpeed() {
        return linkSpeed;
    }

    public int getSignalLevel() {
        return signalLevel;
    }

    public int getDlBandwidth() {
        return dlBandwidth;
    }

    public int getUpBandwidth() {
        return upBandwidth;
    }




    // Constructor: gets application contex, sets whether or not there is connection, and sets the
    // connection-related variables
    public NetworkInformation(Context mContext) {

        this.mContext = mContext;
        setConnectionStatus();
        setNetworkType();

        if (this.networkType == "None") {
            // TODO: handle the case when there's no connection
        }
        if (this.networkType == "Wifi") {
            setWifiInfo();
        }
        if(this.networkType == "Mobile Data") {
            //TODO: setMobileDataInfo()
        }
    }



    /*
    Returns the latency to a given server in mili-seconds by issuing a ping command.
    system will issue 5 ICMP Echo Request packet each having size of 56 bytes
    every second, and returns the avg latency of them.
    Returns 0 when there is no connection
     */
    public double getLatency(String ipAddress){
        String pingCommand = "/system/bin/ping -c 4 " + ipAddress;
        String inputLine = "";
        double avgRtt = 0;

        try {
            // execute the command on the environment interface
            Process process = Runtime.getRuntime().exec(pingCommand);
            // gets the input stream to get the output of the executed command
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()));

            inputLine = bufferedReader.readLine();
            while ((inputLine != null)) {
                if (inputLine.length() > 0 && inputLine.contains("avg")) {  // when we get to the last line of executed ping command
                    break;
                }
                inputLine = bufferedReader.readLine();
            }
        }
        catch (IOException e){
            Log.v(DEBUG_TAG, "getLatency: EXCEPTION");
            e.printStackTrace();
        }

        // Extracting the average round trip time from the inputLine string
        String afterEqual = inputLine.substring(inputLine.indexOf("="), inputLine.length()).trim();
        String afterFirstSlash = afterEqual.substring(afterEqual.indexOf('/') + 1, afterEqual.length()).trim();
        String strAvgRtt = afterFirstSlash.substring(0, afterFirstSlash.indexOf('/'));
        avgRtt = Double.valueOf(strAvgRtt);

        return avgRtt;
    }






    /*
    The following are private utility functions
     */


    // Checks if the device is connected to the Internet, and sets the class variables
    private void setConnectionStatus() {
        ConnectivityManager cm = (ConnectivityManager)mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkInfo activeNetwork = cm.getActiveNetworkInfo();

        // if there is internet connection, set the variabel
        if ((activeNetwork != null) && (activeNetwork.isConnectedOrConnecting()))
            this.isConnected = true;
        else
            this.isConnected = false;
    }



    // Returns the type of network: Internet, Mobile Data, WiMax...
    private void setNetworkType() {
        ConnectivityManager cm = (ConnectivityManager)mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkInfo activeNetwork = cm.getActiveNetworkInfo();

        String networkType = "None";

        switch (activeNetwork.getType()) {
            case ConnectivityManager.TYPE_WIFI:
                networkType = "Wifi";
                break;
            case ConnectivityManager.TYPE_MOBILE:
                networkType = "Mobile Data";
                break;
            case ConnectivityManager.TYPE_ETHERNET:
                networkType = "Ethernet";
                break;
            case ConnectivityManager.TYPE_WIMAX:
                networkType = "WiMAX";
                break;
            case ConnectivityManager.TYPE_VPN:
                networkType = "VPN";
                break;
        }

        this.networkType = networkType;
    }



    // Gets the wifi information, and updates the class variables
    private void setWifiInfo() {

        ConnectivityManager cm = (ConnectivityManager)mContext.getSystemService(Context.CONNECTIVITY_SERVICE);
        Network[] network = cm.getAllNetworks();
        NetworkCapabilities netCapab = cm.getNetworkCapabilities(network[0]);

        WifiManager wifiManager = (WifiManager)mContext.getSystemService(Context.WIFI_SERVICE);
        WifiInfo wifiInfo = wifiManager.getConnectionInfo();

        if (wifiInfo != null) {
            this.ssid = wifiInfo.getSSID();
            this.ipAddress = Formatter.formatIpAddress(wifiInfo.getIpAddress());
            this.macAddress = wifiInfo.getMacAddress();
            this.linkSpeed = wifiInfo.getLinkSpeed();
            this.frequency = wifiInfo.getFrequency();
            this.dlBandwidth = netCapab.getLinkDownstreamBandwidthKbps();
            this.upBandwidth = netCapab.getLinkUpstreamBandwidthKbps();
            this.signalLevel = wifiManager.calculateSignalLevel(wifiManager.getConnectionInfo().getRssi(), SIGNAL_LEVEL_LIMIT);
        }
    }
}

应用一开始运行的时候还好,后来越来越慢,10分钟左右就崩溃了... 我一直在 logcat 中收到此错误,它不断出现: I/Choreographer﹕跳过183帧!应用程序可能在其主线程上做了太多工作。

哦,我是这样处理的:Repeat a task with a time delay? 也是这样: Update TextView Every Second

但是 none 起作用了。请有人救救我...

方法 getLatency(...) 是一个很长的 运行 操作(您启动外部进程并执行一些 I/O)。这种东西不应该在你正在更新的主线程中调用 UI。 Choreographer 试图保持 60fps,因此阻塞主线程 3 秒会导致 180 帧丢失。

启动一些后台任务,获取所有你需要的数据,然后在主线程中更新UI。